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 +}