2003-09-01 18:02:06 +00:00
|
|
|
/* -*- mode: C; c-file-style: "gnu" -*- */
|
|
|
|
|
/* dbus-gparser.c parse DBus description files
|
|
|
|
|
*
|
|
|
|
|
* Copyright (C) 2003 Red Hat, Inc.
|
|
|
|
|
*
|
2003-12-02 10:44:22 +00:00
|
|
|
* Licensed under the Academic Free License version 2.0
|
2003-09-01 18:02:06 +00:00
|
|
|
*
|
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
|
* (at your option) any later version.
|
|
|
|
|
*
|
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
#include "dbus-gparser.h"
|
|
|
|
|
#include "dbus-gidl.h"
|
2003-09-03 02:08:25 +00:00
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
|
|
#include <libintl.h>
|
|
|
|
|
#define _(x) gettext ((x))
|
|
|
|
|
#define N_(x) x
|
|
|
|
|
|
2003-09-07 23:04:54 +00:00
|
|
|
#ifndef DOXYGEN_SHOULD_SKIP_THIS
|
|
|
|
|
|
2003-09-03 02:08:25 +00:00
|
|
|
#define ELEMENT_IS(name) (strcmp (element_name, (name)) == 0)
|
|
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
|
{
|
|
|
|
|
const char *name;
|
|
|
|
|
const char **retloc;
|
|
|
|
|
} LocateAttr;
|
|
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
|
locate_attributes (const char *element_name,
|
|
|
|
|
const char **attribute_names,
|
|
|
|
|
const char **attribute_values,
|
|
|
|
|
GError **error,
|
|
|
|
|
const char *first_attribute_name,
|
|
|
|
|
const char **first_attribute_retloc,
|
|
|
|
|
...)
|
|
|
|
|
{
|
|
|
|
|
va_list args;
|
|
|
|
|
const char *name;
|
|
|
|
|
const char **retloc;
|
|
|
|
|
int n_attrs;
|
|
|
|
|
#define MAX_ATTRS 24
|
|
|
|
|
LocateAttr attrs[MAX_ATTRS];
|
|
|
|
|
gboolean retval;
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
g_return_val_if_fail (first_attribute_name != NULL, FALSE);
|
|
|
|
|
g_return_val_if_fail (first_attribute_retloc != NULL, FALSE);
|
|
|
|
|
|
|
|
|
|
retval = TRUE;
|
|
|
|
|
|
|
|
|
|
n_attrs = 1;
|
|
|
|
|
attrs[0].name = first_attribute_name;
|
|
|
|
|
attrs[0].retloc = first_attribute_retloc;
|
|
|
|
|
*first_attribute_retloc = NULL;
|
|
|
|
|
|
|
|
|
|
va_start (args, first_attribute_retloc);
|
|
|
|
|
|
|
|
|
|
name = va_arg (args, const char*);
|
|
|
|
|
retloc = va_arg (args, const char**);
|
|
|
|
|
|
|
|
|
|
while (name != NULL)
|
|
|
|
|
{
|
|
|
|
|
g_return_val_if_fail (retloc != NULL, FALSE);
|
|
|
|
|
|
|
|
|
|
g_assert (n_attrs < MAX_ATTRS);
|
|
|
|
|
|
|
|
|
|
attrs[n_attrs].name = name;
|
|
|
|
|
attrs[n_attrs].retloc = retloc;
|
|
|
|
|
n_attrs += 1;
|
|
|
|
|
*retloc = NULL;
|
|
|
|
|
|
|
|
|
|
name = va_arg (args, const char*);
|
|
|
|
|
retloc = va_arg (args, const char**);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
va_end (args);
|
|
|
|
|
|
|
|
|
|
if (!retval)
|
|
|
|
|
return retval;
|
|
|
|
|
|
|
|
|
|
i = 0;
|
|
|
|
|
while (attribute_names[i])
|
|
|
|
|
{
|
|
|
|
|
int j;
|
|
|
|
|
gboolean found;
|
|
|
|
|
|
|
|
|
|
found = FALSE;
|
|
|
|
|
j = 0;
|
|
|
|
|
while (j < n_attrs)
|
|
|
|
|
{
|
|
|
|
|
if (strcmp (attrs[j].name, attribute_names[i]) == 0)
|
|
|
|
|
{
|
|
|
|
|
retloc = attrs[j].retloc;
|
|
|
|
|
|
|
|
|
|
if (*retloc != NULL)
|
|
|
|
|
{
|
|
|
|
|
g_set_error (error,
|
|
|
|
|
G_MARKUP_ERROR,
|
|
|
|
|
G_MARKUP_ERROR_PARSE,
|
|
|
|
|
_("Attribute \"%s\" repeated twice on the same <%s> element"),
|
|
|
|
|
attrs[j].name, element_name);
|
|
|
|
|
retval = FALSE;
|
|
|
|
|
goto out;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
*retloc = attribute_values[i];
|
|
|
|
|
found = TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
++j;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!found)
|
|
|
|
|
{
|
|
|
|
|
g_set_error (error,
|
|
|
|
|
G_MARKUP_ERROR,
|
|
|
|
|
G_MARKUP_ERROR_PARSE,
|
|
|
|
|
_("Attribute \"%s\" is invalid on <%s> element in this context"),
|
|
|
|
|
attribute_names[i], element_name);
|
|
|
|
|
retval = FALSE;
|
|
|
|
|
goto out;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
++i;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
out:
|
|
|
|
|
return retval;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
|
check_no_attributes (const char *element_name,
|
|
|
|
|
const char **attribute_names,
|
|
|
|
|
const char **attribute_values,
|
|
|
|
|
GError **error)
|
|
|
|
|
{
|
|
|
|
|
if (attribute_names[0] != NULL)
|
|
|
|
|
{
|
|
|
|
|
g_set_error (error,
|
|
|
|
|
G_MARKUP_ERROR,
|
|
|
|
|
G_MARKUP_ERROR_PARSE,
|
|
|
|
|
_("Attribute \"%s\" is invalid on <%s> element in this context"),
|
|
|
|
|
attribute_names[0], element_name);
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
|
}
|
2003-09-01 18:02:06 +00:00
|
|
|
|
|
|
|
|
struct Parser
|
|
|
|
|
{
|
|
|
|
|
int refcount;
|
|
|
|
|
|
2003-09-03 02:08:25 +00:00
|
|
|
NodeInfo *result; /* Filled in when we pop the last node */
|
|
|
|
|
GSList *node_stack;
|
|
|
|
|
InterfaceInfo *interface;
|
|
|
|
|
MethodInfo *method;
|
|
|
|
|
SignalInfo *signal;
|
|
|
|
|
ArgInfo *arg;
|
2003-09-01 18:02:06 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
Parser*
|
|
|
|
|
parser_new (void)
|
|
|
|
|
{
|
|
|
|
|
Parser *parser;
|
|
|
|
|
|
|
|
|
|
parser = g_new0 (Parser, 1);
|
|
|
|
|
|
|
|
|
|
parser->refcount = 1;
|
|
|
|
|
|
|
|
|
|
return parser;
|
|
|
|
|
}
|
|
|
|
|
|
2003-11-27 01:25:50 +00:00
|
|
|
Parser *
|
2003-09-01 18:02:06 +00:00
|
|
|
parser_ref (Parser *parser)
|
|
|
|
|
{
|
|
|
|
|
parser->refcount += 1;
|
2003-11-27 01:25:50 +00:00
|
|
|
|
|
|
|
|
return parser;
|
2003-09-01 18:02:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
parser_unref (Parser *parser)
|
|
|
|
|
{
|
|
|
|
|
parser->refcount -= 1;
|
|
|
|
|
if (parser->refcount == 0)
|
|
|
|
|
{
|
2003-09-03 02:08:25 +00:00
|
|
|
if (parser->result)
|
|
|
|
|
node_info_unref (parser->result);
|
2003-09-01 18:02:06 +00:00
|
|
|
|
|
|
|
|
g_free (parser);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
|
parser_check_doctype (Parser *parser,
|
|
|
|
|
const char *doctype,
|
|
|
|
|
GError **error)
|
|
|
|
|
{
|
2003-09-03 02:08:25 +00:00
|
|
|
g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
|
2003-09-01 18:02:06 +00:00
|
|
|
|
2004-06-01 03:02:26 +00:00
|
|
|
if (strcmp (doctype, "node") != 0)
|
2003-09-01 18:02:06 +00:00
|
|
|
{
|
|
|
|
|
g_set_error (error,
|
2003-09-03 02:08:25 +00:00
|
|
|
G_MARKUP_ERROR,
|
2003-09-01 18:02:06 +00:00
|
|
|
G_MARKUP_ERROR_PARSE,
|
2004-06-01 03:02:26 +00:00
|
|
|
"D-BUS description file has the wrong document type %s, use node or interface",
|
2003-09-01 18:02:06 +00:00
|
|
|
doctype);
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
return TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
2003-09-03 02:08:25 +00:00
|
|
|
static gboolean
|
|
|
|
|
parse_node (Parser *parser,
|
|
|
|
|
const char *element_name,
|
|
|
|
|
const char **attribute_names,
|
|
|
|
|
const char **attribute_values,
|
|
|
|
|
GError **error)
|
|
|
|
|
{
|
|
|
|
|
const char *name;
|
|
|
|
|
NodeInfo *node;
|
|
|
|
|
|
|
|
|
|
if (parser->interface ||
|
|
|
|
|
parser->method ||
|
|
|
|
|
parser->signal ||
|
|
|
|
|
parser->arg)
|
|
|
|
|
{
|
|
|
|
|
g_set_error (error, G_MARKUP_ERROR,
|
|
|
|
|
G_MARKUP_ERROR_PARSE,
|
|
|
|
|
_("Can't put a <%s> element here"),
|
|
|
|
|
element_name);
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
name = NULL;
|
|
|
|
|
if (!locate_attributes (element_name, attribute_names,
|
|
|
|
|
attribute_values, error,
|
|
|
|
|
"name", &name,
|
|
|
|
|
NULL))
|
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
|
|
/* Only the root node can have no name */
|
|
|
|
|
if (parser->node_stack != NULL && name == NULL)
|
|
|
|
|
{
|
|
|
|
|
g_set_error (error, G_MARKUP_ERROR,
|
|
|
|
|
G_MARKUP_ERROR_PARSE,
|
|
|
|
|
_("\"%s\" attribute required on <%s> element "),
|
|
|
|
|
"name", element_name);
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
2003-09-17 03:52:07 +00:00
|
|
|
|
2003-09-03 02:08:25 +00:00
|
|
|
node = node_info_new (name);
|
2003-09-17 03:52:07 +00:00
|
|
|
|
|
|
|
|
if (parser->node_stack != NULL)
|
|
|
|
|
{
|
|
|
|
|
node_info_add_node (parser->node_stack->data,
|
|
|
|
|
node);
|
|
|
|
|
}
|
|
|
|
|
|
2003-09-03 02:08:25 +00:00
|
|
|
parser->node_stack = g_slist_prepend (parser->node_stack,
|
|
|
|
|
node);
|
|
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
|
parse_interface (Parser *parser,
|
|
|
|
|
const char *element_name,
|
|
|
|
|
const char **attribute_names,
|
|
|
|
|
const char **attribute_values,
|
|
|
|
|
GError **error)
|
|
|
|
|
{
|
|
|
|
|
const char *name;
|
|
|
|
|
InterfaceInfo *iface;
|
|
|
|
|
NodeInfo *top;
|
|
|
|
|
|
|
|
|
|
if (parser->interface ||
|
|
|
|
|
parser->method ||
|
|
|
|
|
parser->signal ||
|
|
|
|
|
parser->arg ||
|
|
|
|
|
(parser->node_stack == NULL))
|
|
|
|
|
{
|
|
|
|
|
g_set_error (error, G_MARKUP_ERROR,
|
|
|
|
|
G_MARKUP_ERROR_PARSE,
|
|
|
|
|
_("Can't put a <%s> element here"),
|
|
|
|
|
element_name);
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
name = NULL;
|
|
|
|
|
if (!locate_attributes (element_name, attribute_names,
|
|
|
|
|
attribute_values, error,
|
|
|
|
|
"name", &name,
|
|
|
|
|
NULL))
|
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
|
|
if (name == NULL)
|
|
|
|
|
{
|
|
|
|
|
g_set_error (error, G_MARKUP_ERROR,
|
|
|
|
|
G_MARKUP_ERROR_PARSE,
|
|
|
|
|
_("\"%s\" attribute required on <%s> element "),
|
|
|
|
|
"name", element_name);
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
top = parser->node_stack->data;
|
|
|
|
|
|
|
|
|
|
iface = interface_info_new (name);
|
|
|
|
|
node_info_add_interface (top, iface);
|
|
|
|
|
interface_info_unref (iface);
|
|
|
|
|
|
|
|
|
|
parser->interface = iface;
|
|
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
|
parse_method (Parser *parser,
|
|
|
|
|
const char *element_name,
|
|
|
|
|
const char **attribute_names,
|
|
|
|
|
const char **attribute_values,
|
|
|
|
|
GError **error)
|
|
|
|
|
{
|
|
|
|
|
const char *name;
|
|
|
|
|
MethodInfo *method;
|
|
|
|
|
NodeInfo *top;
|
|
|
|
|
|
|
|
|
|
if (parser->interface == NULL ||
|
|
|
|
|
parser->node_stack == NULL ||
|
|
|
|
|
parser->method ||
|
|
|
|
|
parser->signal ||
|
|
|
|
|
parser->arg)
|
|
|
|
|
{
|
|
|
|
|
g_set_error (error, G_MARKUP_ERROR,
|
|
|
|
|
G_MARKUP_ERROR_PARSE,
|
|
|
|
|
_("Can't put a <%s> element here"),
|
|
|
|
|
element_name);
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
name = NULL;
|
|
|
|
|
if (!locate_attributes (element_name, attribute_names,
|
|
|
|
|
attribute_values, error,
|
|
|
|
|
"name", &name,
|
|
|
|
|
NULL))
|
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
|
|
if (name == NULL)
|
|
|
|
|
{
|
|
|
|
|
g_set_error (error, G_MARKUP_ERROR,
|
|
|
|
|
G_MARKUP_ERROR_PARSE,
|
|
|
|
|
_("\"%s\" attribute required on <%s> element "),
|
|
|
|
|
"name", element_name);
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
top = parser->node_stack->data;
|
|
|
|
|
|
|
|
|
|
method = method_info_new (name);
|
|
|
|
|
interface_info_add_method (parser->interface, method);
|
|
|
|
|
method_info_unref (method);
|
|
|
|
|
|
|
|
|
|
parser->method = method;
|
|
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
|
parse_signal (Parser *parser,
|
|
|
|
|
const char *element_name,
|
|
|
|
|
const char **attribute_names,
|
|
|
|
|
const char **attribute_values,
|
|
|
|
|
GError **error)
|
|
|
|
|
{
|
|
|
|
|
const char *name;
|
|
|
|
|
SignalInfo *signal;
|
|
|
|
|
NodeInfo *top;
|
|
|
|
|
|
|
|
|
|
if (parser->interface == NULL ||
|
|
|
|
|
parser->node_stack == NULL ||
|
|
|
|
|
parser->signal ||
|
|
|
|
|
parser->signal ||
|
|
|
|
|
parser->arg)
|
|
|
|
|
{
|
|
|
|
|
g_set_error (error, G_MARKUP_ERROR,
|
|
|
|
|
G_MARKUP_ERROR_PARSE,
|
|
|
|
|
_("Can't put a <%s> element here"),
|
|
|
|
|
element_name);
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
name = NULL;
|
|
|
|
|
if (!locate_attributes (element_name, attribute_names,
|
|
|
|
|
attribute_values, error,
|
|
|
|
|
"name", &name,
|
|
|
|
|
NULL))
|
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
|
|
if (name == NULL)
|
|
|
|
|
{
|
|
|
|
|
g_set_error (error, G_MARKUP_ERROR,
|
|
|
|
|
G_MARKUP_ERROR_PARSE,
|
|
|
|
|
_("\"%s\" attribute required on <%s> element "),
|
|
|
|
|
"name", element_name);
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
top = parser->node_stack->data;
|
|
|
|
|
|
|
|
|
|
signal = signal_info_new (name);
|
|
|
|
|
interface_info_add_signal (parser->interface, signal);
|
|
|
|
|
signal_info_unref (signal);
|
|
|
|
|
|
|
|
|
|
parser->signal = signal;
|
|
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
basic_type_from_string (const char *str)
|
|
|
|
|
{
|
|
|
|
|
if (strcmp (str, "string") == 0)
|
|
|
|
|
return DBUS_TYPE_STRING;
|
|
|
|
|
else if (strcmp (str, "int32") == 0)
|
|
|
|
|
return DBUS_TYPE_INT32;
|
|
|
|
|
else if (strcmp (str, "uint32") == 0)
|
|
|
|
|
return DBUS_TYPE_UINT32;
|
|
|
|
|
else if (strcmp (str, "int64") == 0)
|
|
|
|
|
return DBUS_TYPE_INT64;
|
|
|
|
|
else if (strcmp (str, "uint64") == 0)
|
|
|
|
|
return DBUS_TYPE_UINT64;
|
|
|
|
|
else if (strcmp (str, "double") == 0)
|
|
|
|
|
return DBUS_TYPE_DOUBLE;
|
|
|
|
|
else if (strcmp (str, "byte") == 0)
|
|
|
|
|
return DBUS_TYPE_BYTE;
|
|
|
|
|
else if (strcmp (str, "boolean") == 0)
|
|
|
|
|
return DBUS_TYPE_BOOLEAN;
|
|
|
|
|
else if (strcmp (str, "byte") == 0)
|
|
|
|
|
return DBUS_TYPE_BYTE;
|
|
|
|
|
else if (strcmp (str, "object") == 0)
|
|
|
|
|
return DBUS_TYPE_OBJECT_PATH;
|
|
|
|
|
else
|
|
|
|
|
return DBUS_TYPE_INVALID;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
type_from_string (const char *str)
|
|
|
|
|
{
|
|
|
|
|
return basic_type_from_string (str);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
|
parse_arg (Parser *parser,
|
|
|
|
|
const char *element_name,
|
|
|
|
|
const char **attribute_names,
|
|
|
|
|
const char **attribute_values,
|
|
|
|
|
GError **error)
|
|
|
|
|
{
|
|
|
|
|
const char *name;
|
|
|
|
|
const char *type;
|
|
|
|
|
const char *direction;
|
|
|
|
|
ArgDirection dir;
|
|
|
|
|
int t;
|
|
|
|
|
ArgInfo *arg;
|
|
|
|
|
|
|
|
|
|
if (!(parser->method || parser->signal) ||
|
|
|
|
|
parser->node_stack == NULL ||
|
|
|
|
|
parser->arg)
|
|
|
|
|
{
|
|
|
|
|
g_set_error (error, G_MARKUP_ERROR,
|
|
|
|
|
G_MARKUP_ERROR_PARSE,
|
|
|
|
|
_("Can't put a <%s> element here"),
|
|
|
|
|
element_name);
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
name = NULL;
|
|
|
|
|
if (!locate_attributes (element_name, attribute_names,
|
|
|
|
|
attribute_values, error,
|
|
|
|
|
"name", &name,
|
|
|
|
|
"type", &type,
|
|
|
|
|
"direction", &direction,
|
|
|
|
|
NULL))
|
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
|
|
/* name can be null for args */
|
|
|
|
|
|
|
|
|
|
if (type == NULL)
|
|
|
|
|
{
|
|
|
|
|
g_set_error (error, G_MARKUP_ERROR,
|
|
|
|
|
G_MARKUP_ERROR_PARSE,
|
|
|
|
|
_("\"%s\" attribute required on <%s> element "),
|
|
|
|
|
"type", element_name);
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (direction == NULL)
|
|
|
|
|
{
|
|
|
|
|
/* methods default to in, signal to out */
|
|
|
|
|
if (parser->method)
|
|
|
|
|
direction = "in";
|
|
|
|
|
else if (parser->signal)
|
|
|
|
|
direction = "out";
|
|
|
|
|
else
|
|
|
|
|
g_assert_not_reached ();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (strcmp (direction, "in") == 0)
|
|
|
|
|
dir = ARG_IN;
|
|
|
|
|
else if (strcmp (direction, "out") == 0)
|
|
|
|
|
dir = ARG_OUT;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
g_set_error (error, G_MARKUP_ERROR,
|
|
|
|
|
G_MARKUP_ERROR_PARSE,
|
|
|
|
|
_("\"%s\" attribute on <%s> has value \"in\" or \"out\""),
|
|
|
|
|
"direction", element_name);
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
t = type_from_string (type);
|
|
|
|
|
|
|
|
|
|
arg = arg_info_new (name, dir, t);
|
|
|
|
|
if (parser->method)
|
|
|
|
|
method_info_add_arg (parser->method, arg);
|
|
|
|
|
else if (parser->signal)
|
|
|
|
|
signal_info_add_arg (parser->signal, arg);
|
|
|
|
|
else
|
|
|
|
|
g_assert_not_reached ();
|
|
|
|
|
|
|
|
|
|
arg_info_unref (arg);
|
|
|
|
|
|
|
|
|
|
parser->arg = arg;
|
|
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
2003-09-01 18:02:06 +00:00
|
|
|
gboolean
|
|
|
|
|
parser_start_element (Parser *parser,
|
|
|
|
|
const char *element_name,
|
|
|
|
|
const char **attribute_names,
|
|
|
|
|
const char **attribute_values,
|
|
|
|
|
GError **error)
|
|
|
|
|
{
|
2003-09-03 02:08:25 +00:00
|
|
|
g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
|
2003-09-01 18:02:06 +00:00
|
|
|
|
2003-09-03 02:08:25 +00:00
|
|
|
if (ELEMENT_IS ("node"))
|
|
|
|
|
{
|
|
|
|
|
if (!parse_node (parser, element_name, attribute_names,
|
|
|
|
|
attribute_values, error))
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
else if (ELEMENT_IS ("interface"))
|
|
|
|
|
{
|
|
|
|
|
if (!parse_interface (parser, element_name, attribute_names,
|
|
|
|
|
attribute_values, error))
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
else if (ELEMENT_IS ("method"))
|
|
|
|
|
{
|
|
|
|
|
if (!parse_method (parser, element_name, attribute_names,
|
|
|
|
|
attribute_values, error))
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
else if (ELEMENT_IS ("signal"))
|
|
|
|
|
{
|
|
|
|
|
if (!parse_signal (parser, element_name, attribute_names,
|
|
|
|
|
attribute_values, error))
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
else if (ELEMENT_IS ("arg"))
|
|
|
|
|
{
|
|
|
|
|
if (!parse_arg (parser, element_name, attribute_names,
|
|
|
|
|
attribute_values, error))
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
g_set_error (error, G_MARKUP_ERROR,
|
|
|
|
|
G_MARKUP_ERROR_PARSE,
|
|
|
|
|
_("Element <%s> not recognized"),
|
|
|
|
|
element_name);
|
|
|
|
|
}
|
|
|
|
|
|
2003-09-01 18:02:06 +00:00
|
|
|
return TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
|
parser_end_element (Parser *parser,
|
|
|
|
|
const char *element_name,
|
|
|
|
|
GError **error)
|
|
|
|
|
{
|
2003-09-03 02:08:25 +00:00
|
|
|
g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
|
|
|
|
|
|
|
|
|
|
if (ELEMENT_IS ("interface"))
|
|
|
|
|
{
|
|
|
|
|
parser->interface = NULL;
|
|
|
|
|
}
|
|
|
|
|
else if (ELEMENT_IS ("method"))
|
|
|
|
|
{
|
|
|
|
|
parser->method = NULL;
|
|
|
|
|
}
|
|
|
|
|
else if (ELEMENT_IS ("signal"))
|
|
|
|
|
{
|
|
|
|
|
parser->signal = NULL;
|
|
|
|
|
}
|
|
|
|
|
else if (ELEMENT_IS ("arg"))
|
|
|
|
|
{
|
|
|
|
|
parser->arg = NULL;
|
|
|
|
|
}
|
|
|
|
|
else if (ELEMENT_IS ("node"))
|
|
|
|
|
{
|
|
|
|
|
NodeInfo *top;
|
2003-09-01 18:02:06 +00:00
|
|
|
|
2003-09-03 02:08:25 +00:00
|
|
|
g_assert (parser->node_stack != NULL);
|
|
|
|
|
top = parser->node_stack->data;
|
|
|
|
|
|
|
|
|
|
parser->node_stack = g_slist_remove (parser->node_stack,
|
|
|
|
|
top);
|
|
|
|
|
|
|
|
|
|
if (parser->node_stack == NULL)
|
|
|
|
|
parser->result = top; /* We are done, store the result */
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
g_assert_not_reached (); /* should have had an error on start_element */
|
|
|
|
|
|
2003-09-01 18:02:06 +00:00
|
|
|
return TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
|
parser_content (Parser *parser,
|
|
|
|
|
const char *content,
|
|
|
|
|
int len,
|
|
|
|
|
GError **error)
|
|
|
|
|
{
|
2003-09-03 02:08:25 +00:00
|
|
|
g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
|
2003-09-01 18:02:06 +00:00
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
|
parser_finished (Parser *parser,
|
|
|
|
|
GError **error)
|
|
|
|
|
{
|
2003-09-03 02:08:25 +00:00
|
|
|
g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
|
2003-09-01 18:02:06 +00:00
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
|
}
|
2003-09-03 02:08:25 +00:00
|
|
|
|
|
|
|
|
NodeInfo*
|
|
|
|
|
parser_get_nodes (Parser *parser)
|
|
|
|
|
{
|
|
|
|
|
return parser->result;
|
|
|
|
|
}
|
2003-09-07 23:04:54 +00:00
|
|
|
|
|
|
|
|
#endif /* DOXYGEN_SHOULD_SKIP_THIS */
|