dbus_common.c 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420
  1. /*
  2. * wpa_supplicant D-Bus control interface - common functionality
  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) 2009, 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 <dbus/dbus.h>
  18. #include "utils/common.h"
  19. #include "utils/eloop.h"
  20. #include "dbus_common.h"
  21. #include "dbus_common_i.h"
  22. #include "dbus_new.h"
  23. #include "dbus_old.h"
  24. #ifndef SIGPOLL
  25. #ifdef SIGIO
  26. /*
  27. * If we do not have SIGPOLL, try to use SIGIO instead. This is needed for
  28. * FreeBSD.
  29. */
  30. #define SIGPOLL SIGIO
  31. #endif
  32. #endif
  33. /**
  34. * dispatch_initial_dbus_messages - Dispatch initial dbus messages after
  35. * claiming bus name
  36. * @eloop_ctx: the DBusConnection to dispatch on
  37. * @timeout_ctx: unused
  38. *
  39. * If clients are quick to notice that service claimed its bus name,
  40. * there may have been messages that came in before initialization was
  41. * all finished. Dispatch those here.
  42. */
  43. static void dispatch_initial_dbus_messages(void *eloop_ctx, void *timeout_ctx)
  44. {
  45. DBusConnection *con = eloop_ctx;
  46. while (dbus_connection_get_dispatch_status(con) ==
  47. DBUS_DISPATCH_DATA_REMAINS)
  48. dbus_connection_dispatch(con);
  49. }
  50. static void process_watch(struct wpas_dbus_priv *priv,
  51. DBusWatch *watch, eloop_event_type type)
  52. {
  53. dbus_connection_ref(priv->con);
  54. priv->should_dispatch = 0;
  55. if (type == EVENT_TYPE_READ)
  56. dbus_watch_handle(watch, DBUS_WATCH_READABLE);
  57. else if (type == EVENT_TYPE_WRITE)
  58. dbus_watch_handle(watch, DBUS_WATCH_WRITABLE);
  59. else if (type == EVENT_TYPE_EXCEPTION)
  60. dbus_watch_handle(watch, DBUS_WATCH_ERROR);
  61. if (priv->should_dispatch) {
  62. while (dbus_connection_get_dispatch_status(priv->con) ==
  63. DBUS_DISPATCH_DATA_REMAINS)
  64. dbus_connection_dispatch(priv->con);
  65. priv->should_dispatch = 0;
  66. }
  67. dbus_connection_unref(priv->con);
  68. }
  69. static void process_watch_exception(int sock, void *eloop_ctx, void *sock_ctx)
  70. {
  71. process_watch(eloop_ctx, sock_ctx, EVENT_TYPE_EXCEPTION);
  72. }
  73. static void process_watch_read(int sock, void *eloop_ctx, void *sock_ctx)
  74. {
  75. process_watch(eloop_ctx, sock_ctx, EVENT_TYPE_READ);
  76. }
  77. static void process_watch_write(int sock, void *eloop_ctx, void *sock_ctx)
  78. {
  79. process_watch(eloop_ctx, sock_ctx, EVENT_TYPE_WRITE);
  80. }
  81. static void connection_setup_add_watch(struct wpas_dbus_priv *priv,
  82. DBusWatch *watch)
  83. {
  84. unsigned int flags;
  85. int fd;
  86. if (!dbus_watch_get_enabled(watch))
  87. return;
  88. flags = dbus_watch_get_flags(watch);
  89. fd = dbus_watch_get_unix_fd(watch);
  90. eloop_register_sock(fd, EVENT_TYPE_EXCEPTION, process_watch_exception,
  91. priv, watch);
  92. if (flags & DBUS_WATCH_READABLE) {
  93. eloop_register_sock(fd, EVENT_TYPE_READ, process_watch_read,
  94. priv, watch);
  95. }
  96. if (flags & DBUS_WATCH_WRITABLE) {
  97. eloop_register_sock(fd, EVENT_TYPE_WRITE, process_watch_write,
  98. priv, watch);
  99. }
  100. dbus_watch_set_data(watch, priv, NULL);
  101. }
  102. static void connection_setup_remove_watch(struct wpas_dbus_priv *priv,
  103. DBusWatch *watch)
  104. {
  105. unsigned int flags;
  106. int fd;
  107. flags = dbus_watch_get_flags(watch);
  108. fd = dbus_watch_get_unix_fd(watch);
  109. eloop_unregister_sock(fd, EVENT_TYPE_EXCEPTION);
  110. if (flags & DBUS_WATCH_READABLE)
  111. eloop_unregister_sock(fd, EVENT_TYPE_READ);
  112. if (flags & DBUS_WATCH_WRITABLE)
  113. eloop_unregister_sock(fd, EVENT_TYPE_WRITE);
  114. dbus_watch_set_data(watch, NULL, NULL);
  115. }
  116. static dbus_bool_t add_watch(DBusWatch *watch, void *data)
  117. {
  118. connection_setup_add_watch(data, watch);
  119. return TRUE;
  120. }
  121. static void remove_watch(DBusWatch *watch, void *data)
  122. {
  123. connection_setup_remove_watch(data, watch);
  124. }
  125. static void watch_toggled(DBusWatch *watch, void *data)
  126. {
  127. if (dbus_watch_get_enabled(watch))
  128. add_watch(watch, data);
  129. else
  130. remove_watch(watch, data);
  131. }
  132. static void process_timeout(void *eloop_ctx, void *sock_ctx)
  133. {
  134. DBusTimeout *timeout = sock_ctx;
  135. dbus_timeout_handle(timeout);
  136. }
  137. static void connection_setup_add_timeout(struct wpas_dbus_priv *priv,
  138. DBusTimeout *timeout)
  139. {
  140. if (!dbus_timeout_get_enabled(timeout))
  141. return;
  142. eloop_register_timeout(0, dbus_timeout_get_interval(timeout) * 1000,
  143. process_timeout, priv, timeout);
  144. dbus_timeout_set_data(timeout, priv, NULL);
  145. }
  146. static void connection_setup_remove_timeout(struct wpas_dbus_priv *priv,
  147. DBusTimeout *timeout)
  148. {
  149. eloop_cancel_timeout(process_timeout, priv, timeout);
  150. dbus_timeout_set_data(timeout, NULL, NULL);
  151. }
  152. static dbus_bool_t add_timeout(DBusTimeout *timeout, void *data)
  153. {
  154. if (!dbus_timeout_get_enabled(timeout))
  155. return TRUE;
  156. connection_setup_add_timeout(data, timeout);
  157. return TRUE;
  158. }
  159. static void remove_timeout(DBusTimeout *timeout, void *data)
  160. {
  161. connection_setup_remove_timeout(data, timeout);
  162. }
  163. static void timeout_toggled(DBusTimeout *timeout, void *data)
  164. {
  165. if (dbus_timeout_get_enabled(timeout))
  166. add_timeout(timeout, data);
  167. else
  168. remove_timeout(timeout, data);
  169. }
  170. static void process_wakeup_main(int sig, void *signal_ctx)
  171. {
  172. struct wpas_dbus_priv *priv = signal_ctx;
  173. if (sig != SIGPOLL || !priv->con)
  174. return;
  175. if (dbus_connection_get_dispatch_status(priv->con) !=
  176. DBUS_DISPATCH_DATA_REMAINS)
  177. return;
  178. /* Only dispatch once - we do not want to starve other events */
  179. dbus_connection_ref(priv->con);
  180. dbus_connection_dispatch(priv->con);
  181. dbus_connection_unref(priv->con);
  182. }
  183. /**
  184. * wakeup_main - Attempt to wake our mainloop up
  185. * @data: dbus control interface private data
  186. *
  187. * Try to wake up the main eloop so it will process
  188. * dbus events that may have happened.
  189. */
  190. static void wakeup_main(void *data)
  191. {
  192. struct wpas_dbus_priv *priv = data;
  193. /* Use SIGPOLL to break out of the eloop select() */
  194. raise(SIGPOLL);
  195. priv->should_dispatch = 1;
  196. }
  197. /**
  198. * connection_setup_wakeup_main - Tell dbus about our wakeup_main function
  199. * @priv: dbus control interface private data
  200. * Returns: 0 on success, -1 on failure
  201. *
  202. * Register our wakeup_main handler with dbus
  203. */
  204. static int connection_setup_wakeup_main(struct wpas_dbus_priv *priv)
  205. {
  206. if (eloop_register_signal(SIGPOLL, process_wakeup_main, priv))
  207. return -1;
  208. dbus_connection_set_wakeup_main_function(priv->con, wakeup_main,
  209. priv, NULL);
  210. return 0;
  211. }
  212. /**
  213. * integrate_with_eloop - Register our mainloop integration with dbus
  214. * @connection: connection to the system message bus
  215. * @priv: a dbus control interface data structure
  216. * Returns: 0 on success, -1 on failure
  217. *
  218. * We register our mainloop integration functions with dbus here.
  219. */
  220. static int integrate_with_eloop(struct wpas_dbus_priv *priv)
  221. {
  222. if (!dbus_connection_set_watch_functions(priv->con, add_watch,
  223. remove_watch, watch_toggled,
  224. priv, NULL)) {
  225. wpa_printf(MSG_ERROR, "dbus: Not enough memory to set up");
  226. return -1;
  227. }
  228. if (!dbus_connection_set_timeout_functions(priv->con, add_timeout,
  229. remove_timeout,
  230. timeout_toggled, priv,
  231. NULL)) {
  232. wpa_printf(MSG_ERROR, "dbus: Not enough memory to set up");
  233. return -1;
  234. }
  235. if (connection_setup_wakeup_main(priv) < 0) {
  236. wpa_printf(MSG_ERROR, "dbus: Could not setup main wakeup "
  237. "function");
  238. return -1;
  239. }
  240. return 0;
  241. }
  242. static int wpas_dbus_init_common(struct wpas_dbus_priv *priv)
  243. {
  244. DBusError error;
  245. /* Get a reference to the system bus */
  246. dbus_error_init(&error);
  247. priv->con = dbus_bus_get(DBUS_BUS_SYSTEM, &error);
  248. dbus_error_free(&error);
  249. if (!priv->con) {
  250. wpa_printf(MSG_ERROR, "dbus: Could not acquire the system "
  251. "bus: %s", strerror(errno));
  252. return -1;
  253. }
  254. return 0;
  255. }
  256. static int wpas_dbus_init_common_finish(struct wpas_dbus_priv *priv)
  257. {
  258. /* Tell dbus about our mainloop integration functions */
  259. integrate_with_eloop(priv);
  260. /*
  261. * Dispatch initial DBus messages that may have come in since the bus
  262. * name was claimed above. Happens when clients are quick to notice the
  263. * service.
  264. *
  265. * FIXME: is there a better solution to this problem?
  266. */
  267. eloop_register_timeout(0, 50, dispatch_initial_dbus_messages,
  268. priv->con, NULL);
  269. return 0;
  270. }
  271. static void wpas_dbus_deinit_common(struct wpas_dbus_priv *priv)
  272. {
  273. if (priv->con) {
  274. eloop_cancel_timeout(dispatch_initial_dbus_messages,
  275. priv->con, NULL);
  276. dbus_connection_set_watch_functions(priv->con, NULL, NULL,
  277. NULL, NULL, NULL);
  278. dbus_connection_set_timeout_functions(priv->con, NULL, NULL,
  279. NULL, NULL, NULL);
  280. dbus_connection_unref(priv->con);
  281. }
  282. os_free(priv);
  283. }
  284. struct wpas_dbus_priv * wpas_dbus_init(struct wpa_global *global)
  285. {
  286. struct wpas_dbus_priv *priv;
  287. priv = os_zalloc(sizeof(*priv));
  288. if (priv == NULL)
  289. return NULL;
  290. priv->global = global;
  291. if (wpas_dbus_init_common(priv) < 0) {
  292. wpas_dbus_deinit(priv);
  293. return NULL;
  294. }
  295. #ifdef CONFIG_CTRL_IFACE_DBUS_NEW
  296. if (wpas_dbus_ctrl_iface_init(priv) < 0) {
  297. wpas_dbus_deinit(priv);
  298. return NULL;
  299. }
  300. #endif /* CONFIG_CTRL_IFACE_DBUS_NEW */
  301. #ifdef CONFIG_CTRL_IFACE_DBUS
  302. if (wpa_supplicant_dbus_ctrl_iface_init(priv) < 0) {
  303. wpas_dbus_deinit(priv);
  304. return NULL;
  305. }
  306. #endif /* CONFIG_CTRL_IFACE_DBUS */
  307. if (wpas_dbus_init_common_finish(priv) < 0) {
  308. wpas_dbus_deinit(priv);
  309. return NULL;
  310. }
  311. return priv;
  312. }
  313. void wpas_dbus_deinit(struct wpas_dbus_priv *priv)
  314. {
  315. if (priv == NULL)
  316. return;
  317. #ifdef CONFIG_CTRL_IFACE_DBUS_NEW
  318. wpas_dbus_ctrl_iface_deinit(priv);
  319. #endif /* CONFIG_CTRL_IFACE_DBUS_NEW */
  320. #ifdef CONFIG_CTRL_IFACE_DBUS
  321. /* TODO: is any deinit needed? */
  322. #endif /* CONFIG_CTRL_IFACE_DBUS */
  323. wpas_dbus_deinit_common(priv);
  324. }