os/ossrv/ofdbus/dbus-glib/dbus/dbus-binding-tool-glib.c
changeset 0 bde4ae8d615e
     1.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     1.2 +++ b/os/ossrv/ofdbus/dbus-glib/dbus/dbus-binding-tool-glib.c	Fri Jun 15 03:10:57 2012 +0200
     1.3 @@ -0,0 +1,1640 @@
     1.4 +/* -*- mode: C; c-file-style: "gnu" -*- */
     1.5 +/* dbus-binding-tool-glib.c: Output C glue
     1.6 + *
     1.7 + * Copyright (C) 2003, 2004, 2005 Red Hat, Inc.
     1.8 + * Copyright (C) 2005 Nokia
     1.9 + * Portion Copyright © 2008 Nokia Corporation and/or its subsidiary(-ies). All rights reserved.
    1.10 + * Licensed under the Academic Free License version 2.1
    1.11 + *
    1.12 + * This program is free software; you can redistribute it and/or modify
    1.13 + * it under the terms of the GNU General Public License as published by
    1.14 + * the Free Software Foundation; either version 2 of the License, or
    1.15 + * (at your option) any later version.
    1.16 + *
    1.17 + * This program is distributed in the hope that it will be useful,
    1.18 + * but WITHOUT ANY WARRANTY; without even the implied warranty of
    1.19 + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    1.20 + * GNU General Public License for more details.
    1.21 + *
    1.22 + * You should have received a copy of the GNU General Public License
    1.23 + * along with this program; if not, write to the Free Software
    1.24 + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
    1.25 + *
    1.26 + */
    1.27 +
    1.28 +#ifndef __SYMBIAN32__
    1.29 +#include <config.h>
    1.30 +#else
    1.31 +#include "config.h"
    1.32 +#endif //__SYMBIAN32__
    1.33 +#include "dbus/dbus-glib.h"
    1.34 +#include "dbus-gidl.h"
    1.35 +#include "dbus-gparser.h"
    1.36 +#include "dbus-gutils.h"
    1.37 +#include "dbus-gtype-specialized.h"
    1.38 +#include "dbus-gsignature.h"
    1.39 +#include "dbus-gvalue-utils.h"
    1.40 +#include "dbus-glib-tool.h"
    1.41 +#include "dbus-binding-tool-glib.h"
    1.42 +
    1.43 +#ifndef __SYMBIAN32__
    1.44 +#include <glib/gi18n.h>
    1.45 +#include <libintl.h>
    1.46 +#define _(x) dgettext (GETTEXT_PACKAGE, x)
    1.47 +#define N_(x) x
    1.48 +#else
    1.49 +
    1.50 +#define _(x) x
    1.51 +#define N_(x) x
    1.52 +#endif
    1.53 +
    1.54 +#include <stdio.h>
    1.55 +#include <stdlib.h>
    1.56 +#include <string.h>
    1.57 +#include <unistd.h>
    1.58 +
    1.59 +#define MARSHAL_PREFIX "dbus_glib_marshal_"
    1.60 +
    1.61 +typedef struct
    1.62 +{
    1.63 +  gboolean ignore_unsupported;
    1.64 +  const char* prefix;
    1.65 +  GIOChannel *channel;
    1.66 +  
    1.67 +  GError **error;
    1.68 +  
    1.69 +  GHashTable *generated;
    1.70 +  GString *blob;
    1.71 +  GString *signal_blob;
    1.72 +  GString *property_blob;
    1.73 +  guint count;
    1.74 +} DBusBindingToolCData;
    1.75 +
    1.76 +static gboolean gather_marshallers (BaseInfo *base, DBusBindingToolCData *data, GError **error);
    1.77 +static gboolean generate_glue (BaseInfo *base, DBusBindingToolCData *data, GError **error);
    1.78 +static gboolean generate_client_glue (BaseInfo *base, DBusBindingToolCData *data, GError **error);
    1.79 +
    1.80 +static const char *
    1.81 +dbus_g_type_get_marshal_name (GType gtype)
    1.82 +{
    1.83 +  switch (G_TYPE_FUNDAMENTAL (gtype))
    1.84 +    {
    1.85 +    case G_TYPE_NONE:
    1.86 +      return "NONE";
    1.87 +    case G_TYPE_BOOLEAN:
    1.88 +      return "BOOLEAN";
    1.89 +    case G_TYPE_UCHAR:
    1.90 +      return "UCHAR";
    1.91 +    case G_TYPE_INT:
    1.92 +      return "INT";
    1.93 +    case G_TYPE_UINT:
    1.94 +      return "UINT";
    1.95 +    case G_TYPE_INT64:
    1.96 +      return "INT64";
    1.97 +    case G_TYPE_UINT64:
    1.98 +      return "UINT64";
    1.99 +    case G_TYPE_DOUBLE:
   1.100 +      return "DOUBLE";
   1.101 +    case G_TYPE_STRING:
   1.102 +      return "STRING";
   1.103 +    case G_TYPE_POINTER:
   1.104 +      return "POINTER";
   1.105 +    case G_TYPE_BOXED:
   1.106 +      return "BOXED";
   1.107 +    case G_TYPE_OBJECT:
   1.108 +      return "OBJECT";
   1.109 +    default:
   1.110 +      return NULL;
   1.111 +    }
   1.112 +}
   1.113 +
   1.114 +/* This entire function is kind of...ugh. */
   1.115 +static const char *
   1.116 +dbus_g_type_get_c_name (GType gtype)
   1.117 +{
   1.118 +  GType subtype;
   1.119 +  if (dbus_g_type_is_struct (gtype))
   1.120 +    {
   1.121 +      return "GValueArray";
   1.122 +    }
   1.123 +  if (dbus_g_type_is_collection (gtype))
   1.124 +    {
   1.125 +      subtype = dbus_g_type_get_collection_specialization(gtype);
   1.126 +      if (_dbus_g_type_is_fixed (subtype))
   1.127 +        return "GArray";
   1.128 +      else
   1.129 +        return "GPtrArray";
   1.130 +    }
   1.131 +
   1.132 +  if (dbus_g_type_is_map (gtype))
   1.133 +    return "GHashTable";
   1.134 +
   1.135 +  if (g_type_is_a (gtype, G_TYPE_STRING))
   1.136 +    return "char *";
   1.137 +
   1.138 +  /* This one is even more hacky...we get an extra *
   1.139 +   * because G_TYPE_STRV is a G_TYPE_BOXED
   1.140 +   */
   1.141 +  if (g_type_is_a (gtype, G_TYPE_STRV))
   1.142 +    return "char *";
   1.143 +
   1.144 +  if (g_type_is_a (gtype, DBUS_TYPE_G_OBJECT_PATH))
   1.145 +    return "char";
   1.146 +
   1.147 +  return g_type_name (gtype);
   1.148 +}
   1.149 +
   1.150 +static gboolean
   1.151 +compute_gsignature (MethodInfo *method, GType *rettype, GArray **params, GError **error)
   1.152 +{
   1.153 +  GSList *elt;
   1.154 +  GType retval_type;
   1.155 +  GArray *ret;
   1.156 +  gboolean is_async;
   1.157 +  const char *arg_type;
   1.158 +  gboolean retval_signals_error;
   1.159 +  
   1.160 +  is_async = method_info_get_annotation (method, DBUS_GLIB_ANNOTATION_ASYNC) != NULL;
   1.161 +  retval_signals_error = FALSE;
   1.162 +
   1.163 +  ret = g_array_new (TRUE, TRUE, sizeof (GType));
   1.164 +
   1.165 +  if (is_async)
   1.166 +    retval_type = G_TYPE_NONE;
   1.167 +  else
   1.168 +    {
   1.169 +      gboolean found_retval;
   1.170 +
   1.171 +      /* Look for return value */
   1.172 +      found_retval = FALSE;
   1.173 +      for (elt = method_info_get_args (method); elt; elt = elt->next)
   1.174 +	{
   1.175 +	  ArgInfo *arg = elt->data;
   1.176 +	  const char *returnval_annotation;
   1.177 +      
   1.178 +	  returnval_annotation = arg_info_get_annotation (arg, DBUS_GLIB_ANNOTATION_RETURNVAL);
   1.179 +	  if (returnval_annotation != NULL)
   1.180 +	    {
   1.181 +	      arg_type = arg_info_get_type (arg);
   1.182 +	      retval_type = _dbus_gtype_from_signature (arg_type, FALSE);
   1.183 +	      if (retval_type == G_TYPE_INVALID)
   1.184 +		goto invalid_type;
   1.185 +	      found_retval = TRUE;
   1.186 +	      if (!strcmp (returnval_annotation, "error"))
   1.187 +		retval_signals_error = TRUE;
   1.188 +	      break;
   1.189 +	    }
   1.190 +	}
   1.191 +      if (!found_retval)
   1.192 +	{
   1.193 +	  retval_type = G_TYPE_BOOLEAN;
   1.194 +	  retval_signals_error = TRUE;
   1.195 +	}
   1.196 +    }
   1.197 +
   1.198 +  *rettype = retval_type;
   1.199 +
   1.200 +  /* Handle all input arguments */
   1.201 +  for (elt = method_info_get_args (method); elt; elt = elt->next)
   1.202 +    {
   1.203 +      ArgInfo *arg = elt->data;
   1.204 +      if (arg_info_get_direction (arg) == ARG_IN)
   1.205 +	{
   1.206 +	  GType gtype;
   1.207 +	  
   1.208 +	  arg_type = arg_info_get_type (arg);
   1.209 +	  gtype = _dbus_gtype_from_signature (arg_type, FALSE);
   1.210 +	  if (gtype == G_TYPE_INVALID)
   1.211 +	    goto invalid_type;
   1.212 +	  
   1.213 +	  g_array_append_val (ret, gtype);
   1.214 +	}
   1.215 +    }
   1.216 +
   1.217 +  if (!is_async)
   1.218 +    {
   1.219 +      /* Append pointer for each out arg storage */
   1.220 +      for (elt = method_info_get_args (method); elt; elt = elt->next)
   1.221 +	{
   1.222 +	  ArgInfo *arg = elt->data;
   1.223 +
   1.224 +	  /* Skip return value */
   1.225 +	  if (arg_info_get_annotation (arg, DBUS_GLIB_ANNOTATION_RETURNVAL) != NULL)
   1.226 +	    continue;
   1.227 +      
   1.228 +	  if (arg_info_get_direction (arg) == ARG_OUT)
   1.229 +	    {
   1.230 +	      GType gtype;
   1.231 +	      arg_type = arg_info_get_type (arg);
   1.232 +	      gtype = _dbus_gtype_from_signature (arg_type, FALSE);
   1.233 +	      if (gtype == G_TYPE_INVALID)
   1.234 +		goto invalid_type;
   1.235 +	      /* We actually just need a pointer for the return value
   1.236 +		 storage */
   1.237 +	      gtype = G_TYPE_POINTER;
   1.238 +	      g_array_append_val (ret, gtype);
   1.239 +	    }
   1.240 +	}
   1.241 +
   1.242 +      if (retval_signals_error)
   1.243 +	{
   1.244 +	  /* Final GError parameter */
   1.245 +	  GType gtype = G_TYPE_POINTER;
   1.246 +	  g_array_append_val (ret, gtype);
   1.247 +	}
   1.248 +    }
   1.249 +  else
   1.250 +    {
   1.251 +      /* Context pointer */
   1.252 +      GType gtype = G_TYPE_POINTER;
   1.253 +      g_array_append_val (ret, gtype);
   1.254 +    }
   1.255 +
   1.256 +  *params = ret;
   1.257 +  return TRUE;
   1.258 +
   1.259 + invalid_type:
   1.260 +  g_set_error (error,
   1.261 +	       DBUS_BINDING_TOOL_ERROR,
   1.262 +	       DBUS_BINDING_TOOL_ERROR_UNSUPPORTED_CONVERSION,
   1.263 +	       _("Unsupported conversion from D-BUS type %s to glib-genmarshal type"),
   1.264 +	       arg_type);
   1.265 +  return FALSE;
   1.266 +}
   1.267 +  
   1.268 +
   1.269 +static char *
   1.270 +compute_marshaller (MethodInfo *method, GError **error)
   1.271 +{
   1.272 +  GArray *signature;
   1.273 +  GType rettype;
   1.274 +  const char *marshal_name;
   1.275 +  GString *ret;
   1.276 +  guint i;
   1.277 +
   1.278 +  if (!compute_gsignature (method, &rettype, &signature, error))
   1.279 +    return NULL;
   1.280 +
   1.281 +  ret = g_string_new ("");
   1.282 +  marshal_name = dbus_g_type_get_marshal_name (rettype);
   1.283 +  g_assert (marshal_name != NULL);
   1.284 +  g_string_append (ret, marshal_name);
   1.285 +  g_string_append_c (ret, ':');
   1.286 +  for (i = 0; i < signature->len; i++)
   1.287 +    {
   1.288 +      marshal_name = dbus_g_type_get_marshal_name (g_array_index (signature, GType, i));
   1.289 +      g_assert (marshal_name != NULL);
   1.290 +      g_string_append (ret, marshal_name);
   1.291 +      if (i < signature->len - 1)
   1.292 +	g_string_append_c (ret, ',');
   1.293 +    }
   1.294 +  if (signature->len == 0)
   1.295 +    {
   1.296 +      marshal_name = dbus_g_type_get_marshal_name (G_TYPE_NONE);
   1.297 +      g_assert (marshal_name != NULL);
   1.298 +      g_string_append (ret, marshal_name);
   1.299 +    }
   1.300 +  g_array_free (signature, TRUE);
   1.301 +  return g_string_free (ret, FALSE);
   1.302 +}
   1.303 +
   1.304 +static char *
   1.305 +compute_marshaller_name (MethodInfo *method, const char *prefix, GError **error)
   1.306 +{
   1.307 +  GString *ret;
   1.308 +  GArray *signature;
   1.309 +  GType rettype;
   1.310 +  const char *marshal_name;
   1.311 +  guint i;
   1.312 +
   1.313 +  if (!compute_gsignature (method, &rettype, &signature, error))
   1.314 +    return NULL;
   1.315 +
   1.316 +  ret = g_string_new (MARSHAL_PREFIX);
   1.317 +  g_string_append (ret, prefix);
   1.318 +  g_string_append_c (ret, '_');
   1.319 +
   1.320 +  marshal_name = dbus_g_type_get_marshal_name (rettype);
   1.321 +  g_assert (marshal_name != NULL);
   1.322 +  g_string_append (ret, marshal_name);
   1.323 +  g_string_append (ret, "__");
   1.324 +  for (i = 0; i < signature->len; i++)
   1.325 +    {
   1.326 +      marshal_name = dbus_g_type_get_marshal_name (g_array_index (signature, GType, i));
   1.327 +      g_assert (marshal_name != NULL);
   1.328 +      g_string_append (ret, marshal_name);
   1.329 +      if (i < signature->len - 1)
   1.330 +	g_string_append_c (ret, '_');
   1.331 +    }
   1.332 +  if (signature->len == 0)
   1.333 +    {
   1.334 +      marshal_name = dbus_g_type_get_marshal_name (G_TYPE_NONE);
   1.335 +      g_assert (marshal_name != NULL);
   1.336 +      g_string_append (ret, marshal_name);
   1.337 +    }
   1.338 +  g_array_free (signature, TRUE);
   1.339 +  return g_string_free (ret, FALSE);
   1.340 +}
   1.341 +
   1.342 +static gboolean
   1.343 +gather_marshallers_list (GSList *list, DBusBindingToolCData *data, GError **error)
   1.344 +{
   1.345 +  GSList *tmp;
   1.346 +
   1.347 +  tmp = list;
   1.348 +  while (tmp != NULL)
   1.349 +    {
   1.350 +      if (!gather_marshallers (tmp->data, data, error))
   1.351 +	return FALSE;
   1.352 +      tmp = tmp->next;
   1.353 +    }
   1.354 +  return TRUE;
   1.355 +}
   1.356 +
   1.357 +static gboolean
   1.358 +gather_marshallers (BaseInfo *base, DBusBindingToolCData *data, GError **error)
   1.359 +{
   1.360 +  if (base_info_get_type (base) == INFO_TYPE_NODE)
   1.361 +    {
   1.362 +      if (!gather_marshallers_list (node_info_get_nodes ((NodeInfo *) base),
   1.363 +				    data, error))
   1.364 +	return FALSE;
   1.365 +      if (!gather_marshallers_list (node_info_get_interfaces ((NodeInfo *) base),
   1.366 +				    data, error))
   1.367 +	return FALSE;
   1.368 +    }
   1.369 +  else
   1.370 +    {
   1.371 +      InterfaceInfo *interface;
   1.372 +      GSList *methods;
   1.373 +      GSList *tmp;
   1.374 +      const char *interface_c_name;
   1.375 +
   1.376 +      interface = (InterfaceInfo *) base;
   1.377 +      interface_c_name = interface_info_get_annotation (interface, DBUS_GLIB_ANNOTATION_C_SYMBOL);
   1.378 +      if (interface_c_name == NULL)
   1.379 +        {
   1.380 +	  if (!data->prefix)
   1.381 +	    return TRUE;
   1.382 +        }
   1.383 +
   1.384 +      methods = interface_info_get_methods (interface);
   1.385 +
   1.386 +      /* Generate the necessary marshallers for the methods. */
   1.387 +
   1.388 +      for (tmp = methods; tmp != NULL; tmp = g_slist_next (tmp))
   1.389 +        {
   1.390 +          MethodInfo *method;
   1.391 +          char *marshaller_name;
   1.392 +
   1.393 +          method = (MethodInfo *) tmp->data;
   1.394 +
   1.395 +          marshaller_name = compute_marshaller (method, error);
   1.396 +	  if (!marshaller_name)
   1.397 +	    return FALSE;
   1.398 +
   1.399 +	  if (g_hash_table_lookup (data->generated, marshaller_name))
   1.400 +	    {
   1.401 +	      g_free (marshaller_name);
   1.402 +	      continue;
   1.403 +	    }
   1.404 +
   1.405 +	  g_hash_table_insert (data->generated, marshaller_name, NULL);
   1.406 +        }
   1.407 +
   1.408 +    }
   1.409 +  return TRUE;
   1.410 +}
   1.411 +
   1.412 +static gboolean
   1.413 +generate_glue_list (GSList *list, DBusBindingToolCData *data, GError **error)
   1.414 +{
   1.415 +  GSList *tmp;
   1.416 +
   1.417 +  tmp = list;
   1.418 +  while (tmp != NULL)
   1.419 +    {
   1.420 +      if (!generate_glue (tmp->data, data, error))
   1.421 +	return FALSE;
   1.422 +      tmp = tmp->next;
   1.423 +    }
   1.424 +  return TRUE;
   1.425 +}
   1.426 +
   1.427 +#define WRITE_OR_LOSE(x) do { gsize bytes_written; if (!g_io_channel_write_chars (channel, x, -1, &bytes_written, error)) goto io_lose; } while (0)
   1.428 +
   1.429 +static gboolean
   1.430 +write_printf_to_iochannel (const char *fmt, GIOChannel *channel, GError **error, ...)
   1.431 +{
   1.432 +  char *str;
   1.433 +  va_list args;
   1.434 +  GIOStatus status;
   1.435 +  gsize written;
   1.436 +  gboolean ret;
   1.437 +
   1.438 +  va_start (args, error);
   1.439 +
   1.440 +  str = g_strdup_vprintf (fmt, args);
   1.441 +  if ((status = g_io_channel_write_chars (channel, str, -1, &written, error)) == G_IO_STATUS_NORMAL)
   1.442 +    ret = TRUE;
   1.443 +  else
   1.444 +    ret = FALSE;
   1.445 +
   1.446 +  g_free (str);
   1.447 +
   1.448 +  va_end (args);
   1.449 +
   1.450 +  return ret;
   1.451 +}
   1.452 +
   1.453 +static gboolean
   1.454 +write_quoted_string (GIOChannel *channel, GString *string, GError **error)
   1.455 +{
   1.456 +  guint i;
   1.457 +
   1.458 +  WRITE_OR_LOSE ("\"");
   1.459 +  for (i = 0; i < string->len; i++)
   1.460 +    {
   1.461 +      if (string->str[i] != '\0')
   1.462 +	{
   1.463 +	  if (!g_io_channel_write_chars (channel, string->str + i, 1, NULL, error))
   1.464 +	    return FALSE;
   1.465 +	}
   1.466 +      else
   1.467 +	{
   1.468 +	  if (!g_io_channel_write_chars (channel, "\\0", -1, NULL, error))
   1.469 +	    return FALSE;
   1.470 +	}
   1.471 +    }
   1.472 +  WRITE_OR_LOSE ("\\0\"");
   1.473 +  return TRUE;
   1.474 + io_lose:
   1.475 +  return FALSE;
   1.476 +}
   1.477 +
   1.478 +static gboolean
   1.479 +generate_glue (BaseInfo *base, DBusBindingToolCData *data, GError **error)
   1.480 +{
   1.481 +  if (base_info_get_type (base) == INFO_TYPE_NODE)
   1.482 +    {
   1.483 +      GString *object_introspection_data_blob;
   1.484 +      GIOChannel *channel;
   1.485 +
   1.486 +      channel = data->channel;
   1.487 +      
   1.488 +      object_introspection_data_blob = g_string_new_len ("", 0);
   1.489 +      
   1.490 +      data->blob = object_introspection_data_blob;
   1.491 +      data->count = 0;
   1.492 +
   1.493 +      data->signal_blob = g_string_new_len ("", 0);
   1.494 +      data->property_blob = g_string_new_len ("", 0);
   1.495 +
   1.496 +      if (!write_printf_to_iochannel ("static const DBusGMethodInfo dbus_glib_%s_methods[] = {\n", channel, error, data->prefix))
   1.497 +	goto io_lose;
   1.498 +
   1.499 +      if (!generate_glue_list (node_info_get_nodes ((NodeInfo *) base),
   1.500 +			       data, error))
   1.501 +	return FALSE;
   1.502 +      if (!generate_glue_list (node_info_get_interfaces ((NodeInfo *) base),
   1.503 +			       data, error))
   1.504 +	return FALSE;
   1.505 +
   1.506 +      WRITE_OR_LOSE ("};\n\n");
   1.507 +
   1.508 +      /* Information about the object. */
   1.509 +
   1.510 +      if (!write_printf_to_iochannel ("const DBusGObjectInfo dbus_glib_%s_object_info = {\n",
   1.511 +				      channel, error, data->prefix))
   1.512 +	goto io_lose;
   1.513 +      WRITE_OR_LOSE ("  0,\n");
   1.514 +      if (!write_printf_to_iochannel ("  dbus_glib_%s_methods,\n", channel, error, data->prefix))
   1.515 +	goto io_lose;
   1.516 +      if (!write_printf_to_iochannel ("  %d,\n", channel, error, data->count))
   1.517 +	goto io_lose;
   1.518 +
   1.519 +      if (!write_quoted_string (channel, object_introspection_data_blob, error))
   1.520 +	goto io_lose;
   1.521 +      WRITE_OR_LOSE (",\n");
   1.522 +      if (!write_quoted_string (channel, data->signal_blob, error))
   1.523 +	goto io_lose;
   1.524 +      WRITE_OR_LOSE (",\n");
   1.525 +      if (!write_quoted_string (channel, data->property_blob, error))
   1.526 +	goto io_lose;
   1.527 +      WRITE_OR_LOSE ("\n};\n\n");
   1.528 +
   1.529 +      g_string_free (object_introspection_data_blob, TRUE);
   1.530 +      g_string_free (data->signal_blob, TRUE);
   1.531 +      g_string_free (data->property_blob, TRUE);
   1.532 +    }
   1.533 +  else
   1.534 +    {
   1.535 +      GIOChannel *channel;
   1.536 +      InterfaceInfo *interface;
   1.537 +      GSList *methods;
   1.538 +      GSList *signals;
   1.539 +      GSList *properties;
   1.540 +      GSList *tmp;
   1.541 +      const char *interface_c_name;
   1.542 +      GString *object_introspection_data_blob;
   1.543 +
   1.544 +      channel = data->channel;
   1.545 +      object_introspection_data_blob = data->blob;
   1.546 +
   1.547 +      interface = (InterfaceInfo *) base;
   1.548 +      interface_c_name = interface_info_get_annotation (interface, DBUS_GLIB_ANNOTATION_C_SYMBOL);
   1.549 +      if (interface_c_name == NULL)
   1.550 +        {
   1.551 +	  if (data->prefix == NULL)
   1.552 +	    return TRUE;
   1.553 +	  interface_c_name = data->prefix;
   1.554 +        }
   1.555 +
   1.556 +      methods = interface_info_get_methods (interface);
   1.557 +
   1.558 +      /* Table of marshalled methods. */
   1.559 +
   1.560 +      for (tmp = methods; tmp != NULL; tmp = g_slist_next (tmp))
   1.561 +        {
   1.562 +          MethodInfo *method;
   1.563 +          char *marshaller_name;
   1.564 +	  char *method_c_name;
   1.565 +          gboolean async = FALSE;
   1.566 +	  GSList *args;
   1.567 +	  gboolean found_retval = FALSE;
   1.568 +
   1.569 +          method = (MethodInfo *) tmp->data;
   1.570 +	  method_c_name = g_strdup (method_info_get_annotation (method, DBUS_GLIB_ANNOTATION_C_SYMBOL));
   1.571 +          if (method_c_name == NULL)
   1.572 +	    {
   1.573 +	      char *method_name_uscored;
   1.574 +	      method_name_uscored = _dbus_gutils_wincaps_to_uscore (method_info_get_name (method));
   1.575 +              method_c_name = g_strdup_printf ("%s_%s",
   1.576 +					       interface_c_name,
   1.577 +					       method_name_uscored);
   1.578 +	      g_free (method_name_uscored);
   1.579 +            }
   1.580 +
   1.581 +          if (!write_printf_to_iochannel ("  { (GCallback) %s, ", channel, error,
   1.582 +					  method_c_name))
   1.583 +	    goto io_lose;
   1.584 +
   1.585 +          marshaller_name = compute_marshaller_name (method, data->prefix, error);
   1.586 +	  if (!marshaller_name)
   1.587 +	    goto io_lose;
   1.588 +
   1.589 +          if (!write_printf_to_iochannel ("%s, %d },\n", channel, error,
   1.590 +					  marshaller_name,
   1.591 +					  object_introspection_data_blob->len))
   1.592 +	    {
   1.593 +	      g_free (marshaller_name);
   1.594 +	      goto io_lose;
   1.595 +	    }
   1.596 +
   1.597 +          if (method_info_get_annotation (method, DBUS_GLIB_ANNOTATION_ASYNC) != NULL)
   1.598 +            async = TRUE;
   1.599 +
   1.600 +	  /* Object method data blob format:
   1.601 +	   * <iface>\0<name>\0(<argname>\0<argdirection>\0<argtype>\0)*\0
   1.602 +	   */
   1.603 +
   1.604 +	  g_string_append (object_introspection_data_blob, interface_info_get_name (interface));
   1.605 +	  g_string_append_c (object_introspection_data_blob, '\0');
   1.606 +
   1.607 +	  g_string_append (object_introspection_data_blob, method_info_get_name (method));
   1.608 +	  g_string_append_c (object_introspection_data_blob, '\0');
   1.609 +
   1.610 +	  g_string_append_c (object_introspection_data_blob, async ? 'A' : 'S');
   1.611 +	  g_string_append_c (object_introspection_data_blob, '\0');
   1.612 +
   1.613 +	  for (args = method_info_get_args (method); args; args = args->next)
   1.614 +	    {
   1.615 +	      ArgInfo *arg;
   1.616 +	      char direction;
   1.617 +	      const char *returnval_annotation;
   1.618 +
   1.619 +	      arg = args->data;
   1.620 +
   1.621 +	      g_string_append (object_introspection_data_blob, arg_info_get_name (arg));
   1.622 +	      g_string_append_c (object_introspection_data_blob, '\0');
   1.623 +
   1.624 +	      switch (arg_info_get_direction (arg))
   1.625 +		{
   1.626 +		case ARG_IN:
   1.627 +		  direction = 'I';
   1.628 +		  break;
   1.629 +		case ARG_OUT:
   1.630 +		  direction = 'O';
   1.631 +		  break;
   1.632 +		case ARG_INVALID:
   1.633 +                default:
   1.634 +                  g_assert_not_reached ();
   1.635 +                  direction = 0; /* silence gcc */
   1.636 +		  break;
   1.637 +		}
   1.638 +	      g_string_append_c (object_introspection_data_blob, direction);
   1.639 +	      g_string_append_c (object_introspection_data_blob, '\0');
   1.640 +
   1.641 +	      if (arg_info_get_annotation (arg, DBUS_GLIB_ANNOTATION_CONST) != NULL)
   1.642 +		{
   1.643 +		  if (arg_info_get_direction (arg) == ARG_IN)
   1.644 +		    {
   1.645 +		      g_set_error (error,
   1.646 +				   DBUS_BINDING_TOOL_ERROR,
   1.647 +				   DBUS_BINDING_TOOL_ERROR_INVALID_ANNOTATION,
   1.648 +				   "Input argument \"%s\" cannot have const annotation in method \"%s\" of interface \"%s\"\n",
   1.649 +				   arg_info_get_name (arg),
   1.650 +				   method_info_get_name (method),
   1.651 +				   interface_info_get_name (interface));
   1.652 +		      return FALSE;
   1.653 +		    }
   1.654 +		  g_string_append_c (object_introspection_data_blob, 'C');
   1.655 +		  g_string_append_c (object_introspection_data_blob, '\0');
   1.656 +		}
   1.657 +	      else if (arg_info_get_direction (arg) == ARG_OUT)
   1.658 +		{
   1.659 +		  g_string_append_c (object_introspection_data_blob, 'F');
   1.660 +		  g_string_append_c (object_introspection_data_blob, '\0');
   1.661 +		}
   1.662 +
   1.663 +	      returnval_annotation = arg_info_get_annotation (arg, DBUS_GLIB_ANNOTATION_RETURNVAL);
   1.664 +	      if (returnval_annotation != NULL)
   1.665 +		{
   1.666 +		  GType gtype;
   1.667 +
   1.668 +		  if (found_retval)
   1.669 +		    {
   1.670 +		      g_set_error (error,
   1.671 +				   DBUS_BINDING_TOOL_ERROR,
   1.672 +				   DBUS_BINDING_TOOL_ERROR_INVALID_ANNOTATION,
   1.673 +				   "Multiple arguments with return value annotation in method \"%s\" of interface \"%s\"\n",
   1.674 +				   method_info_get_name (method),
   1.675 +				   interface_info_get_name (interface));
   1.676 +		      return FALSE;
   1.677 +		    }
   1.678 +		  found_retval = TRUE;
   1.679 +		  if (arg_info_get_direction (arg) == ARG_IN)
   1.680 +		    {
   1.681 +		      g_set_error (error,
   1.682 +				   DBUS_BINDING_TOOL_ERROR,
   1.683 +				   DBUS_BINDING_TOOL_ERROR_INVALID_ANNOTATION,
   1.684 +				   "Input argument \"%s\" cannot have return value annotation in method \"%s\" of interface \"%s\"\n",
   1.685 +				   arg_info_get_name (arg),
   1.686 +				   method_info_get_name (method),
   1.687 +				   interface_info_get_name (interface));
   1.688 +		      return FALSE;
   1.689 +		    }
   1.690 +		  if (!strcmp ("", returnval_annotation))
   1.691 +		    g_string_append_c (object_introspection_data_blob, 'R');
   1.692 +		  else if (!strcmp ("error", returnval_annotation))
   1.693 +		    {
   1.694 +		      gtype = _dbus_gtype_from_signature (arg_info_get_type (arg), TRUE);
   1.695 +		      if (!_dbus_gtype_can_signal_error (gtype))
   1.696 +			{
   1.697 +			  g_set_error (error,
   1.698 +				       DBUS_BINDING_TOOL_ERROR,
   1.699 +				       DBUS_BINDING_TOOL_ERROR_INVALID_ANNOTATION,
   1.700 +				       "Output argument \"%s\" cannot signal error with type \"%s\" in method \"%s\" of interface \"%s\"\n",
   1.701 +				       arg_info_get_name (arg),
   1.702 +				       g_type_name (gtype),
   1.703 +				       method_info_get_name (method),
   1.704 +				       interface_info_get_name (interface));
   1.705 +			  return FALSE;
   1.706 +			}
   1.707 +		      g_string_append_c (object_introspection_data_blob, 'E');
   1.708 +		    }
   1.709 +		  else
   1.710 +		    {
   1.711 +		      g_set_error (error,
   1.712 +				   DBUS_BINDING_TOOL_ERROR,
   1.713 +				   DBUS_BINDING_TOOL_ERROR_INVALID_ANNOTATION,
   1.714 +				   "Invalid ReturnVal annotation for argument \"%s\" in method \"%s\" of interface \"%s\"\n",
   1.715 +				   arg_info_get_name (arg),
   1.716 +				   method_info_get_name (method),
   1.717 +				   interface_info_get_name (interface));
   1.718 +		      return FALSE;
   1.719 +		    }
   1.720 +		      
   1.721 +		  g_string_append_c (object_introspection_data_blob, '\0');
   1.722 +		}
   1.723 +	      else if (arg_info_get_direction (arg) == ARG_OUT)
   1.724 +		{
   1.725 +		  g_string_append_c (object_introspection_data_blob, 'N');
   1.726 +		  g_string_append_c (object_introspection_data_blob, '\0');
   1.727 +		}
   1.728 +
   1.729 +	      g_string_append (object_introspection_data_blob, arg_info_get_type (arg));
   1.730 +	      g_string_append_c (object_introspection_data_blob, '\0');
   1.731 +	    }
   1.732 +
   1.733 +	  g_string_append_c (object_introspection_data_blob, '\0');
   1.734 +
   1.735 +          data->count++;
   1.736 +        }
   1.737 +
   1.738 +      signals = interface_info_get_signals (interface);
   1.739 +
   1.740 +      for (tmp = signals; tmp != NULL; tmp = g_slist_next (tmp))
   1.741 +        {
   1.742 +          SignalInfo *sig;
   1.743 +	  
   1.744 +	  sig = tmp->data;
   1.745 +
   1.746 +	  g_string_append (data->signal_blob, interface_info_get_name (interface));
   1.747 +	  g_string_append_c (data->signal_blob, '\0');
   1.748 +	  g_string_append (data->signal_blob, signal_info_get_name (sig));
   1.749 +	  g_string_append_c (data->signal_blob, '\0');
   1.750 +	}
   1.751 +
   1.752 +      properties = interface_info_get_properties (interface);
   1.753 +
   1.754 +      for (tmp = properties; tmp != NULL; tmp = g_slist_next (tmp))
   1.755 +        {
   1.756 +          PropertyInfo *prop;
   1.757 +	  
   1.758 +	  prop = tmp->data;
   1.759 +
   1.760 +	  g_string_append (data->property_blob, interface_info_get_name (interface));
   1.761 +	  g_string_append_c (data->property_blob, '\0');
   1.762 +	  g_string_append (data->property_blob, property_info_get_name (prop));
   1.763 +	  g_string_append_c (data->property_blob, '\0');
   1.764 +	}
   1.765 +    }
   1.766 +  return TRUE;
   1.767 + io_lose:
   1.768 +  return FALSE;
   1.769 +}
   1.770 +
   1.771 +static void
   1.772 +write_marshaller (gpointer key, gpointer value, gpointer user_data)
   1.773 +{
   1.774 +  DBusBindingToolCData *data;
   1.775 +  const char *marshaller;
   1.776 +  gsize bytes_written;
   1.777 +
   1.778 +  data = user_data;
   1.779 +  marshaller = key;
   1.780 +
   1.781 +  if (data->error && *data->error)
   1.782 +    return;
   1.783 +
   1.784 +  if (g_io_channel_write_chars (data->channel, marshaller, -1, &bytes_written, data->error) == G_IO_STATUS_NORMAL)
   1.785 +    g_io_channel_write_chars (data->channel, "\n", -1, &bytes_written, data->error);
   1.786 +}
   1.787 +
   1.788 +gboolean
   1.789 +dbus_binding_tool_output_glib_server (BaseInfo *info, GIOChannel *channel, const char *prefix, GError **error)
   1.790 +{
   1.791 +  gboolean ret;
   1.792 +  GPtrArray *argv;
   1.793 +  gint child_stdout;
   1.794 +  GIOChannel *genmarshal_stdout;
   1.795 +  GPid child_pid;
   1.796 +  DBusBindingToolCData data;
   1.797 +  char *tempfile_name;
   1.798 +  gint tempfile_fd;
   1.799 +  GIOStatus iostatus;
   1.800 +  char buf[4096];
   1.801 +  gsize bytes_read, bytes_written;
   1.802 +
   1.803 +  memset (&data, 0, sizeof (data));
   1.804 +
   1.805 +  dbus_g_type_specialized_init ();
   1.806 +  _dbus_g_type_specialized_builtins_init ();
   1.807 +
   1.808 +  data.prefix = prefix;
   1.809 +  data.generated = g_hash_table_new_full (g_str_hash, g_str_equal, (GDestroyNotify) g_free, NULL);
   1.810 +  data.error = error;
   1.811 +  genmarshal_stdout = NULL;
   1.812 +  tempfile_name = NULL;
   1.813 +
   1.814 +  if (!gather_marshallers (info, &data, error))
   1.815 +    goto io_lose;
   1.816 +
   1.817 +  tempfile_fd = g_file_open_tmp ("dbus-binding-tool-c-marshallers.XXXXXX",
   1.818 +				 &tempfile_name, error);
   1.819 +  if (tempfile_fd < 0)
   1.820 +    goto io_lose;
   1.821 +
   1.822 +  data.channel = g_io_channel_unix_new (tempfile_fd);
   1.823 +  if (!g_io_channel_set_encoding (data.channel, NULL, error))
   1.824 +    goto io_lose;
   1.825 +  g_hash_table_foreach (data.generated, write_marshaller, &data); 
   1.826 +  if (error && *error != NULL)
   1.827 +    {
   1.828 +      ret = FALSE;
   1.829 +      g_io_channel_close (data.channel);
   1.830 +      g_io_channel_unref (data.channel);
   1.831 +      goto io_lose;
   1.832 +    }
   1.833 +
   1.834 +  g_io_channel_close (data.channel);
   1.835 +  g_io_channel_unref (data.channel);
   1.836 +  
   1.837 +  /* Now spawn glib-genmarshal to insert all our required marshallers */
   1.838 +  argv = g_ptr_array_new ();
   1.839 +  g_ptr_array_add (argv, "glib-genmarshal");
   1.840 +  g_ptr_array_add (argv, "--header");
   1.841 +  g_ptr_array_add (argv, "--body");
   1.842 +  g_ptr_array_add (argv, g_strdup_printf ("--prefix=%s%s", MARSHAL_PREFIX, prefix));
   1.843 +  g_ptr_array_add (argv, tempfile_name);
   1.844 +  g_ptr_array_add (argv, NULL);
   1.845 +  if (!g_spawn_async_with_pipes (NULL, (char**)argv->pdata, NULL,
   1.846 +				 G_SPAWN_SEARCH_PATH,
   1.847 +				 NULL, NULL,
   1.848 +				 &child_pid,
   1.849 +				 NULL,
   1.850 +				 &child_stdout, NULL, error))
   1.851 +    {
   1.852 +      g_ptr_array_free (argv, TRUE);
   1.853 +      goto io_lose;
   1.854 +    }
   1.855 +  g_ptr_array_free (argv, TRUE);
   1.856 +
   1.857 +  genmarshal_stdout = g_io_channel_unix_new (child_stdout);
   1.858 +  if (!g_io_channel_set_encoding (genmarshal_stdout, NULL, error))
   1.859 +    goto io_lose;
   1.860 +
   1.861 +  WRITE_OR_LOSE ("/* Generated by dbus-binding-tool; do not edit! */\n\n");
   1.862 +
   1.863 +  while ((iostatus = g_io_channel_read_chars (genmarshal_stdout, buf, sizeof (buf),
   1.864 +					      &bytes_read, error)) == G_IO_STATUS_NORMAL)
   1.865 +    if (g_io_channel_write_chars (channel, buf, bytes_read, &bytes_written, error) != G_IO_STATUS_NORMAL)
   1.866 +      goto io_lose;
   1.867 +  if (iostatus != G_IO_STATUS_EOF)
   1.868 +    goto io_lose;
   1.869 +
   1.870 +  g_io_channel_close (genmarshal_stdout);
   1.871 +
   1.872 +  WRITE_OR_LOSE ("#include <dbus/dbus-glib.h>\n");
   1.873 +
   1.874 +  data.channel = channel;
   1.875 +  g_io_channel_ref (data.channel);
   1.876 +  if (!generate_glue (info, &data, error))
   1.877 +    goto io_lose;
   1.878 +  
   1.879 +  ret = TRUE;
   1.880 + cleanup:
   1.881 +  if (tempfile_name)
   1.882 +    unlink (tempfile_name);
   1.883 +  g_free (tempfile_name);
   1.884 +  if (genmarshal_stdout)
   1.885 +    g_io_channel_unref (genmarshal_stdout);
   1.886 +  if (data.channel)
   1.887 +    g_io_channel_unref (data.channel);
   1.888 +  g_hash_table_destroy (data.generated);
   1.889 +
   1.890 +  return ret;
   1.891 + io_lose:
   1.892 +  ret = FALSE;
   1.893 +  goto cleanup;
   1.894 +}
   1.895 +
   1.896 +static char *
   1.897 +iface_to_c_prefix (const char *iface)
   1.898 +{
   1.899 +  char **components;
   1.900 +  char **component;
   1.901 +  GString *ret;
   1.902 +  gboolean first;
   1.903 +  
   1.904 +  components = g_strsplit (iface, ".", 0);
   1.905 +
   1.906 +  first = TRUE;
   1.907 +  ret = g_string_new ("");
   1.908 +  for (component = components; *component; component++)
   1.909 +    {
   1.910 +      if (!first)
   1.911 +	g_string_append_c (ret, '_');
   1.912 +      else
   1.913 +	first = FALSE;
   1.914 +      g_string_append (ret, *component);
   1.915 +    }
   1.916 +  g_strfreev (components);
   1.917 +  return g_string_free (ret, FALSE);
   1.918 +}
   1.919 +
   1.920 +static char *
   1.921 +compute_client_method_name (const char *iface_prefix, MethodInfo *method)
   1.922 +{
   1.923 +  char *method_name_uscored, *ret;
   1.924 +
   1.925 +  method_name_uscored = _dbus_gutils_wincaps_to_uscore (method_info_get_name (method));
   1.926 +  ret = g_strdup_printf ("%s_%s", iface_prefix, method_name_uscored);
   1.927 +  g_free (method_name_uscored);
   1.928 +
   1.929 +  return ret;
   1.930 +}
   1.931 +
   1.932 +static gboolean
   1.933 +write_formal_parameters (InterfaceInfo *iface, MethodInfo *method, GIOChannel *channel, GError **error)
   1.934 +{
   1.935 +  GSList *args;
   1.936 +
   1.937 +  for (args = method_info_get_args (method); args; args = args->next)
   1.938 +    {
   1.939 +      ArgInfo *arg;
   1.940 +      const char *type_str;
   1.941 +      const char *type_suffix;
   1.942 +      GType gtype;
   1.943 +      int direction;
   1.944 +
   1.945 +      arg = args->data;
   1.946 +
   1.947 +      WRITE_OR_LOSE (", ");
   1.948 +
   1.949 +      direction = arg_info_get_direction (arg);
   1.950 +
   1.951 +      gtype = _dbus_gtype_from_signature (arg_info_get_type (arg), TRUE);
   1.952 +      if (gtype == G_TYPE_INVALID)
   1.953 +	{
   1.954 +	  g_set_error (error,
   1.955 +		       DBUS_BINDING_TOOL_ERROR,
   1.956 +		       DBUS_BINDING_TOOL_ERROR_UNSUPPORTED_CONVERSION,
   1.957 +		       _("Unsupported conversion from D-BUS type signature \"%s\" to glib C type in method \"%s\" of interface \"%s\""),
   1.958 +		       arg_info_get_type (arg),
   1.959 +		       method_info_get_name (method),
   1.960 +		       interface_info_get_name (iface));
   1.961 +	  return FALSE;
   1.962 +	}
   1.963 +      type_str = dbus_g_type_get_c_name (gtype);
   1.964 +      g_assert (type_str);
   1.965 +      /* Variants are special...*/
   1.966 +      if (gtype == G_TYPE_VALUE)
   1.967 +	{
   1.968 +	  if (direction == ARG_IN)
   1.969 +	    type_suffix = "*";
   1.970 +	  else
   1.971 +	    type_suffix = "";
   1.972 +	}
   1.973 +      else if ((g_type_is_a (gtype, G_TYPE_BOXED)
   1.974 +	      || g_type_is_a (gtype, G_TYPE_OBJECT)
   1.975 +	   || g_type_is_a (gtype, G_TYPE_POINTER)))
   1.976 +	type_suffix = "*";
   1.977 +      else
   1.978 +	type_suffix = "";
   1.979 +
   1.980 +
   1.981 +      switch (direction)
   1.982 +	{
   1.983 +	case ARG_IN:
   1.984 +	  if (!write_printf_to_iochannel ("const %s%s IN_%s", channel, error,
   1.985 +					  type_str,
   1.986 +					  type_suffix,
   1.987 +					  arg_info_get_name (arg)))
   1.988 +	    goto io_lose;
   1.989 +	  break;
   1.990 +	case ARG_OUT:
   1.991 +	  if (!write_printf_to_iochannel ("%s%s* OUT_%s", channel, error,
   1.992 +					  type_str,
   1.993 +					  type_suffix,
   1.994 +					  arg_info_get_name (arg)))
   1.995 +	    goto io_lose;
   1.996 +	  break;
   1.997 +	case ARG_INVALID:
   1.998 +	  break;
   1.999 +	}
  1.1000 +    }
  1.1001 +
  1.1002 +  return TRUE;
  1.1003 + io_lose:
  1.1004 +  return FALSE;
  1.1005 +}
  1.1006 +
  1.1007 +#define MAP_FUNDAMENTAL(NAME) \
  1.1008 +   case G_TYPE_ ## NAME: \
  1.1009 +     return g_strdup ("G_TYPE_" #NAME);
  1.1010 +#define MAP_KNOWN(NAME) \
  1.1011 +    if (gtype == NAME) \
  1.1012 +      return g_strdup (#NAME)
  1.1013 +static char *
  1.1014 +dbus_g_type_get_lookup_function (GType gtype)
  1.1015 +{
  1.1016 +  char *type_lookup;
  1.1017 +  switch (gtype)
  1.1018 +    {
  1.1019 +      MAP_FUNDAMENTAL(CHAR);
  1.1020 +      MAP_FUNDAMENTAL(UCHAR);
  1.1021 +      MAP_FUNDAMENTAL(BOOLEAN);
  1.1022 +      MAP_FUNDAMENTAL(LONG);
  1.1023 +      MAP_FUNDAMENTAL(ULONG);
  1.1024 +      MAP_FUNDAMENTAL(INT);
  1.1025 +      MAP_FUNDAMENTAL(UINT);
  1.1026 +      MAP_FUNDAMENTAL(INT64);
  1.1027 +      MAP_FUNDAMENTAL(UINT64);
  1.1028 +      MAP_FUNDAMENTAL(FLOAT);
  1.1029 +      MAP_FUNDAMENTAL(DOUBLE);
  1.1030 +      MAP_FUNDAMENTAL(STRING);
  1.1031 +    }
  1.1032 +  if (dbus_g_type_is_collection (gtype))
  1.1033 +    {
  1.1034 +      GType elt_gtype;
  1.1035 +      char *sublookup;
  1.1036 +
  1.1037 +      elt_gtype = dbus_g_type_get_collection_specialization (gtype);
  1.1038 +      sublookup = dbus_g_type_get_lookup_function (elt_gtype);
  1.1039 +      g_assert (sublookup);
  1.1040 +
  1.1041 +      if (_dbus_g_type_is_fixed (elt_gtype))
  1.1042 +        {
  1.1043 +          type_lookup = g_strdup_printf ("dbus_g_type_get_collection "
  1.1044 +              "(\"GArray\", %s)", sublookup);
  1.1045 +        }
  1.1046 +      else
  1.1047 +        {
  1.1048 +          type_lookup = g_strdup_printf ("dbus_g_type_get_collection "
  1.1049 +              "(\"GPtrArray\", %s)", sublookup);
  1.1050 +        }
  1.1051 +
  1.1052 +      g_free (sublookup);
  1.1053 +
  1.1054 +      return type_lookup;
  1.1055 +    }
  1.1056 +  else if (dbus_g_type_is_map (gtype))
  1.1057 +    {
  1.1058 +      GType key_gtype;
  1.1059 +      char *key_lookup;
  1.1060 +      GType value_gtype;
  1.1061 +      char *value_lookup;
  1.1062 +      
  1.1063 +      key_gtype = dbus_g_type_get_map_key_specialization (gtype);
  1.1064 +      value_gtype = dbus_g_type_get_map_value_specialization (gtype);
  1.1065 +      key_lookup = dbus_g_type_get_lookup_function (key_gtype);
  1.1066 +      g_assert (key_lookup);
  1.1067 +      value_lookup = dbus_g_type_get_lookup_function (value_gtype);
  1.1068 +      g_assert (value_lookup);
  1.1069 +      type_lookup = g_strdup_printf ("dbus_g_type_get_map (\"GHashTable\", %s, %s)",
  1.1070 +				     key_lookup, value_lookup);
  1.1071 +      g_free (key_lookup);
  1.1072 +      g_free (value_lookup);
  1.1073 +      return type_lookup;
  1.1074 +    }
  1.1075 +  else if (dbus_g_type_is_struct (gtype))
  1.1076 +    {
  1.1077 +      GType value_gtype;
  1.1078 +      GString *string;
  1.1079 +      char *value_lookup = NULL;
  1.1080 +      guint size, i;
  1.1081 +
  1.1082 +      string = g_string_new ("dbus_g_type_get_struct (\"GValueArray\"");
  1.1083 +
  1.1084 +      size = dbus_g_type_get_struct_size (gtype);
  1.1085 +      for (i=0; i < size; i++)
  1.1086 +        {
  1.1087 +          value_gtype = dbus_g_type_get_struct_member_type(gtype, i);
  1.1088 +          value_lookup = dbus_g_type_get_lookup_function (value_gtype);
  1.1089 +          g_assert (value_lookup);
  1.1090 +          g_string_append_printf (string, ", %s", value_lookup);
  1.1091 +          g_free (value_lookup);
  1.1092 +        }
  1.1093 +      g_string_append (string, ", G_TYPE_INVALID)");
  1.1094 +      return g_string_free (string, FALSE);
  1.1095 +    }
  1.1096 +
  1.1097 +  MAP_KNOWN(G_TYPE_VALUE);
  1.1098 +  MAP_KNOWN(G_TYPE_STRV);
  1.1099 +  MAP_KNOWN(G_TYPE_VALUE_ARRAY);
  1.1100 +  MAP_KNOWN(DBUS_TYPE_G_PROXY);
  1.1101 +  MAP_KNOWN(DBUS_TYPE_G_OBJECT_PATH);
  1.1102 +  return NULL;
  1.1103 +}
  1.1104 +#undef MAP_FUNDAMENTAL
  1.1105 +#undef MAP_KNOWN
  1.1106 +
  1.1107 +static gboolean
  1.1108 +write_args_for_direction (InterfaceInfo *iface, MethodInfo *method, GIOChannel *channel, int direction, GError **error)
  1.1109 +{
  1.1110 +  GSList *args;
  1.1111 +
  1.1112 +  for (args = method_info_get_args (method); args; args = args->next)
  1.1113 +    {
  1.1114 +      ArgInfo *arg;
  1.1115 +      GType gtype;
  1.1116 +      char *type_lookup;
  1.1117 +
  1.1118 +      arg = args->data;
  1.1119 +
  1.1120 +      if (direction != arg_info_get_direction (arg))
  1.1121 +	continue;
  1.1122 +
  1.1123 +      gtype = _dbus_gtype_from_signature (arg_info_get_type (arg), TRUE);
  1.1124 +      g_assert (gtype != G_TYPE_INVALID);
  1.1125 +      type_lookup = dbus_g_type_get_lookup_function (gtype);
  1.1126 +      g_assert (type_lookup != NULL);
  1.1127 +
  1.1128 +      switch (direction)
  1.1129 +	{
  1.1130 +
  1.1131 +	case ARG_IN:
  1.1132 +	  if (!write_printf_to_iochannel ("%s, IN_%s, ", channel, error,
  1.1133 +					  type_lookup,
  1.1134 +					  arg_info_get_name (arg)))
  1.1135 +	    goto io_lose;
  1.1136 +	  break;
  1.1137 +	case ARG_OUT:
  1.1138 +	  if (!write_printf_to_iochannel ("%s, OUT_%s, ", channel, error,
  1.1139 +					  type_lookup,
  1.1140 +					  arg_info_get_name (arg)))
  1.1141 +	    goto io_lose;
  1.1142 +	  break;
  1.1143 +	case ARG_INVALID:
  1.1144 +	  break;
  1.1145 +	}
  1.1146 +      g_free (type_lookup);
  1.1147 +    }
  1.1148 +
  1.1149 +  return TRUE;
  1.1150 + io_lose:
  1.1151 +  return FALSE;
  1.1152 +}
  1.1153 +
  1.1154 +static gboolean
  1.1155 +check_supported_parameters (MethodInfo *method)
  1.1156 +{
  1.1157 +  GSList *args;
  1.1158 +
  1.1159 +  for (args = method_info_get_args (method); args; args = args->next)
  1.1160 +    {
  1.1161 +      ArgInfo *arg;
  1.1162 +      GType gtype;
  1.1163 +
  1.1164 +      arg = args->data;
  1.1165 +      gtype = _dbus_gtype_from_signature (arg_info_get_type (arg), TRUE);
  1.1166 +      if (gtype == G_TYPE_INVALID)
  1.1167 +	return FALSE;
  1.1168 +    }
  1.1169 +  return TRUE;
  1.1170 +}
  1.1171 +
  1.1172 +static gboolean
  1.1173 +write_untyped_out_args (InterfaceInfo *iface, MethodInfo *method, GIOChannel *channel, GError **error)
  1.1174 +{
  1.1175 +  GSList *args;
  1.1176 +
  1.1177 +  for (args = method_info_get_args (method); args; args = args->next)
  1.1178 +    {
  1.1179 +      ArgInfo *arg;
  1.1180 +
  1.1181 +      arg = args->data;
  1.1182 +      if (arg_info_get_direction (arg) != ARG_OUT)
  1.1183 +        continue;
  1.1184 +            
  1.1185 +      if (!write_printf_to_iochannel ("OUT_%s, ", channel, error,
  1.1186 +                                      arg_info_get_name (arg)))
  1.1187 +        goto io_lose;
  1.1188 +     }
  1.1189 +
  1.1190 +   return TRUE;
  1.1191 + io_lose:
  1.1192 +  return FALSE;
  1.1193 +}
  1.1194 +
  1.1195 +static gboolean
  1.1196 +write_formal_declarations_for_direction (InterfaceInfo *iface, MethodInfo *method, GIOChannel *channel, const int direction, GError **error)
  1.1197 + {
  1.1198 +   GSList *args;
  1.1199 + 
  1.1200 +   for (args = method_info_get_args (method); args; args = args->next)
  1.1201 +     {
  1.1202 +       ArgInfo *arg;
  1.1203 +      GType gtype;
  1.1204 +      const char *type_str, *type_suffix;
  1.1205 +      int dir;
  1.1206 +
  1.1207 +       arg = args->data;
  1.1208 +
  1.1209 +      dir = arg_info_get_direction (arg);
  1.1210 +
  1.1211 +      gtype = _dbus_gtype_from_signature (arg_info_get_type (arg), TRUE);
  1.1212 +      type_str = dbus_g_type_get_c_name (gtype);
  1.1213 +
  1.1214 +      if (!type_str)
  1.1215 +       {
  1.1216 +         g_set_error (error,
  1.1217 +                      DBUS_BINDING_TOOL_ERROR,
  1.1218 +                      DBUS_BINDING_TOOL_ERROR_UNSUPPORTED_CONVERSION,
  1.1219 +                      _("Unsupported conversion from D-BUS type signature \"%s\" to glib C type in method \"%s\" of interface \"%s\""),
  1.1220 +                      arg_info_get_type (arg),
  1.1221 +                      method_info_get_name (method),
  1.1222 +                      interface_info_get_name (iface));
  1.1223 +         return FALSE;
  1.1224 +       }
  1.1225 +
  1.1226 +      /* Variants are special...*/
  1.1227 +      if (gtype == G_TYPE_VALUE)
  1.1228 +	{
  1.1229 +	  if (direction == ARG_IN)
  1.1230 +	    type_suffix = "*";
  1.1231 +	  else
  1.1232 +	    type_suffix = "";
  1.1233 +	}
  1.1234 +      else if ((g_type_is_a (gtype, G_TYPE_BOXED)
  1.1235 +	      || g_type_is_a (gtype, G_TYPE_OBJECT)
  1.1236 +	   || g_type_is_a (gtype, G_TYPE_POINTER)))
  1.1237 +	type_suffix = "*";
  1.1238 +      else
  1.1239 +	type_suffix = "";
  1.1240 +
  1.1241 +      if (direction != dir)
  1.1242 +        continue;
  1.1243 +
  1.1244 +          switch (dir)
  1.1245 +       {
  1.1246 +       case ARG_IN:
  1.1247 +         if (!write_printf_to_iochannel ("  %s%s IN_%s;\n", channel, error,
  1.1248 +                                         type_str, type_suffix,
  1.1249 +                                         arg_info_get_name (arg)))
  1.1250 +           goto io_lose;
  1.1251 +         break;
  1.1252 +       case ARG_OUT:
  1.1253 +         if (!write_printf_to_iochannel ("  %s%s OUT_%s;\n", channel, error,
  1.1254 +                                         type_str, type_suffix,
  1.1255 +                                         arg_info_get_name (arg)))
  1.1256 +           goto io_lose;
  1.1257 +         break;
  1.1258 +       case ARG_INVALID:
  1.1259 +         break;
  1.1260 +       }
  1.1261 +     }
  1.1262 +   return TRUE;
  1.1263 + io_lose:
  1.1264 +  return FALSE;
  1.1265 + }
  1.1266 +
  1.1267 +static gboolean
  1.1268 +write_formal_parameters_for_direction (InterfaceInfo *iface, MethodInfo *method, int dir, GIOChannel *channel, GError **error)
  1.1269 +{
  1.1270 +  GSList *args;
  1.1271 +
  1.1272 +  for (args = method_info_get_args (method); args; args = args->next)
  1.1273 +    {
  1.1274 +      ArgInfo *arg;
  1.1275 +      const char *type_str;
  1.1276 +      const char *type_suffix;
  1.1277 +      GType gtype;
  1.1278 +      int direction;
  1.1279 +
  1.1280 +      arg = args->data;
  1.1281 +
  1.1282 +      direction = arg_info_get_direction (arg);
  1.1283 +      if (dir != direction) continue;
  1.1284 +      
  1.1285 +      WRITE_OR_LOSE (", ");
  1.1286 +
  1.1287 +      gtype = _dbus_gtype_from_signature (arg_info_get_type (arg), TRUE);
  1.1288 +      type_str = dbus_g_type_get_c_name (gtype);
  1.1289 +      /* Variants are special...*/
  1.1290 +      if (gtype == G_TYPE_VALUE)
  1.1291 +	{
  1.1292 +	  if (direction == ARG_IN)
  1.1293 +	    type_suffix = "*";
  1.1294 +	  else
  1.1295 +	    type_suffix = "";
  1.1296 +	}
  1.1297 +      else if ((g_type_is_a (gtype, G_TYPE_BOXED)
  1.1298 +	      || g_type_is_a (gtype, G_TYPE_OBJECT)
  1.1299 +	   || g_type_is_a (gtype, G_TYPE_POINTER)))
  1.1300 +	type_suffix = "*";
  1.1301 +      else
  1.1302 +	type_suffix = "";
  1.1303 +
  1.1304 +      if (!type_str)
  1.1305 +	{
  1.1306 +	  g_set_error (error,
  1.1307 +		       DBUS_BINDING_TOOL_ERROR,
  1.1308 +		       DBUS_BINDING_TOOL_ERROR_UNSUPPORTED_CONVERSION,
  1.1309 +		       _("Unsupported conversion from D-BUS type signature \"%s\" to glib C type in method \"%s\" of interface \"%s\""),
  1.1310 +		       arg_info_get_type (arg),
  1.1311 +		       method_info_get_name (method),
  1.1312 +		       interface_info_get_name (iface));
  1.1313 +	  return FALSE;
  1.1314 +	}
  1.1315 + 
  1.1316 +       switch (direction)
  1.1317 + 	{
  1.1318 + 	case ARG_IN:
  1.1319 +	  if (!write_printf_to_iochannel ("const %s%s IN_%s", channel, error,
  1.1320 +					  type_str,
  1.1321 +					  type_suffix,
  1.1322 +					  arg_info_get_name (arg)))
  1.1323 + 	    goto io_lose;
  1.1324 + 	  break;
  1.1325 + 	case ARG_OUT:
  1.1326 +	  if (!write_printf_to_iochannel ("%s%s* OUT_%s", channel, error,
  1.1327 +					  type_str,
  1.1328 +					  type_suffix,
  1.1329 +					  arg_info_get_name (arg)))
  1.1330 + 	    goto io_lose;
  1.1331 + 	  break;
  1.1332 + 	case ARG_INVALID:
  1.1333 +	  break;
  1.1334 +	}
  1.1335 +    }
  1.1336 +  return TRUE;
  1.1337 + io_lose:
  1.1338 +  return FALSE;
  1.1339 +}
  1.1340 +
  1.1341 +static gboolean
  1.1342 +write_typed_args_for_direction (InterfaceInfo *iface, MethodInfo *method, GIOChannel *channel, const int direction, GError **error)
  1.1343 + {
  1.1344 +  GSList *args;
  1.1345 +  
  1.1346 +  for (args = method_info_get_args (method); args; args = args->next)
  1.1347 +    {
  1.1348 +      ArgInfo *arg;
  1.1349 +      int dir;
  1.1350 +      GType gtype;
  1.1351 +      const char *type_lookup;
  1.1352 +      
  1.1353 +      arg = args->data;
  1.1354 +
  1.1355 +      dir = arg_info_get_direction (arg);
  1.1356 +
  1.1357 +      if (dir != direction)
  1.1358 +        continue;
  1.1359 +
  1.1360 +      gtype = _dbus_gtype_from_signature (arg_info_get_type (arg), TRUE);
  1.1361 +      type_lookup = dbus_g_type_get_lookup_function (gtype);
  1.1362 +
  1.1363 +      if (!write_printf_to_iochannel ("%s, &%s_%s, ", channel, error, type_lookup, direction == ARG_IN ? "IN" : "OUT", arg_info_get_name (arg)))
  1.1364 +          goto io_lose;
  1.1365 +    }
  1.1366 +  return TRUE;
  1.1367 + io_lose:
  1.1368 +  return FALSE;
  1.1369 +}
  1.1370 +
  1.1371 +static gboolean
  1.1372 +write_async_method_client (GIOChannel *channel, InterfaceInfo *interface, MethodInfo *method, GError **error)
  1.1373 +{
  1.1374 +  char *method_name, *iface_prefix;
  1.1375 +  const char *interface_c_name;
  1.1376 +
  1.1377 +  iface_prefix = iface_to_c_prefix (interface_info_get_name (interface));
  1.1378 +  interface_c_name = interface_info_get_annotation (interface, DBUS_GLIB_ANNOTATION_CLIENT_C_SYMBOL);
  1.1379 +  if (interface_c_name == NULL)
  1.1380 +    {
  1.1381 +      interface_c_name = (const char *) iface_prefix;
  1.1382 +    }
  1.1383 +
  1.1384 +  method_name = g_strdup (method_info_get_annotation (method, DBUS_GLIB_ANNOTATION_CLIENT_C_SYMBOL));
  1.1385 +  if (method_name == NULL)
  1.1386 +    {
  1.1387 +      method_name = compute_client_method_name (interface_c_name, method);
  1.1388 +    }
  1.1389 +  g_free(iface_prefix);
  1.1390 +
  1.1391 +  /* Write the typedef for the client callback */
  1.1392 +  if (!write_printf_to_iochannel ("typedef void (*%s_reply) (DBusGProxy *proxy, ", channel, error, method_name))
  1.1393 +    goto io_lose;
  1.1394 +  {
  1.1395 +    GSList *args;
  1.1396 +    for (args = method_info_get_args (method); args; args = args->next)
  1.1397 +      {
  1.1398 +	ArgInfo *arg;
  1.1399 +	const char *type_suffix, *type_str;
  1.1400 +	GType gtype;
  1.1401 +	
  1.1402 +	arg = args->data;
  1.1403 +	
  1.1404 +	if (arg_info_get_direction (arg) != ARG_OUT)
  1.1405 +	  continue;
  1.1406 +	gtype = _dbus_gtype_from_signature (arg_info_get_type (arg), TRUE);
  1.1407 +	if (gtype != G_TYPE_VALUE && (g_type_is_a (gtype, G_TYPE_BOXED)
  1.1408 +	     || g_type_is_a (gtype, G_TYPE_OBJECT)
  1.1409 +	     || g_type_is_a (gtype, G_TYPE_POINTER)))
  1.1410 +	  type_suffix = "*";
  1.1411 +	else
  1.1412 +	  type_suffix = "";
  1.1413 +	type_str = dbus_g_type_get_c_name (_dbus_gtype_from_signature (arg_info_get_type (arg), TRUE));
  1.1414 +	if (!write_printf_to_iochannel ("%s %sOUT_%s, ", channel, error, type_str, type_suffix, arg_info_get_name (arg)))
  1.1415 +	  goto io_lose;
  1.1416 +      }
  1.1417 +  }
  1.1418 +  WRITE_OR_LOSE ("GError *error, gpointer userdata);\n\n");
  1.1419 +  
  1.1420 +  
  1.1421 +  /* Write the callback when the call returns */
  1.1422 +  WRITE_OR_LOSE ("static void\n");
  1.1423 +  if (!write_printf_to_iochannel ("%s_async_callback (DBusGProxy *proxy, DBusGProxyCall *call, void *user_data)\n", channel, error, method_name))
  1.1424 +    goto io_lose;
  1.1425 +  WRITE_OR_LOSE ("{\n");
  1.1426 +  WRITE_OR_LOSE ("  DBusGAsyncData *data = (DBusGAsyncData*) user_data;\n  GError *error = NULL;\n");
  1.1427 +  if (!write_formal_declarations_for_direction (interface, method, channel, ARG_OUT, error))
  1.1428 +    goto io_lose;
  1.1429 +  /* TODO: handle return boolean of end_call */
  1.1430 +  WRITE_OR_LOSE ("  dbus_g_proxy_end_call (proxy, call, &error, ");
  1.1431 +  if (!write_typed_args_for_direction (interface, method, channel, ARG_OUT, error))
  1.1432 +    goto io_lose;
  1.1433 +  WRITE_OR_LOSE("G_TYPE_INVALID);\n");
  1.1434 +  if (!write_printf_to_iochannel ("  (*(%s_reply)data->cb) (proxy, ", channel, error, method_name))
  1.1435 +    goto io_lose;
  1.1436 +  if (!write_untyped_out_args (interface, method, channel, error))
  1.1437 +    goto io_lose;
  1.1438 +  WRITE_OR_LOSE ("error, data->userdata);\n");
  1.1439 +  WRITE_OR_LOSE ("  return;\n}\n\n");
  1.1440 +  
  1.1441 +
  1.1442 +  /* Write the main wrapper function */
  1.1443 +  WRITE_OR_LOSE ("static\n#ifdef G_HAVE_INLINE\ninline\n#endif\nDBusGProxyCall*\n");
  1.1444 +  if (!write_printf_to_iochannel ("%s_async (DBusGProxy *proxy", channel, error,
  1.1445 +                                  method_name))
  1.1446 +    goto io_lose;
  1.1447 +  if (!write_formal_parameters_for_direction (interface, method, ARG_IN, channel, error))
  1.1448 +    goto io_lose;
  1.1449 +  
  1.1450 +  if (!write_printf_to_iochannel (", %s_reply callback, gpointer userdata)\n\n", channel, error, method_name))
  1.1451 +    goto io_lose;
  1.1452 +  
  1.1453 +  WRITE_OR_LOSE ("{\n");
  1.1454 +  WRITE_OR_LOSE ("  DBusGAsyncData *stuff;\n  stuff = g_new (DBusGAsyncData, 1);\n  stuff->cb = G_CALLBACK (callback);\n  stuff->userdata = userdata;\n");
  1.1455 +  if (!write_printf_to_iochannel ("  return dbus_g_proxy_begin_call (proxy, \"%s\", %s_async_callback, stuff, g_free, ", channel, error, method_info_get_name (method), method_name))
  1.1456 +    goto io_lose;
  1.1457 +  if (!write_args_for_direction (interface, method, channel, ARG_IN, error))
  1.1458 +    goto io_lose;
  1.1459 +  WRITE_OR_LOSE ("G_TYPE_INVALID);\n}\n");
  1.1460 +
  1.1461 +  g_free (method_name);
  1.1462 +  return TRUE;
  1.1463 + io_lose:
  1.1464 +  g_free (method_name);
  1.1465 +  return FALSE;
  1.1466 + }
  1.1467 +
  1.1468 +static gboolean
  1.1469 +generate_client_glue_list (GSList *list, DBusBindingToolCData *data, GError **error)
  1.1470 +{
  1.1471 +  GSList *tmp;
  1.1472 +
  1.1473 +  tmp = list;
  1.1474 +  while (tmp != NULL)
  1.1475 +    {
  1.1476 +      if (!generate_client_glue (tmp->data, data, error))
  1.1477 +	return FALSE;
  1.1478 +      tmp = tmp->next;
  1.1479 +    }
  1.1480 +  return TRUE;
  1.1481 +}
  1.1482 +
  1.1483 +static gboolean
  1.1484 +generate_client_glue (BaseInfo *base, DBusBindingToolCData *data, GError **error)
  1.1485 +{
  1.1486 +  if (base_info_get_type (base) == INFO_TYPE_NODE)
  1.1487 +    {
  1.1488 +      if (!generate_client_glue_list (node_info_get_nodes ((NodeInfo *) base),
  1.1489 +				      data, error))
  1.1490 +	return FALSE;
  1.1491 +      if (!generate_client_glue_list (node_info_get_interfaces ((NodeInfo *) base),
  1.1492 +				      data, error))
  1.1493 +	return FALSE;
  1.1494 +    }
  1.1495 +  else
  1.1496 +    {
  1.1497 +      GIOChannel *channel;
  1.1498 +      InterfaceInfo *interface;
  1.1499 +      GSList *methods;
  1.1500 +      GSList *tmp;
  1.1501 +      char *iface_prefix;
  1.1502 +      const char *interface_c_name;
  1.1503 +
  1.1504 +      channel = data->channel;
  1.1505 +
  1.1506 +      interface = (InterfaceInfo *) base;
  1.1507 +
  1.1508 +      methods = interface_info_get_methods (interface);
  1.1509 +
  1.1510 +      iface_prefix = iface_to_c_prefix (interface_info_get_name (interface));
  1.1511 +      interface_c_name = interface_info_get_annotation (interface, DBUS_GLIB_ANNOTATION_CLIENT_C_SYMBOL);
  1.1512 +      if (interface_c_name == NULL)
  1.1513 +      {
  1.1514 +          interface_c_name = (const char *) iface_prefix;
  1.1515 +      }
  1.1516 +
  1.1517 +      if (!write_printf_to_iochannel ("#ifndef DBUS_GLIB_CLIENT_WRAPPERS_%s\n"
  1.1518 +				      "#define DBUS_GLIB_CLIENT_WRAPPERS_%s\n\n",
  1.1519 +				      channel, error,
  1.1520 +				      iface_prefix, iface_prefix))
  1.1521 +	{
  1.1522 +	  g_free (iface_prefix);
  1.1523 +	  goto io_lose;
  1.1524 +	}
  1.1525 +
  1.1526 +      for (tmp = methods; tmp != NULL; tmp = g_slist_next (tmp))
  1.1527 +        {
  1.1528 +	  MethodInfo *method;
  1.1529 +	  char *method_c_name;
  1.1530 +	  gboolean is_noreply;
  1.1531 +
  1.1532 +          method = (MethodInfo *) tmp->data;
  1.1533 +	  method_c_name = g_strdup (method_info_get_annotation (method, DBUS_GLIB_ANNOTATION_CLIENT_C_SYMBOL));
  1.1534 +          if (method_c_name == NULL)
  1.1535 +	    {
  1.1536 +              method_c_name = compute_client_method_name (interface_c_name, method);
  1.1537 +            }
  1.1538 +
  1.1539 +	  is_noreply = method_info_get_annotation (method, DBUS_GLIB_ANNOTATION_NOREPLY) != NULL;
  1.1540 +
  1.1541 +	  if (data->ignore_unsupported && !check_supported_parameters (method))
  1.1542 +	    {
  1.1543 +	      g_warning ("Ignoring unsupported signature in method \"%s\" of interface \"%s\"\n",
  1.1544 +			 method_info_get_name (method),
  1.1545 +			 interface_info_get_name (interface));
  1.1546 +	      continue;
  1.1547 +	    }
  1.1548 +
  1.1549 +
  1.1550 +	  WRITE_OR_LOSE ("static\n#ifdef G_HAVE_INLINE\ninline\n#endif\ngboolean\n");
  1.1551 +	  if (!write_printf_to_iochannel ("%s (DBusGProxy *proxy", channel, error,
  1.1552 +					  method_c_name))
  1.1553 +	    goto io_lose;
  1.1554 +	  g_free (method_c_name);
  1.1555 +
  1.1556 +	  if (!write_formal_parameters (interface, method, channel, error))
  1.1557 +	    goto io_lose;
  1.1558 +
  1.1559 +	  WRITE_OR_LOSE (", GError **error)\n\n");
  1.1560 +	  
  1.1561 +	  WRITE_OR_LOSE ("{\n");
  1.1562 +
  1.1563 +	  if (is_noreply) {
  1.1564 +	    if (!write_printf_to_iochannel ("  dbus_g_proxy_call_no_reply (proxy, \"%s\", ", channel, error,
  1.1565 +					    method_info_get_name (method)))
  1.1566 +	      goto io_lose;
  1.1567 +	    
  1.1568 +	    if (!write_args_for_direction (interface, method, channel, ARG_IN, error))
  1.1569 +	      goto io_lose;
  1.1570 +	    
  1.1571 +	    WRITE_OR_LOSE ("G_TYPE_INVALID, ");
  1.1572 +	    
  1.1573 +	    if (!write_args_for_direction (interface, method, channel, ARG_OUT, error))
  1.1574 +	      goto io_lose;
  1.1575 +	    
  1.1576 +	    WRITE_OR_LOSE ("G_TYPE_INVALID);\n");
  1.1577 +	    
  1.1578 +	    WRITE_OR_LOSE ("  return TRUE;\n}\n\n");
  1.1579 +	  } else {
  1.1580 +	    if (!write_printf_to_iochannel ("  return dbus_g_proxy_call (proxy, \"%s\", ", channel, error,
  1.1581 +					    method_info_get_name (method)))
  1.1582 +	      goto io_lose;
  1.1583 +	    
  1.1584 +	    WRITE_OR_LOSE ("error, ");
  1.1585 +	    
  1.1586 +	    if (!write_args_for_direction (interface, method, channel, ARG_IN, error))
  1.1587 +	      goto io_lose;
  1.1588 +	    
  1.1589 +	    WRITE_OR_LOSE ("G_TYPE_INVALID, ");
  1.1590 +	    
  1.1591 +	    if (!write_args_for_direction (interface, method, channel, ARG_OUT, error))
  1.1592 +	      goto io_lose;
  1.1593 +	    
  1.1594 +	    WRITE_OR_LOSE ("G_TYPE_INVALID);\n}\n\n");
  1.1595 +	  }
  1.1596 +
  1.1597 +	  write_async_method_client (channel, interface, method, error);
  1.1598 +	}
  1.1599 +
  1.1600 +      if (!write_printf_to_iochannel ("#endif /* defined DBUS_GLIB_CLIENT_WRAPPERS_%s */\n\n", channel, error, iface_prefix))
  1.1601 +	{
  1.1602 +	  g_free (iface_prefix);
  1.1603 +	  goto io_lose;
  1.1604 +	}
  1.1605 +
  1.1606 +      g_free (iface_prefix);
  1.1607 +    }
  1.1608 +  return TRUE;
  1.1609 + io_lose:
  1.1610 +  return FALSE;
  1.1611 +}
  1.1612 +
  1.1613 +
  1.1614 +gboolean
  1.1615 +dbus_binding_tool_output_glib_client (BaseInfo *info, GIOChannel *channel, gboolean ignore_unsupported, GError **error)
  1.1616 +{
  1.1617 +  DBusBindingToolCData data;
  1.1618 +  gboolean ret;
  1.1619 +
  1.1620 +  memset (&data, 0, sizeof (data));
  1.1621 +  
  1.1622 +  data.channel = channel;
  1.1623 +  data.ignore_unsupported = ignore_unsupported;
  1.1624 +
  1.1625 +  dbus_g_type_specialized_init ();
  1.1626 +  _dbus_g_type_specialized_builtins_init ();
  1.1627 +
  1.1628 +  WRITE_OR_LOSE ("/* Generated by dbus-binding-tool; do not edit! */\n\n");
  1.1629 +  WRITE_OR_LOSE ("#include <glib/gtypes.h>\n");
  1.1630 +  WRITE_OR_LOSE ("#include <glib/gerror.h>\n");
  1.1631 +  WRITE_OR_LOSE ("#include <dbus/dbus-glib.h>\n\n");
  1.1632 +  WRITE_OR_LOSE ("G_BEGIN_DECLS\n\n");
  1.1633 +
  1.1634 +  ret = generate_client_glue (info, &data, error);
  1.1635 +  if (!ret)
  1.1636 +    goto io_lose;
  1.1637 +  
  1.1638 +  WRITE_OR_LOSE ("G_END_DECLS\n");
  1.1639 +
  1.1640 +  return ret;
  1.1641 + io_lose:
  1.1642 +  return FALSE;
  1.1643 +}