Update contrib.
1 /* -*- mode: C; c-file-style: "gnu" -*- */
2 /* dbus-gparser.c parse DBus description files
4 * Copyright (C) 2003, 2005 Red Hat, Inc.
5 * Portion Copyright © 2008 Nokia Corporation and/or its subsidiary(-ies). All rights reserved.
6 * Licensed under the Academic Free License version 2.1
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
23 #include "dbus-gparser.h"
24 #include "dbus/dbus-glib-lowlevel.h"
25 #include "dbus-gidl.h"
26 #include "dbus-gobject.h"
27 #include "dbus/dbus-signature.h"
34 #define _(x) gettext ((x))
43 #ifndef DOXYGEN_SHOULD_SKIP_THIS
45 #define ELEMENT_IS(name) (strcmp (element_name, (name)) == 0)
54 locate_attributes (const char *element_name,
55 const char **attribute_names,
56 const char **attribute_values,
58 const char *first_attribute_name,
59 const char **first_attribute_retloc,
67 LocateAttr attrs[MAX_ATTRS];
71 g_return_val_if_fail (first_attribute_name != NULL, FALSE);
72 g_return_val_if_fail (first_attribute_retloc != NULL, FALSE);
77 attrs[0].name = first_attribute_name;
78 attrs[0].retloc = first_attribute_retloc;
79 *first_attribute_retloc = NULL;
81 va_start (args, first_attribute_retloc);
83 name = va_arg (args, const char*);
84 retloc = va_arg (args, const char**);
88 g_return_val_if_fail (retloc != NULL, FALSE);
90 g_assert (n_attrs < MAX_ATTRS);
92 attrs[n_attrs].name = name;
93 attrs[n_attrs].retloc = retloc;
97 name = va_arg (args, const char*);
98 retloc = va_arg (args, const char**);
107 while (attribute_names[i])
116 if (strcmp (attrs[j].name, attribute_names[i]) == 0)
118 retloc = attrs[j].retloc;
124 G_MARKUP_ERROR_PARSE,
125 _("Attribute \"%s\" repeated twice on the same <%s> element"),
126 attrs[j].name, element_name);
131 *retloc = attribute_values[i];
142 G_MARKUP_ERROR_PARSE,
143 _("Attribute \"%s\" is invalid on <%s> element in this context"),
144 attribute_names[i], element_name);
158 check_no_attributes (const char *element_name,
159 const char **attribute_names,
160 const char **attribute_values,
163 if (attribute_names[0] != NULL)
167 G_MARKUP_ERROR_PARSE,
168 _("Attribute \"%s\" is invalid on <%s> element in this context"),
169 attribute_names[0], element_name);
181 NodeInfo *result; /* Filled in when we pop the last node */
183 InterfaceInfo *interface;
186 PropertyInfo *property;
188 gboolean in_annotation;
196 parser = g_new0 (Parser, 1);
198 parser->refcount = 1;
204 parser_ref (Parser *parser)
206 parser->refcount += 1;
212 parser_unref (Parser *parser)
214 parser->refcount -= 1;
215 if (parser->refcount == 0)
218 node_info_unref (parser->result);
225 parser_check_doctype (Parser *parser,
229 g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
231 if (strcmp (doctype, "node") != 0)
235 G_MARKUP_ERROR_PARSE,
236 "D-BUS description file has the wrong document type %s, use node or interface",
245 parse_node (Parser *parser,
246 const char *element_name,
247 const char **attribute_names,
248 const char **attribute_values,
254 if (parser->interface ||
259 parser->in_annotation)
261 g_set_error (error, G_MARKUP_ERROR,
262 G_MARKUP_ERROR_PARSE,
263 _("Can't put <%s> element here"),
269 if (!locate_attributes (element_name, attribute_names,
270 attribute_values, error,
275 /* Only the root node can have no name */
276 if (parser->node_stack != NULL && name == NULL)
278 g_set_error (error, G_MARKUP_ERROR,
279 G_MARKUP_ERROR_PARSE,
280 _("\"%s\" attribute required on <%s> element "),
281 "name", element_name);
285 /* Root element name must be absolute */
286 if (parser->node_stack == NULL && name && *name != '/')
288 g_set_error (error, G_MARKUP_ERROR,
289 G_MARKUP_ERROR_PARSE,
290 _("\"%s\" attribute on <%s> element must be an absolute object path, \"%s\" not OK"),
291 "name", element_name, name);
295 /* Other element names must not be absolute */
296 if (parser->node_stack != NULL && name && *name == '/')
298 g_set_error (error, G_MARKUP_ERROR,
299 G_MARKUP_ERROR_PARSE,
300 _("\"%s\" attribute on <%s> element must not be an absolute object path, \"%s\" starts with /"),
301 "name", element_name, name);
305 node = node_info_new (name);
307 if (parser->node_stack != NULL)
309 node_info_add_node (parser->node_stack->data,
313 parser->node_stack = g_slist_prepend (parser->node_stack,
320 parse_interface (Parser *parser,
321 const char *element_name,
322 const char **attribute_names,
323 const char **attribute_values,
327 InterfaceInfo *iface;
330 if (parser->interface ||
335 parser->in_annotation ||
336 (parser->node_stack == NULL))
338 g_set_error (error, G_MARKUP_ERROR,
339 G_MARKUP_ERROR_PARSE,
340 _("Can't put <%s> element here"),
346 if (!locate_attributes (element_name, attribute_names,
347 attribute_values, error,
354 g_set_error (error, G_MARKUP_ERROR,
355 G_MARKUP_ERROR_PARSE,
356 _("\"%s\" attribute required on <%s> element "),
357 "name", element_name);
361 top = parser->node_stack->data;
363 iface = interface_info_new (name);
364 node_info_add_interface (top, iface);
365 interface_info_unref (iface);
367 parser->interface = iface;
373 parse_method (Parser *parser,
374 const char *element_name,
375 const char **attribute_names,
376 const char **attribute_values,
382 if (parser->interface == NULL ||
383 parser->node_stack == NULL ||
387 parser->in_annotation ||
390 g_set_error (error, G_MARKUP_ERROR,
391 G_MARKUP_ERROR_PARSE,
392 _("Can't put <%s> element here"),
398 if (!locate_attributes (element_name, attribute_names,
399 attribute_values, error,
406 g_set_error (error, G_MARKUP_ERROR,
407 G_MARKUP_ERROR_PARSE,
408 _("\"%s\" attribute required on <%s> element "),
409 "name", element_name);
413 method = method_info_new (name);
414 interface_info_add_method (parser->interface, method);
415 method_info_unref (method);
417 parser->method = method;
423 parse_signal (Parser *parser,
424 const char *element_name,
425 const char **attribute_names,
426 const char **attribute_values,
432 if (parser->interface == NULL ||
433 parser->node_stack == NULL ||
437 parser->in_annotation ||
440 g_set_error (error, G_MARKUP_ERROR,
441 G_MARKUP_ERROR_PARSE,
442 _("Can't put <%s> element here"),
448 if (!locate_attributes (element_name, attribute_names,
449 attribute_values, error,
456 g_set_error (error, G_MARKUP_ERROR,
457 G_MARKUP_ERROR_PARSE,
458 _("\"%s\" attribute required on <%s> element "),
459 "name", element_name);
463 signal = signal_info_new (name);
464 interface_info_add_signal (parser->interface, signal);
465 signal_info_unref (signal);
467 parser->signal = signal;
473 validate_signature (const char *str,
474 const char *element_name,
479 dbus_error_init (&derror);
481 if (!dbus_signature_validate (str, &derror))
483 dbus_set_g_error (error, &derror);
490 parse_property (Parser *parser,
491 const char *element_name,
492 const char **attribute_names,
493 const char **attribute_values,
499 PropertyInfo *property;
500 PropertyAccessFlags access_flags;
502 if (parser->interface == NULL ||
503 parser->node_stack == NULL ||
507 parser->in_annotation ||
510 g_set_error (error, G_MARKUP_ERROR,
511 G_MARKUP_ERROR_PARSE,
512 _("Can't put <%s> element here"),
518 if (!locate_attributes (element_name, attribute_names,
519 attribute_values, error,
528 g_set_error (error, G_MARKUP_ERROR,
529 G_MARKUP_ERROR_PARSE,
530 _("\"%s\" attribute required on <%s> element "),
531 "name", element_name);
537 g_set_error (error, G_MARKUP_ERROR,
538 G_MARKUP_ERROR_PARSE,
539 _("\"%s\" attribute required on <%s> element "),
540 "access", element_name);
546 g_set_error (error, G_MARKUP_ERROR,
547 G_MARKUP_ERROR_PARSE,
548 _("\"%s\" attribute required on <%s> element "),
549 "type", element_name);
553 if (!validate_signature (type, element_name, error))
557 if (strcmp (access, "readwrite") == 0)
558 access_flags = PROPERTY_READ | PROPERTY_WRITE;
559 else if (strcmp (access, "read") == 0)
560 access_flags = PROPERTY_READ;
561 else if (strcmp (access, "write") == 0)
562 access_flags = PROPERTY_WRITE;
565 g_set_error (error, G_MARKUP_ERROR,
566 G_MARKUP_ERROR_PARSE,
567 _("access=\"%s\" must have value readwrite, read, or write on %s\n"),
568 access, element_name);
572 property = property_info_new (name, type, access_flags);
573 interface_info_add_property (parser->interface, property);
574 property_info_unref (property);
576 parser->property = property;
582 parse_arg (Parser *parser,
583 const char *element_name,
584 const char **attribute_names,
585 const char **attribute_values,
590 const char *direction;
593 char *generated_name;
595 if (!(parser->method || parser->signal) ||
596 parser->node_stack == NULL ||
598 parser->in_annotation ||
601 g_set_error (error, G_MARKUP_ERROR,
602 G_MARKUP_ERROR_PARSE,
603 _("Can't put <%s> element here"),
609 if (!locate_attributes (element_name, attribute_names,
610 attribute_values, error,
613 "direction", &direction,
617 /* name can be null for args */
621 g_set_error (error, G_MARKUP_ERROR,
622 G_MARKUP_ERROR_PARSE,
623 _("\"%s\" attribute required on <%s> element "),
624 "type", element_name);
628 if (direction == NULL)
630 /* methods default to in, signal to out */
633 else if (parser->signal)
636 g_assert_not_reached ();
641 if (strcmp (direction, "in") == 0)
643 else if (strcmp (direction, "out") == 0)
646 if (dir == ARG_INVALID ||
647 (parser->signal && dir == ARG_IN))
650 g_set_error (error, G_MARKUP_ERROR,
651 G_MARKUP_ERROR_PARSE,
652 _("Signals must have direction=\"out\" (just omit the direction attribute)"));
654 g_set_error (error, G_MARKUP_ERROR,
655 G_MARKUP_ERROR_PARSE,
656 _("\"%s\" attribute on <%s> has value \"in\" or \"out\""),
657 "direction", element_name);
661 if (!validate_signature (type, element_name, error))
664 generated_name = NULL;
666 generated_name = g_strdup_printf ("arg%d",
668 method_info_get_n_args (parser->method) :
669 signal_info_get_n_args (parser->signal));
671 arg = arg_info_new (name ? name : generated_name, dir, type);
673 method_info_add_arg (parser->method, arg);
674 else if (parser->signal)
675 signal_info_add_arg (parser->signal, arg);
677 g_assert_not_reached ();
679 g_free (generated_name);
681 arg_info_unref (arg);
689 parse_annotation (Parser *parser,
690 const char *element_name,
691 const char **attribute_names,
692 const char **attribute_values,
698 if (!(parser->method || parser->interface || parser->arg) ||
699 parser->node_stack == NULL ||
702 parser->in_annotation)
704 g_set_error (error, G_MARKUP_ERROR,
705 G_MARKUP_ERROR_PARSE,
706 _("Can't put <%s> element here"),
712 if (!locate_attributes (element_name, attribute_names,
713 attribute_values, error,
721 g_set_error (error, G_MARKUP_ERROR,
722 G_MARKUP_ERROR_PARSE,
723 _("\"%s\" attribute required on <%s> element "),
724 "name", element_name);
729 g_set_error (error, G_MARKUP_ERROR,
730 G_MARKUP_ERROR_PARSE,
731 _("\"%s\" attribute required on <%s> element "),
732 "value", element_name);
737 arg_info_add_annotation (parser->arg, name, value);
738 else if (parser->method)
739 method_info_add_annotation (parser->method, name, value);
740 else if (parser->interface)
741 interface_info_add_annotation (parser->interface, name, value);
743 g_assert_not_reached ();
745 parser->in_annotation = TRUE;
751 parser_start_element (Parser *parser,
752 const char *element_name,
753 const char **attribute_names,
754 const char **attribute_values,
757 g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
759 if (ELEMENT_IS ("node"))
761 if (!parse_node (parser, element_name, attribute_names,
762 attribute_values, error))
765 else if (ELEMENT_IS ("interface"))
767 if (!parse_interface (parser, element_name, attribute_names,
768 attribute_values, error))
771 else if (ELEMENT_IS ("method"))
773 if (!parse_method (parser, element_name, attribute_names,
774 attribute_values, error))
777 else if (ELEMENT_IS ("signal"))
779 if (!parse_signal (parser, element_name, attribute_names,
780 attribute_values, error))
783 else if (ELEMENT_IS ("property"))
785 if (!parse_property (parser, element_name, attribute_names,
786 attribute_values, error))
789 else if (ELEMENT_IS ("arg"))
791 if (!parse_arg (parser, element_name, attribute_names,
792 attribute_values, error))
795 else if (ELEMENT_IS ("annotation"))
797 if (!parse_annotation (parser, element_name, attribute_names,
798 attribute_values, error))
803 g_set_error (error, G_MARKUP_ERROR,
804 G_MARKUP_ERROR_PARSE,
805 _("Element <%s> not recognized"),
813 parser_end_element (Parser *parser,
814 const char *element_name,
817 g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
819 if (ELEMENT_IS ("interface"))
821 parser->interface = NULL;
823 else if (ELEMENT_IS ("method"))
825 parser->method = NULL;
827 else if (ELEMENT_IS ("signal"))
829 parser->signal = NULL;
831 else if (ELEMENT_IS ("property"))
833 parser->property = NULL;
835 else if (ELEMENT_IS ("arg"))
839 else if (ELEMENT_IS ("annotation"))
841 parser->in_annotation = FALSE;
843 else if (ELEMENT_IS ("node"))
847 g_assert (parser->node_stack != NULL);
848 top = parser->node_stack->data;
850 parser->node_stack = g_slist_remove (parser->node_stack,
853 if (parser->node_stack == NULL)
854 parser->result = top; /* We are done, store the result */
857 g_assert_not_reached (); /* should have had an error on start_element */
863 parser_content (Parser *parser,
868 g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
870 /* FIXME check that it's all whitespace */
876 parser_finished (Parser *parser,
879 g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
885 parser_get_nodes (Parser *parser)
887 return parser->result;
890 #endif /* DOXYGEN_SHOULD_SKIP_THIS */