dbus_new_introspect.c 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326
  1. /*
  2. * wpa_supplicant - D-Bus introspection
  3. * Copyright (c) 2006, Dan Williams <dcbw@redhat.com> and Red Hat, Inc.
  4. * Copyright (c) 2009, Witold Sowa <witold.sowa@gmail.com>
  5. * Copyright (c) 2010, Jouni Malinen <j@w1.fi>
  6. *
  7. * This program is free software; you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License version 2 as
  9. * published by the Free Software Foundation.
  10. *
  11. * Alternatively, this software may be distributed under the terms of BSD
  12. * license.
  13. *
  14. * See README and COPYING for more details.
  15. */
  16. #include "utils/includes.h"
  17. #include "utils/common.h"
  18. #include "utils/list.h"
  19. #include "utils/wpabuf.h"
  20. #include "dbus_common_i.h"
  21. #include "dbus_new_helpers.h"
  22. struct interfaces {
  23. struct dl_list list;
  24. char *dbus_interface;
  25. struct wpabuf *xml;
  26. };
  27. static void add_interface(struct dl_list *list, const char *dbus_interface)
  28. {
  29. struct interfaces *iface;
  30. dl_list_for_each(iface, list, struct interfaces, list) {
  31. if (os_strcmp(iface->dbus_interface, dbus_interface) == 0)
  32. return; /* already in the list */
  33. }
  34. iface = os_zalloc(sizeof(struct interfaces));
  35. if (!iface)
  36. return;
  37. iface->xml = wpabuf_alloc(3000);
  38. if (iface->xml == NULL) {
  39. os_free(iface);
  40. return;
  41. }
  42. wpabuf_printf(iface->xml, "<interface name=\"%s\">", dbus_interface);
  43. dl_list_add_tail(list, &iface->list);
  44. iface->dbus_interface = os_strdup(dbus_interface);
  45. }
  46. static void extract_interfaces_methods(struct dl_list *list,
  47. struct wpa_dbus_method_desc *methods)
  48. {
  49. struct wpa_dbus_method_desc *dsc;
  50. for (dsc = methods; dsc; dsc = dsc->next)
  51. add_interface(list, dsc->dbus_interface);
  52. }
  53. static void extract_interfaces_signals(struct dl_list *list,
  54. struct wpa_dbus_signal_desc *signals)
  55. {
  56. struct wpa_dbus_signal_desc *dsc;
  57. for (dsc = signals; dsc; dsc = dsc->next)
  58. add_interface(list, dsc->dbus_interface);
  59. }
  60. static void extract_interfaces_properties(
  61. struct dl_list *list, struct wpa_dbus_property_desc *properties)
  62. {
  63. struct wpa_dbus_property_desc *dsc;
  64. for (dsc = properties; dsc; dsc = dsc->next)
  65. add_interface(list, dsc->dbus_interface);
  66. }
  67. /**
  68. * extract_interfaces - Extract interfaces from methods, signals and props
  69. * @list: Interface list to be filled
  70. * @obj_dsc: Description of object from which interfaces will be extracted
  71. *
  72. * Iterates over all methods, signals, and properties registered with an
  73. * object and collects all declared DBus interfaces and create interfaces'
  74. * node in XML root node for each. Returned list elements contain interface
  75. * name and XML node of corresponding interface.
  76. */
  77. static void extract_interfaces(struct dl_list *list,
  78. struct wpa_dbus_object_desc *obj_dsc)
  79. {
  80. extract_interfaces_methods(list, obj_dsc->methods);
  81. extract_interfaces_signals(list, obj_dsc->signals);
  82. extract_interfaces_properties(list, obj_dsc->properties);
  83. }
  84. static void add_interfaces(struct dl_list *list, struct wpabuf *xml)
  85. {
  86. struct interfaces *iface, *n;
  87. dl_list_for_each_safe(iface, n, list, struct interfaces, list) {
  88. wpabuf_put_buf(xml, iface->xml);
  89. wpabuf_put_str(xml, "</interface>");
  90. dl_list_del(&iface->list);
  91. wpabuf_free(iface->xml);
  92. os_free(iface->dbus_interface);
  93. os_free(iface);
  94. }
  95. }
  96. static struct interfaces * get_interface(struct dl_list *list,
  97. const char *dbus_interface)
  98. {
  99. struct interfaces *iface;
  100. dl_list_for_each(iface, list, struct interfaces, list) {
  101. if (os_strcmp(iface->dbus_interface, dbus_interface) == 0)
  102. return iface;
  103. }
  104. return NULL;
  105. }
  106. static void add_child_nodes(struct wpabuf *xml, DBusConnection *con,
  107. const char *path)
  108. {
  109. char **children;
  110. int i;
  111. /* add child nodes to introspection tree */
  112. dbus_connection_list_registered(con, path, &children);
  113. for (i = 0; children[i]; i++)
  114. wpabuf_printf(xml, "<node name=\"%s\"/>", children[i]);
  115. dbus_free_string_array(children);
  116. }
  117. static void add_arg(struct wpabuf *xml, const char *name, const char *type,
  118. const char *direction)
  119. {
  120. wpabuf_printf(xml, "<arg name=\"%s\"", name);
  121. if (type)
  122. wpabuf_printf(xml, " type=\"%s\"", type);
  123. if (direction)
  124. wpabuf_printf(xml, " direction=\"%s\"", direction);
  125. wpabuf_put_str(xml, "/>");
  126. }
  127. static void add_introspectable_interface(struct wpabuf *xml)
  128. {
  129. wpabuf_printf(xml, "<interface name=\"%s\">"
  130. "<method name=\"%s\">"
  131. "<arg name=\"data\" type=\"s\" direction=\"out\"/>"
  132. "</method>"
  133. "</interface>",
  134. WPA_DBUS_INTROSPECTION_INTERFACE,
  135. WPA_DBUS_INTROSPECTION_METHOD);
  136. }
  137. static void add_properties_interface(struct wpabuf *xml)
  138. {
  139. wpabuf_printf(xml, "<interface name=\"%s\">",
  140. WPA_DBUS_PROPERTIES_INTERFACE);
  141. wpabuf_printf(xml, "<method name=\"%s\">", WPA_DBUS_PROPERTIES_GET);
  142. add_arg(xml, "interface", "s", "in");
  143. add_arg(xml, "propname", "s", "in");
  144. add_arg(xml, "value", "v", "out");
  145. wpabuf_put_str(xml, "</method>");
  146. wpabuf_printf(xml, "<method name=\"%s\">", WPA_DBUS_PROPERTIES_GETALL);
  147. add_arg(xml, "interface", "s", "in");
  148. add_arg(xml, "props", "a{sv}", "out");
  149. wpabuf_put_str(xml, "</method>");
  150. wpabuf_printf(xml, "<method name=\"%s\">", WPA_DBUS_PROPERTIES_SET);
  151. add_arg(xml, "interface", "s", "in");
  152. add_arg(xml, "propname", "s", "in");
  153. add_arg(xml, "value", "v", "in");
  154. wpabuf_put_str(xml, "</method>");
  155. wpabuf_put_str(xml, "</interface>");
  156. }
  157. static void add_entry(struct wpabuf *xml, char *type, char *name, int args_num,
  158. struct wpa_dbus_argument *args, int include_dir)
  159. {
  160. int i;
  161. if (args_num == 0) {
  162. wpabuf_printf(xml, "<%s name=\"%s\"/>", type, name);
  163. return;
  164. }
  165. wpabuf_printf(xml, "<%s name=\"%s\">", type, name);
  166. for (i = 0; i < args_num; i++) {
  167. struct wpa_dbus_argument *arg = &args[i];
  168. add_arg(xml, arg->name, arg->type,
  169. include_dir ? (arg->dir == ARG_IN ? "in" : "out") :
  170. NULL);
  171. }
  172. wpabuf_printf(xml, "</%s>", type);
  173. }
  174. static void add_property(struct wpabuf *xml,
  175. struct wpa_dbus_property_desc *dsc)
  176. {
  177. wpabuf_printf(xml, "<property name=\"%s\" type=\"%s\" access=\"%s\"/>",
  178. dsc->dbus_property, dsc->type,
  179. (dsc->access == R ? "read" :
  180. (dsc->access == W ? "write" : "readwrite")));
  181. }
  182. static void create_method_nodes(struct dl_list *list,
  183. struct wpa_dbus_method_desc *methods)
  184. {
  185. struct wpa_dbus_method_desc *dsc;
  186. struct interfaces *iface;
  187. for (dsc = methods; dsc; dsc = dsc->next) {
  188. iface = get_interface(list, dsc->dbus_interface);
  189. if (!iface)
  190. continue;
  191. add_entry(iface->xml, "method", dsc->dbus_method,
  192. dsc->args_num, dsc->args, 1);
  193. }
  194. }
  195. static void create_signal_nodes(struct dl_list *list,
  196. struct wpa_dbus_signal_desc *signals)
  197. {
  198. struct wpa_dbus_signal_desc *dsc;
  199. struct interfaces *iface;
  200. for (dsc = signals; dsc; dsc = dsc->next) {
  201. iface = get_interface(list, dsc->dbus_interface);
  202. if (!iface)
  203. continue;
  204. add_entry(iface->xml, "signal", dsc->dbus_signal,
  205. dsc->args_num, dsc->args, 0);
  206. }
  207. }
  208. static void create_property_nodes(struct dl_list *list,
  209. struct wpa_dbus_property_desc *properties)
  210. {
  211. struct wpa_dbus_property_desc *dsc;
  212. struct interfaces *iface;
  213. for (dsc = properties; dsc; dsc = dsc->next) {
  214. iface = get_interface(list, dsc->dbus_interface);
  215. if (!iface)
  216. continue;
  217. add_property(iface->xml, dsc);
  218. }
  219. }
  220. static void add_wpas_interfaces(struct wpabuf *xml,
  221. struct wpa_dbus_object_desc *obj_dsc)
  222. {
  223. struct dl_list ifaces;
  224. dl_list_init(&ifaces);
  225. extract_interfaces(&ifaces, obj_dsc);
  226. create_method_nodes(&ifaces, obj_dsc->methods);
  227. create_signal_nodes(&ifaces, obj_dsc->signals);
  228. create_property_nodes(&ifaces, obj_dsc->properties);
  229. add_interfaces(&ifaces, xml);
  230. }
  231. /**
  232. * wpa_dbus_introspect - Responds for Introspect calls on object
  233. * @message: Message with Introspect call
  234. * @obj_dsc: Object description on which Introspect was called
  235. * Returns: Message with introspection result XML string as only argument
  236. *
  237. * Iterates over all methods, signals and properties registered with
  238. * object and generates introspection data for the object as XML string.
  239. */
  240. DBusMessage * wpa_dbus_introspect(DBusMessage *message,
  241. struct wpa_dbus_object_desc *obj_dsc)
  242. {
  243. DBusMessage *reply;
  244. struct wpabuf *xml;
  245. const char *intro_str;
  246. xml = wpabuf_alloc(4000);
  247. if (xml == NULL)
  248. return NULL;
  249. wpabuf_put_str(xml, "<?xml version=\"1.0\"?>\n");
  250. wpabuf_put_str(xml, DBUS_INTROSPECT_1_0_XML_DOCTYPE_DECL_NODE);
  251. wpabuf_put_str(xml, "<node>");
  252. add_introspectable_interface(xml);
  253. add_properties_interface(xml);
  254. add_wpas_interfaces(xml, obj_dsc);
  255. add_child_nodes(xml, obj_dsc->connection,
  256. dbus_message_get_path(message));
  257. wpabuf_put_str(xml, "</node>\n");
  258. reply = dbus_message_new_method_return(message);
  259. if (reply == NULL) {
  260. wpabuf_free(xml);
  261. return NULL;
  262. }
  263. intro_str = wpabuf_head(xml);
  264. dbus_message_append_args(reply, DBUS_TYPE_STRING, &intro_str,
  265. DBUS_TYPE_INVALID);
  266. wpabuf_free(xml);
  267. return reply;
  268. }