sbus_codegen revision e8058322725ba050014777ee2484f7e833ab1e3a
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#!/usr/bin/env python
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# Authors:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# Stef Walter <stefw@redhat.com>
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# Copyright (C) 2014 Red Hat
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# This program is free software; you can redistribute it and/or modify
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# it under the terms of the GNU General Public License as published by
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# the Free Software Foundation; either version 3 of the License, or
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# (at your option) any later version.
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# This program is distributed in the hope that it will be useful,
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# but WITHOUT ANY WARRANTY; without even the implied warranty of
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# GNU General Public License for more details.
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# You should have received a copy of the GNU General Public License
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# along with this program. If not, see <http://www.gnu.org/licenses/>.
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# Some parser code from GLib
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# Copyright (C) 2008-2011 Red Hat, Inc.
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# This library is free software; you can redistribute it and/or
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# modify it under the terms of the GNU Lesser General Public
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# License as published by the Free Software Foundation; either
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# version 2 of the License, or (at your option) any later version.
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# This library is distributed in the hope that it will be useful,
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# but WITHOUT ANY WARRANTY; without even the implied warranty of
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# Lesser General Public License for more details.
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# You should have received a copy of the GNU Lesser General
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# Public License along with this library; if not, write to the
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# Free Software Foundation, Inc., 59 Temple Place, Suite 330,
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# Boston, MA 02111-1307, USA.
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# Portions by: David Zeuthen <davidz@redhat.com>
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# DBus interfaces are defined here:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# http://dbus.freedesktop.org/doc/dbus-specification.html#introspection-format
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# The introspection data format has become the standard way to represent a
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# DBus interface. For many examples see /usr/share/dbus-1/interfaces/ on a
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# typical linux machine.
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# A word about annotations. These are extra flags or values that can be
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# assigned to anything. So far, the codegen supports this annotation:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# org.freedesktop.DBus.GLib.CSymbol
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# - An annotation specified in the specification that tells us what C symbol
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# to generate for a given interface or method. By default the codegen will
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# build up a symbol name from the DBus name.
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync#
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncimport optparse
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncimport os
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncimport re
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncimport StringIO
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncimport sys
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncimport xml.parsers.expat
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# -----------------------------------------------------------------------------
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# Objects
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncclass DBusXmlException(Exception):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync line = 0
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync file = None
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync # Lets us print problems like a compiler would
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def __str__(self):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync message = Exception.__str__(self)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if self.file and self.line:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return "%s:%d: %s" % (self.file, self.line, message)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync elif self.file:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return "%s: %s" % (self.file, message)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync else:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return message
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncclass Base:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def __init__(self, name):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if not name:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync raise DBusXmlException('No name on element')
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.name = name
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.annotations = { }
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def validate(self):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync pass
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def c_name(self):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return self.annotations.get("org.freedesktop.DBus.GLib.CSymbol", self.name)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# The basic types that we support marshalling right now. These
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# are the ones we can pass as basic arguments to libdbus directly.
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# If the dbus and sssd types are identical we pass things directly.
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# otherwise some copying is necessary.
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncBASIC_TYPES = {
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync 'y': ( "DBUS_TYPE_BYTE", "uint8_t", "uint8_t" ),
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync 'b': ( "DBUS_TYPE_BOOLEAN", "dbus_bool_t", "bool" ),
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync 'n': ( "DBUS_TYPE_INT16", "int16_t", "int16_t" ),
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync 'q': ( "DBUS_TYPE_UINT16", "uint16_t", "uint16_t" ),
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync 'i': ( "DBUS_TYPE_INT32", "int32_t", "int32_t" ),
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync 'u': ( "DBUS_TYPE_UINT32", "uint32_t", "uint32_t" ),
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync 'x': ( "DBUS_TYPE_INT64", "int64_t", "int64_t" ),
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync 't': ( "DBUS_TYPE_UINT64", "uint64_t", "uint64_t" ),
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync 'd': ( "DBUS_TYPE_DOUBLE", "double", "double" ),
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync 's': ( "DBUS_TYPE_STRING", "const char *", "const char *" ),
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync 'o': ( "DBUS_TYPE_OBJECT_PATH", "const char *", "const char *" ),
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync}
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncclass Typed(Base):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def __init__(self, name, type):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync Base.__init__(self, name)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.type = type
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.is_basic = False
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.is_array = False
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.dbus_constant = None
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.dbus_type = None
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.sssd_type = None
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if type[0] == 'a':
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync type = type[1:]
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.is_array = True
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if type in BASIC_TYPES:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync (self.dbus_constant, self.dbus_type, self.sssd_type) = BASIC_TYPES[type]
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync # If types are not identical, we can't do array (yet)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if self.is_array:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.is_basic = (self.dbus_type == self.sssd_type)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync else:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.is_basic = True
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncclass Arg(Typed):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def __init__(self, method, name, type):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync Typed.__init__(self, name, type)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.method = method
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncclass Method(Base):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def __init__(self, iface, name):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync Base.__init__(self, name)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.iface = iface
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.in_args = []
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.out_args = []
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def validate(self):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if not self.only_basic_args() and not self.use_raw_handler():
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync raise DBusXmlException("Method has complex arguments and requires " +
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync "the 'org.freedesktop.sssd.RawHandler' annotation")
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def fq_c_name(self):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return "%s_%s" % (self.iface.c_name(), self.c_name())
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def use_raw_handler(self):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync anno = 'org.freedesktop.sssd.RawHandler'
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return self.annotations.get(anno, self.iface.annotations.get(anno)) == 'true'
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def in_signature(self):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return "".join([arg.type for arg in self.in_args])
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def only_basic_args(self):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync for arg in self.in_args + self.out_args:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if not arg.is_basic:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return False
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return True
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncclass Signal(Base):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def __init__(self, iface, name):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync Base.__init__(self, name)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.iface = iface
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.args = []
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def fq_c_name(self):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return "%s_%s" % (self.iface.c_name(), self.c_name())
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncclass Property(Typed):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def __init__(self, iface, name, type, access):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync Typed.__init__(self, name, type)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.iface = iface
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.readable = False
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.writable = False
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if access == 'readwrite':
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.readable = True
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.writable = True
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync elif access == 'read':
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.readable = True
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync elif access == 'write':
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.writable = True
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync else:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync raise DBusXmlException('Invalid access type %s'%self.access)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def fq_c_name(self):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return "%s_%s" % (self.iface.c_name(), self.c_name())
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def getter_name(self):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return "%s_get_%s" % (self.iface.c_name(), self.c_name())
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def getter_invoker_name(self):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return "invoke_get_%s" % self.type
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def getter_signature(self, name):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync sig = "void (*%s)(struct sbus_request *, void *data, %s *" % (name, self.sssd_type)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if self.is_array:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync sig += " *, int *"
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync sig += ")"
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return sig
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncclass Interface(Base):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def __init__(self, name):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync Base.__init__(self, name)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.methods = []
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.signals = []
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.properties = []
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync def c_name(self):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return self.annotations.get("org.freedesktop.DBus.GLib.CSymbol",
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync self.name.replace(".", "_"))
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# -----------------------------------------------------------------------------
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync# Code Generation
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncdef out(format, *args, **kwargs):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync str = format % args
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync sys.stdout.write(str)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync # NOTE: Would like to use the following syntax for this function
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync # but need to wait until python3 until it is supported:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync # def out(format, *args, new_line=True)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if kwargs.pop("new_line", True):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync sys.stdout.write("\n")
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync assert not kwargs, "unknown keyword argument(s): %s" % str(kwargs)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncdef method_arg_types(args, with_names=False):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync str = ""
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync for arg in args:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync str += ", "
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync str += arg.sssd_type
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if with_names:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if str[-1] != '*':
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync str += " "
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync str += "arg_"
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync str += arg.c_name()
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if arg.is_array:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync str += "[], int"
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if with_names:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync str += " len_"
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync str += arg.c_name()
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return str
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncdef method_function_pointer(meth, name, with_names=False):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if meth.use_raw_handler():
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return "sbus_msg_handler_fn " + name
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync else:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return "int (*%s)(struct sbus_request *%s, void *%s%s)" % \
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync (name, with_names and "req" or "",
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync with_names and "data" or "",
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync method_arg_types(meth.in_args, with_names))
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncdef property_handlers(prop):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync return prop.getter_signature(prop.getter_name())
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncdef forward_method_invoker(signature, args):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out("")
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out("/* invokes a handler with a '%s' DBus signature */", signature)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out("static int invoke_%s_method(struct sbus_request *dbus_req, void *function_ptr);", signature)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncdef source_method_invoker(signature, args):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out("")
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out("/* invokes a handler with a '%s' DBus signature */", signature)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out("static int invoke_%s_method(struct sbus_request *dbus_req, void *function_ptr)", signature)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out("{")
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync for i in range(0, len(args)):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync arg = args[i]
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if arg.is_array:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(" %s *arg_%d;", arg.dbus_type, i)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(" int len_%d;", i)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync else:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(" %s arg_%d;", arg.dbus_type, i)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(" int (*handler)(struct sbus_request *, void *%s) = function_ptr;", method_arg_types(args))
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out("")
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(" if (!sbus_request_parse_or_finish(dbus_req,")
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync for i in range(0, len(args)):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync arg = args[i]
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if arg.is_array:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(" DBUS_TYPE_ARRAY, %s, &arg_%d, &len_%d,",
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync arg.dbus_constant, i, i)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync else:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(" %s, &arg_%d,", arg.dbus_constant, i)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(" DBUS_TYPE_INVALID)) {")
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(" return EOK; /* request handled */")
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(" }")
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out("")
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(" return (handler)(dbus_req, dbus_req->intf->handler_data", new_line=False)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync for i in range(0, len(args)):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync arg = args[i]
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(",\n arg_%d", i, new_line=False)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if arg.is_array:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(",\n len_%d", i, new_line=False)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(");")
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out("}")
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncdef source_prop_types(prop, type_prefix=False):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync prefix = "%s_" % prop.type if type_prefix else ""
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync if prop.is_array:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(" %s *%sprop_val;", prop.sssd_type, prefix)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(" int %sprop_len;", prefix)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(" %s *%sout_val;", prop.dbus_type, prefix)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync else:
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(" %s %sprop_val;", prop.sssd_type, prefix)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(" %s %sout_val;", prop.dbus_type, prefix)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsyncdef source_prop_handler(prop, type_prefix=False):
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync prefix = "%s_" % prop.type if type_prefix else ""
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(" %s", prop.getter_signature("%shandler" % prefix), new_line=False)
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync out(";")
4fd606d1f5abe38e1f42c38de1d2e895166bd0f4vboxsync
def source_getter_invoker(prop):
out("")
if prop.is_array:
out("/* invokes a getter with an array of '%s' DBus type */", prop.dbus_type)
else:
out("/* invokes a getter with a '%s' DBus type */", prop.dbus_type)
out("static int %s(struct sbus_request *dbus_req, void *function_ptr)",
prop.getter_invoker_name())
out("{")
source_prop_types(prop)
out("")
out(" %s", prop.getter_signature("handler"), new_line=False)
out(" = function_ptr;")
out("")
out(" (handler)(dbus_req, dbus_req->intf->handler_data, &prop_val", new_line=False)
if prop.is_array:
out(", &prop_len", new_line=False)
out(");")
out("")
if prop.type == "s":
out(" out_val = prop_val == NULL ? \"\" : prop_val;")
elif prop.type == "o":
out(" out_val = prop_val == NULL ? \"/\" : prop_val;")
else:
out(" out_val = prop_val;")
if prop.is_array:
out(" return sbus_request_return_array_as_variant(dbus_req, %s, (uint8_t*)out_val, prop_len, sizeof(%s));", prop.dbus_constant, prop.sssd_type)
else:
out(" return sbus_request_return_as_variant(dbus_req, %s, &out_val);", prop.dbus_constant)
out("}")
def source_getall_invoker(iface, prop_invokers):
out("")
out("/* invokes GetAll for the '%s' interface */", iface.name)
out("static int invoke_%s_get_all(struct sbus_request *dbus_req, void *function_ptr)",
iface.c_name())
out("{")
out(" DBusMessage *reply;")
out(" dbus_bool_t dbret;")
out(" DBusMessageIter iter;")
out(" DBusMessageIter iter_dict;")
if iface.properties:
out(" int ret;")
out(" struct sbus_interface *intf = dbus_req->intf;")
out(" const struct sbus_property_meta *property;")
iface_types = [ p.type for p in iface.properties ]
for prop in [ p for p in prop_invokers.values() if p.type in iface_types ]:
source_prop_types(prop, type_prefix=True)
source_prop_handler(prop, type_prefix=True)
out("")
out(" reply = dbus_message_new_method_return(dbus_req->message);")
out(" if (!reply) return ENOMEM;")
out(" dbus_message_iter_init_append(reply, &iter);")
out(" dbret = dbus_message_iter_open_container(")
out(" &iter, DBUS_TYPE_ARRAY,")
out(" DBUS_DICT_ENTRY_BEGIN_CHAR_AS_STRING")
out(" DBUS_TYPE_STRING_AS_STRING")
out(" DBUS_TYPE_VARIANT_AS_STRING")
out(" DBUS_DICT_ENTRY_END_CHAR_AS_STRING,")
out(" &iter_dict);")
out(" if (!dbret) return ENOMEM;")
out("")
for prop in iface.properties:
out(" property = sbus_meta_find_property(intf->vtable->meta, \"%s\");", prop.c_name())
out(" if (property != NULL && property->flags & SBUS_PROPERTY_READABLE) {")
out(" %s_handler = VTABLE_FUNC(intf->vtable, property->vtable_offset_get);", prop.type)
out(" if (%s_handler) {", prop.type)
out(" (%s_handler)(dbus_req, dbus_req->intf->handler_data, &%s_prop_val", prop.type, prop.type, new_line=False)
if prop.is_array:
out(", &%s_prop_len", prop.type, new_line=False)
out(");")
if prop.type == "s":
out(" %s_out_val = %s_prop_val == NULL ? \"\" : %s_prop_val;",
prop.type, prop.type, prop.type)
elif prop.type == "o":
out(" %s_out_val = %s_prop_val == NULL ? \"/\" : %s_prop_val;",
prop.type, prop.type, prop.type)
else:
out(" %s_out_val = %s_prop_val;", prop.type, prop.type)
if prop.is_array:
out(" ret = sbus_add_array_as_variant_to_dict(&iter_dict, \"%s\", %s, (uint8_t*)%s_out_val, %s_prop_len, sizeof(%s));", prop.c_name(), prop.dbus_constant, prop.type, prop.type, prop.sssd_type)
else:
out(" ret = sbus_add_variant_to_dict(&iter_dict, \"%s\", %s, &%s_out_val);", prop.c_name(), prop.dbus_constant, prop.type)
out(" if (ret != EOK) return ret;")
out(" }")
out(" }")
out("")
out(" dbret = dbus_message_iter_close_container(&iter, &iter_dict);")
out(" if (!dbret) return ENOMEM;")
out("")
out(" return sbus_request_finish(dbus_req, reply);")
out("}")
def forward_method_invokers(ifaces):
invokers = { }
for iface in ifaces:
for meth in iface.methods:
if meth.use_raw_handler() or not meth.in_args:
continue
signature = meth.in_signature()
if signature in invokers:
continue
forward_method_invoker(signature, meth.in_args)
invokers[signature] = meth
return invokers
def source_method_invokers(invokers):
for (signature, meth) in invokers.items():
source_method_invoker(signature, meth.in_args)
def forward_prop_invoker(prop):
out("static int %s(struct sbus_request *dbus_req, void *function_ptr);",
prop.getter_invoker_name())
def forward_prop_invokers(ifaces):
invokers = { }
for iface in ifaces:
for prop in iface.properties:
if not prop.is_basic:
continue
if prop.type in invokers:
continue
forward_prop_invoker(prop)
invokers[prop.type] = prop
return invokers
def source_prop_invokers(invokers):
for (type, prop) in invokers.items():
if prop.readable:
source_getter_invoker(prop)
def source_finisher(meth):
out("")
out("int %s_finish(struct sbus_request *req%s)",
meth.fq_c_name(), method_arg_types(meth.out_args, with_names=True))
out("{")
for arg in meth.out_args:
if arg.dbus_type != arg.sssd_type:
out(" %s cast_%s = arg_%s;", arg.dbus_type, arg.c_name(), arg.c_name())
out(" return sbus_request_return_and_finish(req,")
for arg in meth.out_args:
out(" ", new_line=False)
if arg.is_array:
out("DBUS_TYPE_ARRAY, %s, &arg_%s, len_%s,",
arg.dbus_constant, arg.c_name(), arg.c_name())
elif arg.dbus_type != arg.sssd_type:
out("%s, &cast_%s,", arg.dbus_constant, arg.c_name())
else:
out("%s, &arg_%s,", arg.dbus_constant, arg.c_name())
out(" DBUS_TYPE_INVALID);")
out("}")
def header_reply(meth):
for arg in meth.out_args:
if arg.is_array:
out(" %s *%s", arg.dbus_type, arg.c_name())
out(" int %s__len", arg.c_name())
else:
out(" %s %s;", arg.dbus_type, arg.c_name())
types = [arg.sssd_type for arg in meth.in_args]
def source_args(parent, args, suffix):
out("")
out("/* arguments for %s.%s */", parent.iface.name, parent.name)
out("const struct sbus_arg_meta %s%s[] = {", parent.fq_c_name(), suffix)
for arg in args:
out(" { \"%s\", \"%s\" },", arg.name, arg.type)
out(" { NULL, }")
out("};")
def source_methods(iface, methods):
for meth in methods:
if meth.in_args:
source_args(meth, meth.in_args, "__in")
if meth.out_args:
source_args(meth, meth.out_args, "__out")
if not meth.use_raw_handler():
source_finisher(meth)
out("")
out("/* methods for %s */", iface.name)
out("const struct sbus_method_meta %s__methods[] = {", iface.c_name())
for meth in methods:
out(" {")
out(" \"%s\", /* name */", meth.name)
if meth.in_args:
out(" %s__in,", meth.fq_c_name())
else:
out(" NULL, /* no in_args */")
if meth.out_args:
out(" %s__out,", meth.fq_c_name())
else:
out(" NULL, /* no out_args */")
out(" offsetof(struct %s, %s),", iface.c_name(), meth.c_name())
if meth.use_raw_handler() or not meth.in_args:
out(" NULL, /* no invoker */")
else:
out(" invoke_%s_method,", meth.in_signature())
out(" },")
out(" { NULL, }")
out("};")
def source_signals(iface, signals):
for sig in iface.signals:
if sig.args:
source_args(sig, sig.args, "__args")
out("")
out("/* signals for %s */", iface.name)
out("const struct sbus_signal_meta %s__signals[] = {", iface.c_name())
for sig in signals:
out(" {")
out(" \"%s\", /* name */", sig.name)
if sig.args:
out(" %s__args", sig.fq_c_name())
else:
out(" NULL, /* no args */")
out(" },")
out(" { NULL, }")
out("};")
def source_properties(iface, properties):
out("")
out("/* property info for %s */", iface.name)
out("const struct sbus_property_meta %s__properties[] = {", iface.c_name())
for prop in properties:
out(" {")
out(" \"%s\", /* name */", prop.name)
out(" \"%s\", /* type */", prop.type)
if prop.readable and prop.writable:
out(" SBUS_PROPERTY_READABLE | SBUS_PROPERTY_WRITABLE,")
elif prop.readable:
out(" SBUS_PROPERTY_READABLE,")
elif prop.writable:
out(" SBUS_PROPERTY_WRITABLE,")
else:
assert False, "should not be reached"
if prop.readable:
out(" offsetof(struct %s, %s),", iface.c_name(), prop.getter_name())
out(" %s,", prop.getter_invoker_name())
else:
out(" 0, /* not readable */")
out(" NULL, /* no invoker */")
out(" 0, /* not writable */")
out(" NULL, /* no invoker */")
out(" },")
out(" { NULL, }")
out("};")
def header_interface(iface):
out("")
out("/* interface info for %s */", iface.name)
out("extern const struct sbus_interface_meta %s_meta;", iface.c_name())
def source_interface(iface):
out("")
out("/* interface info for %s */", iface.name)
out("const struct sbus_interface_meta %s_meta = {", iface.c_name())
out(" \"%s\", /* name */", iface.name)
if iface.methods:
out(" %s__methods,", iface.c_name())
else:
out(" NULL, /* no methods */")
if iface.signals:
out(" %s__signals,", iface.c_name())
else:
out(" NULL, /* no signals */")
if iface.properties:
out(" %s__properties,", iface.c_name())
else:
out(" NULL, /* no properties */")
out(" invoke_%s_get_all, /* GetAll invoker */", iface.c_name())
out("};")
def generate_source(ifaces, filename, include_header=None):
basename = os.path.basename(filename)
out("/* The following definitions are auto-generated from %s */", basename)
out("")
out("#include \"util/util.h\"")
out("#include \"sbus/sssd_dbus.h\"")
out("#include \"sbus/sssd_dbus_meta.h\"")
if include_header:
out("#include \"%s\"", os.path.basename(include_header))
meth_invokers = forward_method_invokers(ifaces)
prop_invokers = forward_prop_invokers(ifaces)
for iface in ifaces:
# The methods
if iface.methods:
source_methods(iface, iface.methods)
# The signals array
if iface.signals:
source_signals(iface, iface.signals)
# The properties array
if iface.properties:
source_properties(iface, iface.properties)
# always generate getall, for interfaces without properties
# let's return an empty array
source_getall_invoker(iface, prop_invokers)
# The sbus_interface structure
source_interface(iface)
source_method_invokers(meth_invokers)
source_prop_invokers(prop_invokers)
def header_finisher(iface, meth):
if meth.use_raw_handler():
return
out("")
out("/* finish function for %s */", meth.name)
out("int %s_finish(struct sbus_request *req%s);",
meth.fq_c_name(), method_arg_types(meth.out_args, with_names=True))
def header_vtable(iface, methods):
out("")
out("/* vtable for %s */", iface.name)
out("struct %s {", iface.c_name())
out(" struct sbus_vtable vtable; /* derive from sbus_vtable */")
# All methods
for meth in iface.methods:
out(" %s;", method_function_pointer(meth, meth.c_name(), with_names=True))
for prop in iface.properties:
out(" %s;", property_handlers(prop))
out("};")
def header_constants(iface):
out("")
out("/* constants for %s */", iface.name)
out("#define %s \"%s\"", iface.c_name().upper(), iface.name)
for meth in iface.methods:
out("#define %s \"%s\"", meth.fq_c_name().upper(), meth.name)
for sig in iface.signals:
out("#define %s \"%s\"", sig.fq_c_name().upper(), sig.name)
for prop in iface.properties:
out("#define %s \"%s\"", prop.fq_c_name().upper(), prop.name)
def generate_header(ifaces, filename):
basename = os.path.basename(filename)
guard = "__%s__" % re.sub(r'([^_A-Z0-9])', "_", basename.upper())
out("/* The following declarations are auto-generated from %s */", basename)
out("")
out("#ifndef %s", guard)
out("#define %s", guard)
out("")
out("#include \"sbus/sssd_dbus.h\"")
out("")
out("/* ------------------------------------------------------------------------")
out(" * DBus Constants")
out(" *")
out(" * Various constants of interface and method names mostly for use by clients")
out(" */")
for iface in ifaces:
header_constants(iface)
out("")
out("/* ------------------------------------------------------------------------")
out(" * DBus handlers")
out(" *")
out(" * These structures are filled in by implementors of the different")
out(" * dbus interfaces to handle method calls.")
out(" *")
out(" * Handler functions of type sbus_msg_handler_fn accept raw messages,")
out(" * other handlers are typed appropriately. If a handler that is")
out(" * set to NULL is invoked it will result in a")
out(" * org.freedesktop.DBus.Error.NotSupported error for the caller.")
out(" *")
out(" * Handlers have a matching xxx_finish() function (unless the method has")
out(" * accepts raw messages). These finish functions the")
out(" * sbus_request_return_and_finish() with the appropriate arguments to")
out(" * construct a valid reply. Once a finish function has been called, the")
out(" * @dbus_req it was called with is freed and no longer valid.")
out(" */")
for iface in ifaces:
if iface.methods or iface.properties:
header_vtable(iface, iface.methods)
for meth in iface.methods:
header_finisher(iface, meth)
out("")
out("/* ------------------------------------------------------------------------")
out(" * DBus Interface Metadata")
out(" *")
out(" * These structure definitions are filled in with the information about")
out(" * the interfaces, methods, properties and so on.")
out(" *")
out(" * The actual definitions are found in the accompanying C file next")
out(" * to this header.")
out(" */")
for iface in ifaces:
header_interface(iface)
out("")
out("#endif /* %s */", guard)
# -----------------------------------------------------------------------------
# XML Interface Parsing
STATE_TOP = 'top'
STATE_NODE = 'node'
STATE_INTERFACE = 'interface'
STATE_METHOD = 'method'
STATE_SIGNAL = 'signal'
STATE_PROPERTY = 'property'
STATE_ARG = 'arg'
STATE_ANNOTATION = 'annotation'
STATE_IGNORED = 'ignored'
def expect_attr(attrs, name):
if name not in attrs:
raise DBusXmlException("Missing attribute '%s'" % name)
if attrs[name] == "":
raise DBusXmlException("Empty attribute '%s'" % name)
return attrs[name]
class DBusXMLParser:
def __init__(self, filename):
parser = xml.parsers.expat.ParserCreate()
parser.CommentHandler = self.handle_comment
parser.CharacterDataHandler = self.handle_char_data
parser.StartElementHandler = self.handle_start_element
parser.EndElementHandler = self.handle_end_element
self.parsed_interfaces = []
self.cur_object = None
self.state = STATE_TOP
self.state_stack = []
self.cur_object = None
self.cur_object_stack = []
self.arg_count = 0
try:
with open(filename, "r") as f:
parser.ParseFile(f)
except DBusXmlException, ex:
ex.line = parser.CurrentLineNumber
ex.file = filename
raise
except xml.parsers.expat.ExpatError, ex:
exc = DBusXmlException(str(ex))
exc.line = ex.lineno
exc.file = filename
raise exc
def handle_comment(self, data):
pass
def handle_char_data(self, data):
pass
def handle_start_element(self, name, attrs):
old_state = self.state
old_cur_object = self.cur_object
if self.state == STATE_IGNORED:
self.state = STATE_IGNORED
elif self.cur_object and name == STATE_ANNOTATION:
val = attrs.get('value', '')
self.cur_object.annotations[expect_attr(attrs, 'name')] = val
self.state = STATE_IGNORED
elif self.state == STATE_TOP:
if name == STATE_NODE:
self.state = STATE_NODE
else:
self.state = STATE_IGNORED
elif self.state == STATE_NODE:
if name == STATE_INTERFACE:
self.state = STATE_INTERFACE
iface = Interface(expect_attr(attrs, 'name'))
self.cur_object = iface
self.parsed_interfaces.append(iface)
else:
self.state = STATE_IGNORED
elif self.state == STATE_INTERFACE:
if name == STATE_METHOD:
self.state = STATE_METHOD
method = Method(self.cur_object, expect_attr(attrs, 'name'))
self.cur_object.methods.append(method)
self.cur_object = method
self.arg_count = 0
elif name == STATE_SIGNAL:
self.state = STATE_SIGNAL
signal = Signal(self.cur_object, expect_attr(attrs, 'name'))
self.cur_object.signals.append(signal)
self.cur_object = signal
self.arg_count = 0
elif name == STATE_PROPERTY:
self.state = STATE_PROPERTY
prop = Property(self.cur_object,
expect_attr(attrs, 'name'),
expect_attr(attrs, 'type'),
expect_attr(attrs, 'access'))
self.cur_object.properties.append(prop)
self.cur_object = prop
else:
self.state = STATE_IGNORED
elif self.state == STATE_METHOD:
if name == STATE_ARG:
self.state = STATE_ARG
arg = Arg(self.cur_object,
expect_attr(attrs, 'name'),
expect_attr(attrs, 'type'))
direction = attrs.get('direction', 'in')
if direction == 'in':
self.cur_object.in_args.append(arg)
elif direction == 'out':
self.cur_object.out_args.append(arg)
else:
raise DBusXmlException('Invalid direction "%s"' % direction)
self.cur_object = arg
else:
self.state = STATE_IGNORED
elif self.state == STATE_SIGNAL:
if name == STATE_ARG:
self.state = STATE_ARG
arg = Arg(self.cur_object,
expect_attr(attrs, 'name'),
expect_attr(attrs, 'type'))
self.cur_object.args.append(arg)
self.cur_object = arg
else:
self.state = STATE_IGNORED
elif self.state == STATE_PROPERTY:
self.state = STATE_IGNORED
elif self.state == STATE_ARG:
self.state = STATE_IGNORED
else:
assert False, 'Unhandled state "%s" while entering element with name "%s"' % (self.state, name)
self.state_stack.append(old_state)
self.cur_object_stack.append(old_cur_object)
def handle_end_element(self, name):
if self.cur_object:
self.cur_object.validate()
self.state = self.state_stack.pop()
self.cur_object = self.cur_object_stack.pop()
def parse_options():
parser = optparse.OptionParser("usage: %prog [options] introspect.xml ...")
parser.set_description("sbus_codegen generates sbus interface structures \
from standard XML Introspect data.")
parser.add_option("--mode",
dest="mode", default="header",
help="'header' or 'source' (default: header)",
metavar="MODE")
parser.add_option("--output",
dest="output", default=None,
help="Set output file name (default: stdout)",
metavar="FILE")
parser.add_option("--include",
dest="include", default=None,
help="name of a header to #include",
metavar="HEADER")
(options, args) = parser.parse_args()
if not args:
print >> sys.stderr, "sbus_codegen: no input file specified"
sys.exit(2)
if options.mode not in ["header", "source"]:
print >> sys.stderr, "sbus_codegen: specify --mode=header or --mode=source"
return options, args
def main():
options, args = parse_options()
if options.output:
sys.stdout = buf = StringIO.StringIO()
for filename in args:
parser = DBusXMLParser(filename)
if options.mode == "header":
generate_header(parser.parsed_interfaces, filename)
elif options.mode == "source":
generate_source(parser.parsed_interfaces, filename, options.include)
else:
assert False, "should not be reached"
# Write output at end to be nice to 'make'
if options.output:
output = open(options.output, "w")
output.write(buf.getvalue())
output.close()
if __name__ == "__main__":
try:
main()
except DBusXmlException, ex:
print >> sys.stderr, str(ex)
sys.exit(1)