/* * wpa_supplicant - D-Bus introspection * Copyright (c) 2006, Dan Williams <dcbw@redhat.com> and Red Hat, Inc. * Copyright (c) 2009, Witold Sowa <witold.sowa@gmail.com> * Copyright (c) 2010, Jouni Malinen <j@w1.fi> * * This software may be distributed under the terms of the BSD license. * See README for more details. */ #include "utils/includes.h" #include "utils/common.h" #include "utils/list.h" #include "utils/wpabuf.h" #include "dbus_common_i.h" #include "dbus_new_helpers.h" struct interfaces { struct dl_list list; char *dbus_interface; struct wpabuf *xml; }; static struct interfaces * add_interface(struct dl_list *list, const char *dbus_interface) { struct interfaces *iface; dl_list_for_each(iface, list, struct interfaces, list) { if (os_strcmp(iface->dbus_interface, dbus_interface) == 0) return iface; /* already in the list */ } iface = os_zalloc(sizeof(struct interfaces)); if (!iface) return NULL; iface->xml = wpabuf_alloc(6000); if (iface->xml == NULL) { os_free(iface); return NULL; } wpabuf_printf(iface->xml, "<interface name=\"%s\">", dbus_interface); dl_list_add_tail(list, &iface->list); iface->dbus_interface = os_strdup(dbus_interface); return iface; } static void add_arg(struct wpabuf *xml, const char *name, const char *type, const char *direction) { wpabuf_printf(xml, "<arg name=\"%s\"", name); if (type) wpabuf_printf(xml, " type=\"%s\"", type); if (direction) wpabuf_printf(xml, " direction=\"%s\"", direction); wpabuf_put_str(xml, "/>"); } static void add_entry(struct wpabuf *xml, const char *type, const char *name, const struct wpa_dbus_argument *args, int include_dir) { const struct wpa_dbus_argument *arg; if (args == NULL || args->name == NULL) { wpabuf_printf(xml, "<%s name=\"%s\"/>", type, name); return; } wpabuf_printf(xml, "<%s name=\"%s\">", type, name); for (arg = args; arg && arg->name; arg++) { add_arg(xml, arg->name, arg->type, include_dir ? (arg->dir == ARG_IN ? "in" : "out") : NULL); } wpabuf_printf(xml, "</%s>", type); } static void add_property(struct wpabuf *xml, const struct wpa_dbus_property_desc *dsc) { wpabuf_printf(xml, "<property name=\"%s\" type=\"%s\" " "access=\"%s%s\"/>", dsc->dbus_property, dsc->type, dsc->getter ? "read" : "", dsc->setter ? "write" : ""); } static void extract_interfaces_methods( struct dl_list *list, const struct wpa_dbus_method_desc *methods) { const struct wpa_dbus_method_desc *dsc; struct interfaces *iface; for (dsc = methods; dsc && dsc->dbus_method; dsc++) { iface = add_interface(list, dsc->dbus_interface); if (iface) add_entry(iface->xml, "method", dsc->dbus_method, dsc->args, 1); } } static void extract_interfaces_signals( struct dl_list *list, const struct wpa_dbus_signal_desc *signals) { const struct wpa_dbus_signal_desc *dsc; struct interfaces *iface; for (dsc = signals; dsc && dsc->dbus_signal; dsc++) { iface = add_interface(list, dsc->dbus_interface); if (iface) add_entry(iface->xml, "signal", dsc->dbus_signal, dsc->args, 0); } } static void extract_interfaces_properties( struct dl_list *list, const struct wpa_dbus_property_desc *properties) { const struct wpa_dbus_property_desc *dsc; struct interfaces *iface; for (dsc = properties; dsc && dsc->dbus_property; dsc++) { iface = add_interface(list, dsc->dbus_interface); if (iface) add_property(iface->xml, dsc); } } /** * extract_interfaces - Extract interfaces from methods, signals and props * @list: Interface list to be filled * @obj_dsc: Description of object from which interfaces will be extracted * * Iterates over all methods, signals, and properties registered with an * object and collects all declared DBus interfaces and create interfaces' * node in XML root node for each. Returned list elements contain interface * name and XML node of corresponding interface. */ static void extract_interfaces(struct dl_list *list, struct wpa_dbus_object_desc *obj_dsc) { extract_interfaces_methods(list, obj_dsc->methods); extract_interfaces_signals(list, obj_dsc->signals); extract_interfaces_properties(list, obj_dsc->properties); } static void add_interfaces(struct dl_list *list, struct wpabuf *xml) { struct interfaces *iface, *n; dl_list_for_each_safe(iface, n, list, struct interfaces, list) { if (wpabuf_len(iface->xml) + 20 < wpabuf_tailroom(xml)) { wpabuf_put_buf(xml, iface->xml); wpabuf_put_str(xml, "</interface>"); } else { wpa_printf(MSG_DEBUG, "dbus: Not enough room for " "add_interfaces inspect data: tailroom %u, " "add %u", (unsigned int) wpabuf_tailroom(xml), (unsigned int) wpabuf_len(iface->xml)); } dl_list_del(&iface->list); wpabuf_free(iface->xml); os_free(iface->dbus_interface); os_free(iface); } } static void add_child_nodes(struct wpabuf *xml, DBusConnection *con, const char *path) { char **children; int i; /* add child nodes to introspection tree */ dbus_connection_list_registered(con, path, &children); for (i = 0; children[i]; i++) wpabuf_printf(xml, "<node name=\"%s\"/>", children[i]); dbus_free_string_array(children); } static void add_introspectable_interface(struct wpabuf *xml) { wpabuf_printf(xml, "<interface name=\"%s\">" "<method name=\"%s\">" "<arg name=\"data\" type=\"s\" direction=\"out\"/>" "</method>" "</interface>", WPA_DBUS_INTROSPECTION_INTERFACE, WPA_DBUS_INTROSPECTION_METHOD); } static void add_properties_interface(struct wpabuf *xml) { wpabuf_printf(xml, "<interface name=\"%s\">", WPA_DBUS_PROPERTIES_INTERFACE); wpabuf_printf(xml, "<method name=\"%s\">", WPA_DBUS_PROPERTIES_GET); add_arg(xml, "interface", "s", "in"); add_arg(xml, "propname", "s", "in"); add_arg(xml, "value", "v", "out"); wpabuf_put_str(xml, "</method>"); wpabuf_printf(xml, "<method name=\"%s\">", WPA_DBUS_PROPERTIES_GETALL); add_arg(xml, "interface", "s", "in"); add_arg(xml, "props", "a{sv}", "out"); wpabuf_put_str(xml, "</method>"); wpabuf_printf(xml, "<method name=\"%s\">", WPA_DBUS_PROPERTIES_SET); add_arg(xml, "interface", "s", "in"); add_arg(xml, "propname", "s", "in"); add_arg(xml, "value", "v", "in"); wpabuf_put_str(xml, "</method>"); wpabuf_put_str(xml, "</interface>"); } static void add_wpas_interfaces(struct wpabuf *xml, struct wpa_dbus_object_desc *obj_dsc) { struct dl_list ifaces; dl_list_init(&ifaces); extract_interfaces(&ifaces, obj_dsc); add_interfaces(&ifaces, xml); } /** * wpa_dbus_introspect - Responds for Introspect calls on object * @message: Message with Introspect call * @obj_dsc: Object description on which Introspect was called * Returns: Message with introspection result XML string as only argument * * Iterates over all methods, signals and properties registered with * object and generates introspection data for the object as XML string. */ DBusMessage * wpa_dbus_introspect(DBusMessage *message, struct wpa_dbus_object_desc *obj_dsc) { DBusMessage *reply; struct wpabuf *xml; xml = wpabuf_alloc(10000); if (xml == NULL) return NULL; wpabuf_put_str(xml, "<?xml version=\"1.0\"?>\n"); wpabuf_put_str(xml, DBUS_INTROSPECT_1_0_XML_DOCTYPE_DECL_NODE); wpabuf_put_str(xml, "<node>"); add_introspectable_interface(xml); add_properties_interface(xml); add_wpas_interfaces(xml, obj_dsc); add_child_nodes(xml, obj_dsc->connection, dbus_message_get_path(message)); wpabuf_put_str(xml, "</node>\n"); reply = dbus_message_new_method_return(message); if (reply) { const char *intro_str = wpabuf_head(xml); dbus_message_append_args(reply, DBUS_TYPE_STRING, &intro_str, DBUS_TYPE_INVALID); } wpabuf_free(xml); return reply; }