Update contrib.
1 /* -*- mode: C; c-file-style: "gnu" -*- */
2 /* dbus-binding-tool-glib.c: Output C glue
4 * Copyright (C) 2003, 2004, 2005 Red Hat, Inc.
5 * Copyright (C) 2005 Nokia
6 * Portion Copyright © 2008 Nokia Corporation and/or its subsidiary(-ies). All rights reserved.
7 * Licensed under the Academic Free License version 2.1
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (at your option) any later version.
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
29 #endif //__SYMBIAN32__
30 #include "dbus/dbus-glib.h"
31 #include "dbus-gidl.h"
32 #include "dbus-gparser.h"
33 #include "dbus-gutils.h"
34 #include "dbus-gtype-specialized.h"
35 #include "dbus-gsignature.h"
36 #include "dbus-gvalue-utils.h"
37 #include "dbus-glib-tool.h"
38 #include "dbus-binding-tool-glib.h"
41 #include <glib/gi18n.h>
43 #define _(x) dgettext (GETTEXT_PACKAGE, x)
56 #define MARSHAL_PREFIX "dbus_glib_marshal_"
60 gboolean ignore_unsupported;
66 GHashTable *generated;
69 GString *property_blob;
71 } DBusBindingToolCData;
73 static gboolean gather_marshallers (BaseInfo *base, DBusBindingToolCData *data, GError **error);
74 static gboolean generate_glue (BaseInfo *base, DBusBindingToolCData *data, GError **error);
75 static gboolean generate_client_glue (BaseInfo *base, DBusBindingToolCData *data, GError **error);
78 dbus_g_type_get_marshal_name (GType gtype)
80 switch (G_TYPE_FUNDAMENTAL (gtype))
111 /* This entire function is kind of...ugh. */
113 dbus_g_type_get_c_name (GType gtype)
116 if (dbus_g_type_is_struct (gtype))
118 return "GValueArray";
120 if (dbus_g_type_is_collection (gtype))
122 subtype = dbus_g_type_get_collection_specialization(gtype);
123 if (_dbus_g_type_is_fixed (subtype))
129 if (dbus_g_type_is_map (gtype))
132 if (g_type_is_a (gtype, G_TYPE_STRING))
135 /* This one is even more hacky...we get an extra *
136 * because G_TYPE_STRV is a G_TYPE_BOXED
138 if (g_type_is_a (gtype, G_TYPE_STRV))
141 if (g_type_is_a (gtype, DBUS_TYPE_G_OBJECT_PATH))
144 return g_type_name (gtype);
148 compute_gsignature (MethodInfo *method, GType *rettype, GArray **params, GError **error)
154 const char *arg_type;
155 gboolean retval_signals_error;
157 is_async = method_info_get_annotation (method, DBUS_GLIB_ANNOTATION_ASYNC) != NULL;
158 retval_signals_error = FALSE;
160 ret = g_array_new (TRUE, TRUE, sizeof (GType));
163 retval_type = G_TYPE_NONE;
166 gboolean found_retval;
168 /* Look for return value */
169 found_retval = FALSE;
170 for (elt = method_info_get_args (method); elt; elt = elt->next)
172 ArgInfo *arg = elt->data;
173 const char *returnval_annotation;
175 returnval_annotation = arg_info_get_annotation (arg, DBUS_GLIB_ANNOTATION_RETURNVAL);
176 if (returnval_annotation != NULL)
178 arg_type = arg_info_get_type (arg);
179 retval_type = _dbus_gtype_from_signature (arg_type, FALSE);
180 if (retval_type == G_TYPE_INVALID)
183 if (!strcmp (returnval_annotation, "error"))
184 retval_signals_error = TRUE;
190 retval_type = G_TYPE_BOOLEAN;
191 retval_signals_error = TRUE;
195 *rettype = retval_type;
197 /* Handle all input arguments */
198 for (elt = method_info_get_args (method); elt; elt = elt->next)
200 ArgInfo *arg = elt->data;
201 if (arg_info_get_direction (arg) == ARG_IN)
205 arg_type = arg_info_get_type (arg);
206 gtype = _dbus_gtype_from_signature (arg_type, FALSE);
207 if (gtype == G_TYPE_INVALID)
210 g_array_append_val (ret, gtype);
216 /* Append pointer for each out arg storage */
217 for (elt = method_info_get_args (method); elt; elt = elt->next)
219 ArgInfo *arg = elt->data;
221 /* Skip return value */
222 if (arg_info_get_annotation (arg, DBUS_GLIB_ANNOTATION_RETURNVAL) != NULL)
225 if (arg_info_get_direction (arg) == ARG_OUT)
228 arg_type = arg_info_get_type (arg);
229 gtype = _dbus_gtype_from_signature (arg_type, FALSE);
230 if (gtype == G_TYPE_INVALID)
232 /* We actually just need a pointer for the return value
234 gtype = G_TYPE_POINTER;
235 g_array_append_val (ret, gtype);
239 if (retval_signals_error)
241 /* Final GError parameter */
242 GType gtype = G_TYPE_POINTER;
243 g_array_append_val (ret, gtype);
248 /* Context pointer */
249 GType gtype = G_TYPE_POINTER;
250 g_array_append_val (ret, gtype);
258 DBUS_BINDING_TOOL_ERROR,
259 DBUS_BINDING_TOOL_ERROR_UNSUPPORTED_CONVERSION,
260 _("Unsupported conversion from D-BUS type %s to glib-genmarshal type"),
267 compute_marshaller (MethodInfo *method, GError **error)
271 const char *marshal_name;
275 if (!compute_gsignature (method, &rettype, &signature, error))
278 ret = g_string_new ("");
279 marshal_name = dbus_g_type_get_marshal_name (rettype);
280 g_assert (marshal_name != NULL);
281 g_string_append (ret, marshal_name);
282 g_string_append_c (ret, ':');
283 for (i = 0; i < signature->len; i++)
285 marshal_name = dbus_g_type_get_marshal_name (g_array_index (signature, GType, i));
286 g_assert (marshal_name != NULL);
287 g_string_append (ret, marshal_name);
288 if (i < signature->len - 1)
289 g_string_append_c (ret, ',');
291 if (signature->len == 0)
293 marshal_name = dbus_g_type_get_marshal_name (G_TYPE_NONE);
294 g_assert (marshal_name != NULL);
295 g_string_append (ret, marshal_name);
297 g_array_free (signature, TRUE);
298 return g_string_free (ret, FALSE);
302 compute_marshaller_name (MethodInfo *method, const char *prefix, GError **error)
307 const char *marshal_name;
310 if (!compute_gsignature (method, &rettype, &signature, error))
313 ret = g_string_new (MARSHAL_PREFIX);
314 g_string_append (ret, prefix);
315 g_string_append_c (ret, '_');
317 marshal_name = dbus_g_type_get_marshal_name (rettype);
318 g_assert (marshal_name != NULL);
319 g_string_append (ret, marshal_name);
320 g_string_append (ret, "__");
321 for (i = 0; i < signature->len; i++)
323 marshal_name = dbus_g_type_get_marshal_name (g_array_index (signature, GType, i));
324 g_assert (marshal_name != NULL);
325 g_string_append (ret, marshal_name);
326 if (i < signature->len - 1)
327 g_string_append_c (ret, '_');
329 if (signature->len == 0)
331 marshal_name = dbus_g_type_get_marshal_name (G_TYPE_NONE);
332 g_assert (marshal_name != NULL);
333 g_string_append (ret, marshal_name);
335 g_array_free (signature, TRUE);
336 return g_string_free (ret, FALSE);
340 gather_marshallers_list (GSList *list, DBusBindingToolCData *data, GError **error)
347 if (!gather_marshallers (tmp->data, data, error))
355 gather_marshallers (BaseInfo *base, DBusBindingToolCData *data, GError **error)
357 if (base_info_get_type (base) == INFO_TYPE_NODE)
359 if (!gather_marshallers_list (node_info_get_nodes ((NodeInfo *) base),
362 if (!gather_marshallers_list (node_info_get_interfaces ((NodeInfo *) base),
368 InterfaceInfo *interface;
371 const char *interface_c_name;
373 interface = (InterfaceInfo *) base;
374 interface_c_name = interface_info_get_annotation (interface, DBUS_GLIB_ANNOTATION_C_SYMBOL);
375 if (interface_c_name == NULL)
381 methods = interface_info_get_methods (interface);
383 /* Generate the necessary marshallers for the methods. */
385 for (tmp = methods; tmp != NULL; tmp = g_slist_next (tmp))
388 char *marshaller_name;
390 method = (MethodInfo *) tmp->data;
392 marshaller_name = compute_marshaller (method, error);
393 if (!marshaller_name)
396 if (g_hash_table_lookup (data->generated, marshaller_name))
398 g_free (marshaller_name);
402 g_hash_table_insert (data->generated, marshaller_name, NULL);
410 generate_glue_list (GSList *list, DBusBindingToolCData *data, GError **error)
417 if (!generate_glue (tmp->data, data, error))
424 #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)
427 write_printf_to_iochannel (const char *fmt, GIOChannel *channel, GError **error, ...)
435 va_start (args, error);
437 str = g_strdup_vprintf (fmt, args);
438 if ((status = g_io_channel_write_chars (channel, str, -1, &written, error)) == G_IO_STATUS_NORMAL)
451 write_quoted_string (GIOChannel *channel, GString *string, GError **error)
455 WRITE_OR_LOSE ("\"");
456 for (i = 0; i < string->len; i++)
458 if (string->str[i] != '\0')
460 if (!g_io_channel_write_chars (channel, string->str + i, 1, NULL, error))
465 if (!g_io_channel_write_chars (channel, "\\0", -1, NULL, error))
469 WRITE_OR_LOSE ("\\0\"");
476 generate_glue (BaseInfo *base, DBusBindingToolCData *data, GError **error)
478 if (base_info_get_type (base) == INFO_TYPE_NODE)
480 GString *object_introspection_data_blob;
483 channel = data->channel;
485 object_introspection_data_blob = g_string_new_len ("", 0);
487 data->blob = object_introspection_data_blob;
490 data->signal_blob = g_string_new_len ("", 0);
491 data->property_blob = g_string_new_len ("", 0);
493 if (!write_printf_to_iochannel ("static const DBusGMethodInfo dbus_glib_%s_methods[] = {\n", channel, error, data->prefix))
496 if (!generate_glue_list (node_info_get_nodes ((NodeInfo *) base),
499 if (!generate_glue_list (node_info_get_interfaces ((NodeInfo *) base),
503 WRITE_OR_LOSE ("};\n\n");
505 /* Information about the object. */
507 if (!write_printf_to_iochannel ("const DBusGObjectInfo dbus_glib_%s_object_info = {\n",
508 channel, error, data->prefix))
510 WRITE_OR_LOSE (" 0,\n");
511 if (!write_printf_to_iochannel (" dbus_glib_%s_methods,\n", channel, error, data->prefix))
513 if (!write_printf_to_iochannel (" %d,\n", channel, error, data->count))
516 if (!write_quoted_string (channel, object_introspection_data_blob, error))
518 WRITE_OR_LOSE (",\n");
519 if (!write_quoted_string (channel, data->signal_blob, error))
521 WRITE_OR_LOSE (",\n");
522 if (!write_quoted_string (channel, data->property_blob, error))
524 WRITE_OR_LOSE ("\n};\n\n");
526 g_string_free (object_introspection_data_blob, TRUE);
527 g_string_free (data->signal_blob, TRUE);
528 g_string_free (data->property_blob, TRUE);
533 InterfaceInfo *interface;
538 const char *interface_c_name;
539 GString *object_introspection_data_blob;
541 channel = data->channel;
542 object_introspection_data_blob = data->blob;
544 interface = (InterfaceInfo *) base;
545 interface_c_name = interface_info_get_annotation (interface, DBUS_GLIB_ANNOTATION_C_SYMBOL);
546 if (interface_c_name == NULL)
548 if (data->prefix == NULL)
550 interface_c_name = data->prefix;
553 methods = interface_info_get_methods (interface);
555 /* Table of marshalled methods. */
557 for (tmp = methods; tmp != NULL; tmp = g_slist_next (tmp))
560 char *marshaller_name;
562 gboolean async = FALSE;
564 gboolean found_retval = FALSE;
566 method = (MethodInfo *) tmp->data;
567 method_c_name = g_strdup (method_info_get_annotation (method, DBUS_GLIB_ANNOTATION_C_SYMBOL));
568 if (method_c_name == NULL)
570 char *method_name_uscored;
571 method_name_uscored = _dbus_gutils_wincaps_to_uscore (method_info_get_name (method));
572 method_c_name = g_strdup_printf ("%s_%s",
574 method_name_uscored);
575 g_free (method_name_uscored);
578 if (!write_printf_to_iochannel (" { (GCallback) %s, ", channel, error,
582 marshaller_name = compute_marshaller_name (method, data->prefix, error);
583 if (!marshaller_name)
586 if (!write_printf_to_iochannel ("%s, %d },\n", channel, error,
588 object_introspection_data_blob->len))
590 g_free (marshaller_name);
594 if (method_info_get_annotation (method, DBUS_GLIB_ANNOTATION_ASYNC) != NULL)
597 /* Object method data blob format:
598 * <iface>\0<name>\0(<argname>\0<argdirection>\0<argtype>\0)*\0
601 g_string_append (object_introspection_data_blob, interface_info_get_name (interface));
602 g_string_append_c (object_introspection_data_blob, '\0');
604 g_string_append (object_introspection_data_blob, method_info_get_name (method));
605 g_string_append_c (object_introspection_data_blob, '\0');
607 g_string_append_c (object_introspection_data_blob, async ? 'A' : 'S');
608 g_string_append_c (object_introspection_data_blob, '\0');
610 for (args = method_info_get_args (method); args; args = args->next)
614 const char *returnval_annotation;
618 g_string_append (object_introspection_data_blob, arg_info_get_name (arg));
619 g_string_append_c (object_introspection_data_blob, '\0');
621 switch (arg_info_get_direction (arg))
631 g_assert_not_reached ();
632 direction = 0; /* silence gcc */
635 g_string_append_c (object_introspection_data_blob, direction);
636 g_string_append_c (object_introspection_data_blob, '\0');
638 if (arg_info_get_annotation (arg, DBUS_GLIB_ANNOTATION_CONST) != NULL)
640 if (arg_info_get_direction (arg) == ARG_IN)
643 DBUS_BINDING_TOOL_ERROR,
644 DBUS_BINDING_TOOL_ERROR_INVALID_ANNOTATION,
645 "Input argument \"%s\" cannot have const annotation in method \"%s\" of interface \"%s\"\n",
646 arg_info_get_name (arg),
647 method_info_get_name (method),
648 interface_info_get_name (interface));
651 g_string_append_c (object_introspection_data_blob, 'C');
652 g_string_append_c (object_introspection_data_blob, '\0');
654 else if (arg_info_get_direction (arg) == ARG_OUT)
656 g_string_append_c (object_introspection_data_blob, 'F');
657 g_string_append_c (object_introspection_data_blob, '\0');
660 returnval_annotation = arg_info_get_annotation (arg, DBUS_GLIB_ANNOTATION_RETURNVAL);
661 if (returnval_annotation != NULL)
668 DBUS_BINDING_TOOL_ERROR,
669 DBUS_BINDING_TOOL_ERROR_INVALID_ANNOTATION,
670 "Multiple arguments with return value annotation in method \"%s\" of interface \"%s\"\n",
671 method_info_get_name (method),
672 interface_info_get_name (interface));
676 if (arg_info_get_direction (arg) == ARG_IN)
679 DBUS_BINDING_TOOL_ERROR,
680 DBUS_BINDING_TOOL_ERROR_INVALID_ANNOTATION,
681 "Input argument \"%s\" cannot have return value annotation in method \"%s\" of interface \"%s\"\n",
682 arg_info_get_name (arg),
683 method_info_get_name (method),
684 interface_info_get_name (interface));
687 if (!strcmp ("", returnval_annotation))
688 g_string_append_c (object_introspection_data_blob, 'R');
689 else if (!strcmp ("error", returnval_annotation))
691 gtype = _dbus_gtype_from_signature (arg_info_get_type (arg), TRUE);
692 if (!_dbus_gtype_can_signal_error (gtype))
695 DBUS_BINDING_TOOL_ERROR,
696 DBUS_BINDING_TOOL_ERROR_INVALID_ANNOTATION,
697 "Output argument \"%s\" cannot signal error with type \"%s\" in method \"%s\" of interface \"%s\"\n",
698 arg_info_get_name (arg),
700 method_info_get_name (method),
701 interface_info_get_name (interface));
704 g_string_append_c (object_introspection_data_blob, 'E');
709 DBUS_BINDING_TOOL_ERROR,
710 DBUS_BINDING_TOOL_ERROR_INVALID_ANNOTATION,
711 "Invalid ReturnVal annotation for argument \"%s\" in method \"%s\" of interface \"%s\"\n",
712 arg_info_get_name (arg),
713 method_info_get_name (method),
714 interface_info_get_name (interface));
718 g_string_append_c (object_introspection_data_blob, '\0');
720 else if (arg_info_get_direction (arg) == ARG_OUT)
722 g_string_append_c (object_introspection_data_blob, 'N');
723 g_string_append_c (object_introspection_data_blob, '\0');
726 g_string_append (object_introspection_data_blob, arg_info_get_type (arg));
727 g_string_append_c (object_introspection_data_blob, '\0');
730 g_string_append_c (object_introspection_data_blob, '\0');
735 signals = interface_info_get_signals (interface);
737 for (tmp = signals; tmp != NULL; tmp = g_slist_next (tmp))
743 g_string_append (data->signal_blob, interface_info_get_name (interface));
744 g_string_append_c (data->signal_blob, '\0');
745 g_string_append (data->signal_blob, signal_info_get_name (sig));
746 g_string_append_c (data->signal_blob, '\0');
749 properties = interface_info_get_properties (interface);
751 for (tmp = properties; tmp != NULL; tmp = g_slist_next (tmp))
757 g_string_append (data->property_blob, interface_info_get_name (interface));
758 g_string_append_c (data->property_blob, '\0');
759 g_string_append (data->property_blob, property_info_get_name (prop));
760 g_string_append_c (data->property_blob, '\0');
769 write_marshaller (gpointer key, gpointer value, gpointer user_data)
771 DBusBindingToolCData *data;
772 const char *marshaller;
778 if (data->error && *data->error)
781 if (g_io_channel_write_chars (data->channel, marshaller, -1, &bytes_written, data->error) == G_IO_STATUS_NORMAL)
782 g_io_channel_write_chars (data->channel, "\n", -1, &bytes_written, data->error);
786 dbus_binding_tool_output_glib_server (BaseInfo *info, GIOChannel *channel, const char *prefix, GError **error)
791 GIOChannel *genmarshal_stdout;
793 DBusBindingToolCData data;
798 gsize bytes_read, bytes_written;
800 memset (&data, 0, sizeof (data));
802 dbus_g_type_specialized_init ();
803 _dbus_g_type_specialized_builtins_init ();
805 data.prefix = prefix;
806 data.generated = g_hash_table_new_full (g_str_hash, g_str_equal, (GDestroyNotify) g_free, NULL);
808 genmarshal_stdout = NULL;
809 tempfile_name = NULL;
811 if (!gather_marshallers (info, &data, error))
814 tempfile_fd = g_file_open_tmp ("dbus-binding-tool-c-marshallers.XXXXXX",
815 &tempfile_name, error);
819 data.channel = g_io_channel_unix_new (tempfile_fd);
820 if (!g_io_channel_set_encoding (data.channel, NULL, error))
822 g_hash_table_foreach (data.generated, write_marshaller, &data);
823 if (error && *error != NULL)
826 g_io_channel_close (data.channel);
827 g_io_channel_unref (data.channel);
831 g_io_channel_close (data.channel);
832 g_io_channel_unref (data.channel);
834 /* Now spawn glib-genmarshal to insert all our required marshallers */
835 argv = g_ptr_array_new ();
836 g_ptr_array_add (argv, "glib-genmarshal");
837 g_ptr_array_add (argv, "--header");
838 g_ptr_array_add (argv, "--body");
839 g_ptr_array_add (argv, g_strdup_printf ("--prefix=%s%s", MARSHAL_PREFIX, prefix));
840 g_ptr_array_add (argv, tempfile_name);
841 g_ptr_array_add (argv, NULL);
842 if (!g_spawn_async_with_pipes (NULL, (char**)argv->pdata, NULL,
847 &child_stdout, NULL, error))
849 g_ptr_array_free (argv, TRUE);
852 g_ptr_array_free (argv, TRUE);
854 genmarshal_stdout = g_io_channel_unix_new (child_stdout);
855 if (!g_io_channel_set_encoding (genmarshal_stdout, NULL, error))
858 WRITE_OR_LOSE ("/* Generated by dbus-binding-tool; do not edit! */\n\n");
860 while ((iostatus = g_io_channel_read_chars (genmarshal_stdout, buf, sizeof (buf),
861 &bytes_read, error)) == G_IO_STATUS_NORMAL)
862 if (g_io_channel_write_chars (channel, buf, bytes_read, &bytes_written, error) != G_IO_STATUS_NORMAL)
864 if (iostatus != G_IO_STATUS_EOF)
867 g_io_channel_close (genmarshal_stdout);
869 WRITE_OR_LOSE ("#include <dbus/dbus-glib.h>\n");
871 data.channel = channel;
872 g_io_channel_ref (data.channel);
873 if (!generate_glue (info, &data, error))
879 unlink (tempfile_name);
880 g_free (tempfile_name);
881 if (genmarshal_stdout)
882 g_io_channel_unref (genmarshal_stdout);
884 g_io_channel_unref (data.channel);
885 g_hash_table_destroy (data.generated);
894 iface_to_c_prefix (const char *iface)
901 components = g_strsplit (iface, ".", 0);
904 ret = g_string_new ("");
905 for (component = components; *component; component++)
908 g_string_append_c (ret, '_');
911 g_string_append (ret, *component);
913 g_strfreev (components);
914 return g_string_free (ret, FALSE);
918 compute_client_method_name (const char *iface_prefix, MethodInfo *method)
920 char *method_name_uscored, *ret;
922 method_name_uscored = _dbus_gutils_wincaps_to_uscore (method_info_get_name (method));
923 ret = g_strdup_printf ("%s_%s", iface_prefix, method_name_uscored);
924 g_free (method_name_uscored);
930 write_formal_parameters (InterfaceInfo *iface, MethodInfo *method, GIOChannel *channel, GError **error)
934 for (args = method_info_get_args (method); args; args = args->next)
937 const char *type_str;
938 const char *type_suffix;
944 WRITE_OR_LOSE (", ");
946 direction = arg_info_get_direction (arg);
948 gtype = _dbus_gtype_from_signature (arg_info_get_type (arg), TRUE);
949 if (gtype == G_TYPE_INVALID)
952 DBUS_BINDING_TOOL_ERROR,
953 DBUS_BINDING_TOOL_ERROR_UNSUPPORTED_CONVERSION,
954 _("Unsupported conversion from D-BUS type signature \"%s\" to glib C type in method \"%s\" of interface \"%s\""),
955 arg_info_get_type (arg),
956 method_info_get_name (method),
957 interface_info_get_name (iface));
960 type_str = dbus_g_type_get_c_name (gtype);
962 /* Variants are special...*/
963 if (gtype == G_TYPE_VALUE)
965 if (direction == ARG_IN)
970 else if ((g_type_is_a (gtype, G_TYPE_BOXED)
971 || g_type_is_a (gtype, G_TYPE_OBJECT)
972 || g_type_is_a (gtype, G_TYPE_POINTER)))
981 if (!write_printf_to_iochannel ("const %s%s IN_%s", channel, error,
984 arg_info_get_name (arg)))
988 if (!write_printf_to_iochannel ("%s%s* OUT_%s", channel, error,
991 arg_info_get_name (arg)))
1004 #define MAP_FUNDAMENTAL(NAME) \
1005 case G_TYPE_ ## NAME: \
1006 return g_strdup ("G_TYPE_" #NAME);
1007 #define MAP_KNOWN(NAME) \
1008 if (gtype == NAME) \
1009 return g_strdup (#NAME)
1011 dbus_g_type_get_lookup_function (GType gtype)
1016 MAP_FUNDAMENTAL(CHAR);
1017 MAP_FUNDAMENTAL(UCHAR);
1018 MAP_FUNDAMENTAL(BOOLEAN);
1019 MAP_FUNDAMENTAL(LONG);
1020 MAP_FUNDAMENTAL(ULONG);
1021 MAP_FUNDAMENTAL(INT);
1022 MAP_FUNDAMENTAL(UINT);
1023 MAP_FUNDAMENTAL(INT64);
1024 MAP_FUNDAMENTAL(UINT64);
1025 MAP_FUNDAMENTAL(FLOAT);
1026 MAP_FUNDAMENTAL(DOUBLE);
1027 MAP_FUNDAMENTAL(STRING);
1029 if (dbus_g_type_is_collection (gtype))
1034 elt_gtype = dbus_g_type_get_collection_specialization (gtype);
1035 sublookup = dbus_g_type_get_lookup_function (elt_gtype);
1036 g_assert (sublookup);
1038 if (_dbus_g_type_is_fixed (elt_gtype))
1040 type_lookup = g_strdup_printf ("dbus_g_type_get_collection "
1041 "(\"GArray\", %s)", sublookup);
1045 type_lookup = g_strdup_printf ("dbus_g_type_get_collection "
1046 "(\"GPtrArray\", %s)", sublookup);
1053 else if (dbus_g_type_is_map (gtype))
1060 key_gtype = dbus_g_type_get_map_key_specialization (gtype);
1061 value_gtype = dbus_g_type_get_map_value_specialization (gtype);
1062 key_lookup = dbus_g_type_get_lookup_function (key_gtype);
1063 g_assert (key_lookup);
1064 value_lookup = dbus_g_type_get_lookup_function (value_gtype);
1065 g_assert (value_lookup);
1066 type_lookup = g_strdup_printf ("dbus_g_type_get_map (\"GHashTable\", %s, %s)",
1067 key_lookup, value_lookup);
1068 g_free (key_lookup);
1069 g_free (value_lookup);
1072 else if (dbus_g_type_is_struct (gtype))
1076 char *value_lookup = NULL;
1079 string = g_string_new ("dbus_g_type_get_struct (\"GValueArray\"");
1081 size = dbus_g_type_get_struct_size (gtype);
1082 for (i=0; i < size; i++)
1084 value_gtype = dbus_g_type_get_struct_member_type(gtype, i);
1085 value_lookup = dbus_g_type_get_lookup_function (value_gtype);
1086 g_assert (value_lookup);
1087 g_string_append_printf (string, ", %s", value_lookup);
1088 g_free (value_lookup);
1090 g_string_append (string, ", G_TYPE_INVALID)");
1091 return g_string_free (string, FALSE);
1094 MAP_KNOWN(G_TYPE_VALUE);
1095 MAP_KNOWN(G_TYPE_STRV);
1096 MAP_KNOWN(G_TYPE_VALUE_ARRAY);
1097 MAP_KNOWN(DBUS_TYPE_G_PROXY);
1098 MAP_KNOWN(DBUS_TYPE_G_OBJECT_PATH);
1101 #undef MAP_FUNDAMENTAL
1105 write_args_for_direction (InterfaceInfo *iface, MethodInfo *method, GIOChannel *channel, int direction, GError **error)
1109 for (args = method_info_get_args (method); args; args = args->next)
1117 if (direction != arg_info_get_direction (arg))
1120 gtype = _dbus_gtype_from_signature (arg_info_get_type (arg), TRUE);
1121 g_assert (gtype != G_TYPE_INVALID);
1122 type_lookup = dbus_g_type_get_lookup_function (gtype);
1123 g_assert (type_lookup != NULL);
1129 if (!write_printf_to_iochannel ("%s, IN_%s, ", channel, error,
1131 arg_info_get_name (arg)))
1135 if (!write_printf_to_iochannel ("%s, OUT_%s, ", channel, error,
1137 arg_info_get_name (arg)))
1143 g_free (type_lookup);
1152 check_supported_parameters (MethodInfo *method)
1156 for (args = method_info_get_args (method); args; args = args->next)
1162 gtype = _dbus_gtype_from_signature (arg_info_get_type (arg), TRUE);
1163 if (gtype == G_TYPE_INVALID)
1170 write_untyped_out_args (InterfaceInfo *iface, MethodInfo *method, GIOChannel *channel, GError **error)
1174 for (args = method_info_get_args (method); args; args = args->next)
1179 if (arg_info_get_direction (arg) != ARG_OUT)
1182 if (!write_printf_to_iochannel ("OUT_%s, ", channel, error,
1183 arg_info_get_name (arg)))
1193 write_formal_declarations_for_direction (InterfaceInfo *iface, MethodInfo *method, GIOChannel *channel, const int direction, GError **error)
1197 for (args = method_info_get_args (method); args; args = args->next)
1201 const char *type_str, *type_suffix;
1206 dir = arg_info_get_direction (arg);
1208 gtype = _dbus_gtype_from_signature (arg_info_get_type (arg), TRUE);
1209 type_str = dbus_g_type_get_c_name (gtype);
1214 DBUS_BINDING_TOOL_ERROR,
1215 DBUS_BINDING_TOOL_ERROR_UNSUPPORTED_CONVERSION,
1216 _("Unsupported conversion from D-BUS type signature \"%s\" to glib C type in method \"%s\" of interface \"%s\""),
1217 arg_info_get_type (arg),
1218 method_info_get_name (method),
1219 interface_info_get_name (iface));
1223 /* Variants are special...*/
1224 if (gtype == G_TYPE_VALUE)
1226 if (direction == ARG_IN)
1231 else if ((g_type_is_a (gtype, G_TYPE_BOXED)
1232 || g_type_is_a (gtype, G_TYPE_OBJECT)
1233 || g_type_is_a (gtype, G_TYPE_POINTER)))
1238 if (direction != dir)
1244 if (!write_printf_to_iochannel (" %s%s IN_%s;\n", channel, error,
1245 type_str, type_suffix,
1246 arg_info_get_name (arg)))
1250 if (!write_printf_to_iochannel (" %s%s OUT_%s;\n", channel, error,
1251 type_str, type_suffix,
1252 arg_info_get_name (arg)))
1265 write_formal_parameters_for_direction (InterfaceInfo *iface, MethodInfo *method, int dir, GIOChannel *channel, GError **error)
1269 for (args = method_info_get_args (method); args; args = args->next)
1272 const char *type_str;
1273 const char *type_suffix;
1279 direction = arg_info_get_direction (arg);
1280 if (dir != direction) continue;
1282 WRITE_OR_LOSE (", ");
1284 gtype = _dbus_gtype_from_signature (arg_info_get_type (arg), TRUE);
1285 type_str = dbus_g_type_get_c_name (gtype);
1286 /* Variants are special...*/
1287 if (gtype == G_TYPE_VALUE)
1289 if (direction == ARG_IN)
1294 else if ((g_type_is_a (gtype, G_TYPE_BOXED)
1295 || g_type_is_a (gtype, G_TYPE_OBJECT)
1296 || g_type_is_a (gtype, G_TYPE_POINTER)))
1304 DBUS_BINDING_TOOL_ERROR,
1305 DBUS_BINDING_TOOL_ERROR_UNSUPPORTED_CONVERSION,
1306 _("Unsupported conversion from D-BUS type signature \"%s\" to glib C type in method \"%s\" of interface \"%s\""),
1307 arg_info_get_type (arg),
1308 method_info_get_name (method),
1309 interface_info_get_name (iface));
1316 if (!write_printf_to_iochannel ("const %s%s IN_%s", channel, error,
1319 arg_info_get_name (arg)))
1323 if (!write_printf_to_iochannel ("%s%s* OUT_%s", channel, error,
1326 arg_info_get_name (arg)))
1339 write_typed_args_for_direction (InterfaceInfo *iface, MethodInfo *method, GIOChannel *channel, const int direction, GError **error)
1343 for (args = method_info_get_args (method); args; args = args->next)
1348 const char *type_lookup;
1352 dir = arg_info_get_direction (arg);
1354 if (dir != direction)
1357 gtype = _dbus_gtype_from_signature (arg_info_get_type (arg), TRUE);
1358 type_lookup = dbus_g_type_get_lookup_function (gtype);
1360 if (!write_printf_to_iochannel ("%s, &%s_%s, ", channel, error, type_lookup, direction == ARG_IN ? "IN" : "OUT", arg_info_get_name (arg)))
1369 write_async_method_client (GIOChannel *channel, InterfaceInfo *interface, MethodInfo *method, GError **error)
1371 char *method_name, *iface_prefix;
1372 const char *interface_c_name;
1374 iface_prefix = iface_to_c_prefix (interface_info_get_name (interface));
1375 interface_c_name = interface_info_get_annotation (interface, DBUS_GLIB_ANNOTATION_CLIENT_C_SYMBOL);
1376 if (interface_c_name == NULL)
1378 interface_c_name = (const char *) iface_prefix;
1381 method_name = g_strdup (method_info_get_annotation (method, DBUS_GLIB_ANNOTATION_CLIENT_C_SYMBOL));
1382 if (method_name == NULL)
1384 method_name = compute_client_method_name (interface_c_name, method);
1386 g_free(iface_prefix);
1388 /* Write the typedef for the client callback */
1389 if (!write_printf_to_iochannel ("typedef void (*%s_reply) (DBusGProxy *proxy, ", channel, error, method_name))
1393 for (args = method_info_get_args (method); args; args = args->next)
1396 const char *type_suffix, *type_str;
1401 if (arg_info_get_direction (arg) != ARG_OUT)
1403 gtype = _dbus_gtype_from_signature (arg_info_get_type (arg), TRUE);
1404 if (gtype != G_TYPE_VALUE && (g_type_is_a (gtype, G_TYPE_BOXED)
1405 || g_type_is_a (gtype, G_TYPE_OBJECT)
1406 || g_type_is_a (gtype, G_TYPE_POINTER)))
1410 type_str = dbus_g_type_get_c_name (_dbus_gtype_from_signature (arg_info_get_type (arg), TRUE));
1411 if (!write_printf_to_iochannel ("%s %sOUT_%s, ", channel, error, type_str, type_suffix, arg_info_get_name (arg)))
1415 WRITE_OR_LOSE ("GError *error, gpointer userdata);\n\n");
1418 /* Write the callback when the call returns */
1419 WRITE_OR_LOSE ("static void\n");
1420 if (!write_printf_to_iochannel ("%s_async_callback (DBusGProxy *proxy, DBusGProxyCall *call, void *user_data)\n", channel, error, method_name))
1422 WRITE_OR_LOSE ("{\n");
1423 WRITE_OR_LOSE (" DBusGAsyncData *data = (DBusGAsyncData*) user_data;\n GError *error = NULL;\n");
1424 if (!write_formal_declarations_for_direction (interface, method, channel, ARG_OUT, error))
1426 /* TODO: handle return boolean of end_call */
1427 WRITE_OR_LOSE (" dbus_g_proxy_end_call (proxy, call, &error, ");
1428 if (!write_typed_args_for_direction (interface, method, channel, ARG_OUT, error))
1430 WRITE_OR_LOSE("G_TYPE_INVALID);\n");
1431 if (!write_printf_to_iochannel (" (*(%s_reply)data->cb) (proxy, ", channel, error, method_name))
1433 if (!write_untyped_out_args (interface, method, channel, error))
1435 WRITE_OR_LOSE ("error, data->userdata);\n");
1436 WRITE_OR_LOSE (" return;\n}\n\n");
1439 /* Write the main wrapper function */
1440 WRITE_OR_LOSE ("static\n#ifdef G_HAVE_INLINE\ninline\n#endif\nDBusGProxyCall*\n");
1441 if (!write_printf_to_iochannel ("%s_async (DBusGProxy *proxy", channel, error,
1444 if (!write_formal_parameters_for_direction (interface, method, ARG_IN, channel, error))
1447 if (!write_printf_to_iochannel (", %s_reply callback, gpointer userdata)\n\n", channel, error, method_name))
1450 WRITE_OR_LOSE ("{\n");
1451 WRITE_OR_LOSE (" DBusGAsyncData *stuff;\n stuff = g_new (DBusGAsyncData, 1);\n stuff->cb = G_CALLBACK (callback);\n stuff->userdata = userdata;\n");
1452 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))
1454 if (!write_args_for_direction (interface, method, channel, ARG_IN, error))
1456 WRITE_OR_LOSE ("G_TYPE_INVALID);\n}\n");
1458 g_free (method_name);
1461 g_free (method_name);
1466 generate_client_glue_list (GSList *list, DBusBindingToolCData *data, GError **error)
1473 if (!generate_client_glue (tmp->data, data, error))
1481 generate_client_glue (BaseInfo *base, DBusBindingToolCData *data, GError **error)
1483 if (base_info_get_type (base) == INFO_TYPE_NODE)
1485 if (!generate_client_glue_list (node_info_get_nodes ((NodeInfo *) base),
1488 if (!generate_client_glue_list (node_info_get_interfaces ((NodeInfo *) base),
1494 GIOChannel *channel;
1495 InterfaceInfo *interface;
1499 const char *interface_c_name;
1501 channel = data->channel;
1503 interface = (InterfaceInfo *) base;
1505 methods = interface_info_get_methods (interface);
1507 iface_prefix = iface_to_c_prefix (interface_info_get_name (interface));
1508 interface_c_name = interface_info_get_annotation (interface, DBUS_GLIB_ANNOTATION_CLIENT_C_SYMBOL);
1509 if (interface_c_name == NULL)
1511 interface_c_name = (const char *) iface_prefix;
1514 if (!write_printf_to_iochannel ("#ifndef DBUS_GLIB_CLIENT_WRAPPERS_%s\n"
1515 "#define DBUS_GLIB_CLIENT_WRAPPERS_%s\n\n",
1517 iface_prefix, iface_prefix))
1519 g_free (iface_prefix);
1523 for (tmp = methods; tmp != NULL; tmp = g_slist_next (tmp))
1526 char *method_c_name;
1527 gboolean is_noreply;
1529 method = (MethodInfo *) tmp->data;
1530 method_c_name = g_strdup (method_info_get_annotation (method, DBUS_GLIB_ANNOTATION_CLIENT_C_SYMBOL));
1531 if (method_c_name == NULL)
1533 method_c_name = compute_client_method_name (interface_c_name, method);
1536 is_noreply = method_info_get_annotation (method, DBUS_GLIB_ANNOTATION_NOREPLY) != NULL;
1538 if (data->ignore_unsupported && !check_supported_parameters (method))
1540 g_warning ("Ignoring unsupported signature in method \"%s\" of interface \"%s\"\n",
1541 method_info_get_name (method),
1542 interface_info_get_name (interface));
1547 WRITE_OR_LOSE ("static\n#ifdef G_HAVE_INLINE\ninline\n#endif\ngboolean\n");
1548 if (!write_printf_to_iochannel ("%s (DBusGProxy *proxy", channel, error,
1551 g_free (method_c_name);
1553 if (!write_formal_parameters (interface, method, channel, error))
1556 WRITE_OR_LOSE (", GError **error)\n\n");
1558 WRITE_OR_LOSE ("{\n");
1561 if (!write_printf_to_iochannel (" dbus_g_proxy_call_no_reply (proxy, \"%s\", ", channel, error,
1562 method_info_get_name (method)))
1565 if (!write_args_for_direction (interface, method, channel, ARG_IN, error))
1568 WRITE_OR_LOSE ("G_TYPE_INVALID, ");
1570 if (!write_args_for_direction (interface, method, channel, ARG_OUT, error))
1573 WRITE_OR_LOSE ("G_TYPE_INVALID);\n");
1575 WRITE_OR_LOSE (" return TRUE;\n}\n\n");
1577 if (!write_printf_to_iochannel (" return dbus_g_proxy_call (proxy, \"%s\", ", channel, error,
1578 method_info_get_name (method)))
1581 WRITE_OR_LOSE ("error, ");
1583 if (!write_args_for_direction (interface, method, channel, ARG_IN, error))
1586 WRITE_OR_LOSE ("G_TYPE_INVALID, ");
1588 if (!write_args_for_direction (interface, method, channel, ARG_OUT, error))
1591 WRITE_OR_LOSE ("G_TYPE_INVALID);\n}\n\n");
1594 write_async_method_client (channel, interface, method, error);
1597 if (!write_printf_to_iochannel ("#endif /* defined DBUS_GLIB_CLIENT_WRAPPERS_%s */\n\n", channel, error, iface_prefix))
1599 g_free (iface_prefix);
1603 g_free (iface_prefix);
1612 dbus_binding_tool_output_glib_client (BaseInfo *info, GIOChannel *channel, gboolean ignore_unsupported, GError **error)
1614 DBusBindingToolCData data;
1617 memset (&data, 0, sizeof (data));
1619 data.channel = channel;
1620 data.ignore_unsupported = ignore_unsupported;
1622 dbus_g_type_specialized_init ();
1623 _dbus_g_type_specialized_builtins_init ();
1625 WRITE_OR_LOSE ("/* Generated by dbus-binding-tool; do not edit! */\n\n");
1626 WRITE_OR_LOSE ("#include <glib/gtypes.h>\n");
1627 WRITE_OR_LOSE ("#include <glib/gerror.h>\n");
1628 WRITE_OR_LOSE ("#include <dbus/dbus-glib.h>\n\n");
1629 WRITE_OR_LOSE ("G_BEGIN_DECLS\n\n");
1631 ret = generate_client_glue (info, &data, error);
1635 WRITE_OR_LOSE ("G_END_DECLS\n");