3 * WPA supplicant library with GLib integration
5 * Copyright (C) 2012-2013 Intel Corporation. All rights reserved.
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.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
29 #include <dbus/dbus.h>
36 static DBusConnection *connection;
38 static GSList *method_calls;
40 struct method_call_data {
42 DBusPendingCall *pending_call;
43 supplicant_dbus_result_function function;
47 static void method_call_free(void *pointer)
49 struct method_call_data *method_call = pointer;
50 method_calls = g_slist_remove(method_calls, method_call);
54 static int find_method_call_by_caller(gconstpointer a, gconstpointer b)
56 const struct method_call_data *method_call = a;
57 gconstpointer caller = b;
59 return method_call->caller != caller;
62 void supplicant_dbus_setup(DBusConnection *conn)
68 void supplicant_dbus_array_foreach(DBusMessageIter *iter,
69 supplicant_dbus_array_function function,
72 DBusMessageIter entry;
74 if (dbus_message_iter_get_arg_type(iter) != DBUS_TYPE_ARRAY)
77 dbus_message_iter_recurse(iter, &entry);
79 while (dbus_message_iter_get_arg_type(&entry) != DBUS_TYPE_INVALID) {
81 function(&entry, user_data);
83 dbus_message_iter_next(&entry);
87 void supplicant_dbus_property_foreach(DBusMessageIter *iter,
88 supplicant_dbus_property_function function,
93 if (dbus_message_iter_get_arg_type(iter) != DBUS_TYPE_ARRAY)
96 dbus_message_iter_recurse(iter, &dict);
98 while (dbus_message_iter_get_arg_type(&dict) == DBUS_TYPE_DICT_ENTRY) {
99 DBusMessageIter entry, value;
102 dbus_message_iter_recurse(&dict, &entry);
104 if (dbus_message_iter_get_arg_type(&entry) != DBUS_TYPE_STRING)
107 dbus_message_iter_get_basic(&entry, &key);
108 dbus_message_iter_next(&entry);
110 if (dbus_message_iter_get_arg_type(&entry) != DBUS_TYPE_VARIANT)
113 dbus_message_iter_recurse(&entry, &value);
116 if (strcmp(key, "Properties") == 0)
117 supplicant_dbus_property_foreach(&value,
118 function, user_data);
120 function(key, &value, user_data);
123 dbus_message_iter_next(&dict);
127 struct property_get_data {
128 supplicant_dbus_property_function function;
132 static void property_get_all_reply(DBusPendingCall *call, void *user_data)
134 struct property_get_data *data = user_data;
136 DBusMessageIter iter;
138 reply = dbus_pending_call_steal_reply(call);
140 if (dbus_message_get_type(reply) == DBUS_MESSAGE_TYPE_ERROR)
143 if (!dbus_message_iter_init(reply, &iter))
146 supplicant_dbus_property_foreach(&iter, data->function,
150 data->function(NULL, NULL, data->user_data);
153 dbus_message_unref(reply);
155 dbus_pending_call_unref(call);
158 int supplicant_dbus_property_get_all(const char *path, const char *interface,
159 supplicant_dbus_property_function function,
162 struct property_get_data *data;
163 DBusMessage *message;
164 DBusPendingCall *call;
169 if (!path || !interface)
172 data = dbus_malloc0(sizeof(*data));
176 message = dbus_message_new_method_call(SUPPLICANT_SERVICE, path,
177 DBUS_INTERFACE_PROPERTIES, "GetAll");
183 dbus_message_set_auto_start(message, FALSE);
185 dbus_message_append_args(message, DBUS_TYPE_STRING, &interface, NULL);
187 if (!dbus_connection_send_with_reply(connection, message,
189 dbus_message_unref(message);
195 dbus_message_unref(message);
200 data->function = function;
201 data->user_data = user_data;
203 dbus_pending_call_set_notify(call, property_get_all_reply,
206 dbus_message_unref(message);
211 static void property_get_reply(DBusPendingCall *call, void *user_data)
213 struct property_get_data *data = user_data;
215 DBusMessageIter iter;
217 reply = dbus_pending_call_steal_reply(call);
219 if (dbus_message_get_type(reply) == DBUS_MESSAGE_TYPE_ERROR)
222 if (!dbus_message_iter_init(reply, &iter))
225 if (dbus_message_iter_get_arg_type(&iter) == DBUS_TYPE_VARIANT) {
226 DBusMessageIter variant;
228 dbus_message_iter_recurse(&iter, &variant);
231 data->function(NULL, &variant, data->user_data);
234 dbus_message_unref(reply);
236 dbus_pending_call_unref(call);
239 int supplicant_dbus_property_get(const char *path, const char *interface,
241 supplicant_dbus_property_function function,
244 struct property_get_data *data;
245 DBusMessage *message;
246 DBusPendingCall *call;
251 if (!path || !interface || !method)
254 data = dbus_malloc0(sizeof(*data));
258 message = dbus_message_new_method_call(SUPPLICANT_SERVICE, path,
259 DBUS_INTERFACE_PROPERTIES, "Get");
266 dbus_message_set_auto_start(message, FALSE);
268 dbus_message_append_args(message, DBUS_TYPE_STRING, &interface,
269 DBUS_TYPE_STRING, &method, NULL);
271 if (!dbus_connection_send_with_reply(connection, message,
273 dbus_message_unref(message);
279 dbus_message_unref(message);
284 data->function = function;
285 data->user_data = user_data;
287 dbus_pending_call_set_notify(call, property_get_reply,
290 dbus_message_unref(message);
295 struct property_set_data {
296 supplicant_dbus_result_function function;
300 static void property_set_reply(DBusPendingCall *call, void *user_data)
302 struct property_set_data *data = user_data;
304 DBusMessageIter iter;
307 reply = dbus_pending_call_steal_reply(call);
309 if (dbus_message_get_type(reply) == DBUS_MESSAGE_TYPE_ERROR)
310 error = dbus_message_get_error_name(reply);
314 dbus_message_iter_init(reply, &iter);
317 data->function(error, &iter, data->user_data);
319 dbus_message_unref(reply);
321 dbus_pending_call_unref(call);
324 int supplicant_dbus_property_set(const char *path, const char *interface,
325 const char *key, const char *signature,
326 supplicant_dbus_setup_function setup,
327 supplicant_dbus_result_function function,
330 struct property_set_data *data;
331 DBusMessage *message;
332 DBusMessageIter iter, value;
333 DBusPendingCall *call;
338 if (!path || !interface)
341 if (!key || !signature || !setup)
344 data = dbus_malloc0(sizeof(*data));
348 message = dbus_message_new_method_call(SUPPLICANT_SERVICE, path,
349 DBUS_INTERFACE_PROPERTIES, "Set");
355 dbus_message_set_auto_start(message, FALSE);
357 dbus_message_iter_init_append(message, &iter);
358 dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &interface);
359 dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &key);
361 dbus_message_iter_open_container(&iter, DBUS_TYPE_VARIANT,
363 setup(&value, user_data);
364 dbus_message_iter_close_container(&iter, &value);
366 if (!dbus_connection_send_with_reply(connection, message,
368 dbus_message_unref(message);
374 dbus_message_unref(message);
379 data->function = function;
380 data->user_data = user_data;
382 dbus_pending_call_set_notify(call, property_set_reply,
385 dbus_message_unref(message);
390 void supplicant_dbus_method_call_cancel_all(gpointer caller)
392 while (method_calls) {
393 struct method_call_data *method_call;
394 GSList *elem = g_slist_find_custom(method_calls, caller,
395 find_method_call_by_caller);
399 method_call = elem->data;
400 method_calls = g_slist_delete_link(method_calls, elem);
402 dbus_pending_call_cancel(method_call->pending_call);
404 if (method_call->function)
405 method_call->function("net.connman.Error.OperationAborted",
406 NULL, method_call->user_data);
408 dbus_pending_call_unref(method_call->pending_call);
412 static void method_call_reply(DBusPendingCall *call, void *user_data)
414 struct method_call_data *method_call = user_data;
416 DBusMessageIter iter;
419 reply = dbus_pending_call_steal_reply(call);
421 if (dbus_message_get_type(reply) == DBUS_MESSAGE_TYPE_ERROR)
422 error = dbus_message_get_error_name(reply);
426 dbus_message_iter_init(reply, &iter);
428 if (method_call && method_call->function)
429 method_call->function(error, &iter, method_call->user_data);
431 dbus_message_unref(reply);
433 dbus_pending_call_unref(call);
436 int supplicant_dbus_method_call(const char *path,
437 const char *interface, const char *method,
438 supplicant_dbus_setup_function setup,
439 supplicant_dbus_result_function function,
443 struct method_call_data *method_call = NULL;
444 DBusMessage *message;
445 DBusMessageIter iter;
446 DBusPendingCall *call;
451 if (!path || !interface || !method)
454 method_call = g_try_new0(struct method_call_data, 1);
458 message = dbus_message_new_method_call(SUPPLICANT_SERVICE, path,
465 dbus_message_set_auto_start(message, FALSE);
467 dbus_message_iter_init_append(message, &iter);
469 setup(&iter, user_data);
471 if (!dbus_connection_send_with_reply(connection, message,
473 dbus_message_unref(message);
479 dbus_message_unref(message);
484 method_call->caller = caller;
485 method_call->pending_call = call;
486 method_call->function = function;
487 method_call->user_data = user_data;
488 method_calls = g_slist_prepend(method_calls, method_call);
490 dbus_pending_call_set_notify(call, method_call_reply, method_call,
493 dbus_message_unref(message);
498 void supplicant_dbus_property_append_basic(DBusMessageIter *iter,
499 const char *key, int type, void *val)
501 DBusMessageIter value;
502 const char *signature;
504 dbus_message_iter_append_basic(iter, DBUS_TYPE_STRING, &key);
507 case DBUS_TYPE_BOOLEAN:
508 signature = DBUS_TYPE_BOOLEAN_AS_STRING;
510 case DBUS_TYPE_STRING:
511 signature = DBUS_TYPE_STRING_AS_STRING;
514 signature = DBUS_TYPE_BYTE_AS_STRING;
516 case DBUS_TYPE_UINT16:
517 signature = DBUS_TYPE_UINT16_AS_STRING;
519 case DBUS_TYPE_INT16:
520 signature = DBUS_TYPE_INT16_AS_STRING;
522 case DBUS_TYPE_UINT32:
523 signature = DBUS_TYPE_UINT32_AS_STRING;
525 case DBUS_TYPE_INT32:
526 signature = DBUS_TYPE_INT32_AS_STRING;
528 case DBUS_TYPE_OBJECT_PATH:
529 signature = DBUS_TYPE_OBJECT_PATH_AS_STRING;
532 signature = DBUS_TYPE_VARIANT_AS_STRING;
536 dbus_message_iter_open_container(iter, DBUS_TYPE_VARIANT,
538 dbus_message_iter_append_basic(&value, type, val);
539 dbus_message_iter_close_container(iter, &value);
542 void supplicant_dbus_property_append_fixed_array(DBusMessageIter *iter,
543 const char *key, int type, void *val, int len)
545 DBusMessageIter value, array;
546 const char *variant_sig, *array_sig;
550 variant_sig = DBUS_TYPE_ARRAY_AS_STRING
551 DBUS_TYPE_BYTE_AS_STRING;
552 array_sig = DBUS_TYPE_BYTE_AS_STRING;
558 dbus_message_iter_append_basic(iter, DBUS_TYPE_STRING, &key);
560 dbus_message_iter_open_container(iter, DBUS_TYPE_VARIANT,
561 variant_sig, &value);
563 dbus_message_iter_open_container(&value, DBUS_TYPE_ARRAY,
565 dbus_message_iter_append_fixed_array(&array, type, val, len);
566 dbus_message_iter_close_container(&value, &array);
568 dbus_message_iter_close_container(iter, &value);
571 void supplicant_dbus_property_append_array(DBusMessageIter *iter,
572 const char *key, int type,
573 supplicant_dbus_array_function function,
576 DBusMessageIter value, array;
577 const char *variant_sig, *array_sig;
580 case DBUS_TYPE_STRING:
581 variant_sig = DBUS_TYPE_ARRAY_AS_STRING
582 DBUS_TYPE_ARRAY_AS_STRING
583 DBUS_TYPE_BYTE_AS_STRING;
584 array_sig = DBUS_TYPE_ARRAY_AS_STRING DBUS_TYPE_BYTE_AS_STRING;
590 dbus_message_iter_append_basic(iter, DBUS_TYPE_STRING, &key);
592 dbus_message_iter_open_container(iter, DBUS_TYPE_VARIANT,
593 variant_sig, &value);
595 dbus_message_iter_open_container(&value, DBUS_TYPE_ARRAY,
598 function(&array, user_data);
600 dbus_message_iter_close_container(&value, &array);
602 dbus_message_iter_close_container(iter, &value);