5 * Copyright (C) 2007-2008 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
35 static DBusConnection *connection;
37 static GNode *element_root = NULL;
38 static GSList *driver_list = NULL;
39 static gchar *device_filter = NULL;
42 enum connman_property_id id;
47 { CONNMAN_PROPERTY_ID_IPV4_METHOD,
48 DBUS_TYPE_STRING, "IPv4.Method", "dhcp" },
49 { CONNMAN_PROPERTY_ID_IPV4_ADDRESS,
50 DBUS_TYPE_STRING, "IPv4.Address" },
51 { CONNMAN_PROPERTY_ID_IPV4_NETMASK,
52 DBUS_TYPE_STRING, "IPv4.Netmask" },
53 { CONNMAN_PROPERTY_ID_IPV4_GATEWAY,
54 DBUS_TYPE_STRING, "IPv4.Gateway" },
55 { CONNMAN_PROPERTY_ID_IPV4_BROADCAST,
56 DBUS_TYPE_STRING, "IPv4.Broadcast" },
57 { CONNMAN_PROPERTY_ID_IPV4_NAMESERVER,
58 DBUS_TYPE_STRING, "IPv4.Nameserver" },
60 { CONNMAN_PROPERTY_ID_WIFI_SECURITY,
61 DBUS_TYPE_STRING, "WiFi.Security" },
62 { CONNMAN_PROPERTY_ID_WIFI_PASSPHRASE,
63 DBUS_TYPE_STRING, "WiFi.Passphrase" },
68 static int propid2type(enum connman_property_id id)
72 for (i = 0; propid_table[i].name; i++) {
73 if (propid_table[i].id == id)
74 return propid_table[i].type;
77 return DBUS_TYPE_INVALID;
80 static const char *propid2name(enum connman_property_id id)
84 for (i = 0; propid_table[i].name; i++) {
85 if (propid_table[i].id == id)
86 return propid_table[i].name;
92 static const char *type2string(enum connman_element_type type)
95 case CONNMAN_ELEMENT_TYPE_UNKNOWN:
97 case CONNMAN_ELEMENT_TYPE_ROOT:
99 case CONNMAN_ELEMENT_TYPE_PROFILE:
101 case CONNMAN_ELEMENT_TYPE_DEVICE:
103 case CONNMAN_ELEMENT_TYPE_NETWORK:
105 case CONNMAN_ELEMENT_TYPE_SERVICE:
107 case CONNMAN_ELEMENT_TYPE_IPV4:
109 case CONNMAN_ELEMENT_TYPE_IPV6:
111 case CONNMAN_ELEMENT_TYPE_DHCP:
113 case CONNMAN_ELEMENT_TYPE_BOOTP:
115 case CONNMAN_ELEMENT_TYPE_ZEROCONF:
117 case CONNMAN_ELEMENT_TYPE_CONNECTION:
124 static const char *subtype2string(enum connman_element_subtype type)
127 case CONNMAN_ELEMENT_SUBTYPE_UNKNOWN:
129 case CONNMAN_ELEMENT_SUBTYPE_FAKE:
131 case CONNMAN_ELEMENT_SUBTYPE_NETWORK:
133 case CONNMAN_ELEMENT_SUBTYPE_ETHERNET:
135 case CONNMAN_ELEMENT_SUBTYPE_WIFI:
137 case CONNMAN_ELEMENT_SUBTYPE_WIMAX:
139 case CONNMAN_ELEMENT_SUBTYPE_MODEM:
141 case CONNMAN_ELEMENT_SUBTYPE_BLUETOOTH:
148 static const char *subtype2description(enum connman_element_subtype type)
151 case CONNMAN_ELEMENT_SUBTYPE_UNKNOWN:
152 case CONNMAN_ELEMENT_SUBTYPE_FAKE:
153 case CONNMAN_ELEMENT_SUBTYPE_NETWORK:
155 case CONNMAN_ELEMENT_SUBTYPE_ETHERNET:
157 case CONNMAN_ELEMENT_SUBTYPE_WIFI:
159 case CONNMAN_ELEMENT_SUBTYPE_WIMAX:
161 case CONNMAN_ELEMENT_SUBTYPE_MODEM:
163 case CONNMAN_ELEMENT_SUBTYPE_BLUETOOTH:
170 const char *__connman_element_policy2string(enum connman_element_policy policy)
173 case CONNMAN_ELEMENT_POLICY_UNKNOWN:
175 case CONNMAN_ELEMENT_POLICY_IGNORE:
177 case CONNMAN_ELEMENT_POLICY_AUTO:
179 case CONNMAN_ELEMENT_POLICY_ASK:
186 enum connman_element_policy __connman_element_string2policy(const char *policy)
188 if (strcasecmp(policy, "ignore") == 0)
189 return CONNMAN_ELEMENT_POLICY_IGNORE;
190 else if (strcasecmp(policy, "auto") == 0)
191 return CONNMAN_ELEMENT_POLICY_AUTO;
192 else if (strcasecmp(policy, "ask") == 0)
193 return CONNMAN_ELEMENT_POLICY_ASK;
195 return CONNMAN_ELEMENT_POLICY_UNKNOWN;
198 const char *__connman_ipv4_method2string(enum connman_ipv4_method method)
201 case CONNMAN_IPV4_METHOD_UNKNOWN:
203 case CONNMAN_IPV4_METHOD_OFF:
205 case CONNMAN_IPV4_METHOD_STATIC:
207 case CONNMAN_IPV4_METHOD_DHCP:
214 enum connman_ipv4_method __connman_ipv4_string2method(const char *method)
216 if (strcasecmp(method, "off") == 0)
217 return CONNMAN_IPV4_METHOD_OFF;
218 else if (strcasecmp(method, "static") == 0)
219 return CONNMAN_IPV4_METHOD_STATIC;
220 else if (strcasecmp(method, "dhcp") == 0)
221 return CONNMAN_IPV4_METHOD_DHCP;
223 return CONNMAN_IPV4_METHOD_UNKNOWN;
226 static void append_property(DBusMessageIter *dict,
227 struct connman_property *property)
229 if (property->value == NULL)
232 switch (property->type) {
233 case DBUS_TYPE_ARRAY:
234 connman_dbus_dict_append_array(dict, property->name,
235 property->subtype, &property->value, property->size);
237 case DBUS_TYPE_STRING:
238 connman_dbus_dict_append_variant(dict, property->name,
239 property->type, &property->value);
242 connman_dbus_dict_append_variant(dict, property->name,
243 property->type, property->value);
248 static void add_common_properties(struct connman_element *element,
249 DBusMessageIter *dict)
251 const char *address = NULL, *netmask = NULL, *gateway = NULL;
254 connman_element_get_value(element,
255 CONNMAN_PROPERTY_ID_IPV4_ADDRESS, &address);
256 connman_element_get_value(element,
257 CONNMAN_PROPERTY_ID_IPV4_NETMASK, &netmask);
258 connman_element_get_value(element,
259 CONNMAN_PROPERTY_ID_IPV4_GATEWAY, &gateway);
261 if (element->priority > 0)
262 connman_dbus_dict_append_variant(dict, "Priority",
263 DBUS_TYPE_UINT16, &element->priority);
266 connman_dbus_dict_append_variant(dict, "IPv4.Address",
267 DBUS_TYPE_STRING, &address);
269 connman_dbus_dict_append_variant(dict, "IPv4.Netmask",
270 DBUS_TYPE_STRING, &netmask);
272 connman_dbus_dict_append_variant(dict, "IPv4.Gateway",
273 DBUS_TYPE_STRING, &gateway);
275 if (element->wifi.security != NULL) {
276 const char *passphrase = "";
278 connman_dbus_dict_append_variant(dict, "WiFi.Security",
279 DBUS_TYPE_STRING, &element->wifi.security);
281 if (element->wifi.passphrase != NULL)
282 passphrase = element->wifi.passphrase;
284 connman_dbus_dict_append_variant(dict, "WiFi.Passphrase",
285 DBUS_TYPE_STRING, &passphrase);
288 __connman_element_lock(element);
290 for (list = element->properties; list; list = list->next) {
291 struct connman_property *property = list->data;
293 append_property(dict, property);
296 __connman_element_unlock(element);
299 static void set_common_property(struct connman_element *element,
300 const char *name, DBusMessageIter *value)
304 if (g_str_equal(name, "Priority") == TRUE) {
305 dbus_message_iter_get_basic(value, &element->priority);
309 __connman_element_lock(element);
311 for (list = element->properties; list; list = list->next) {
312 struct connman_property *property = list->data;
315 if (g_str_equal(property->name, name) == FALSE)
318 if (property->flags & CONNMAN_PROPERTY_FLAG_STATIC)
321 property->flags &= ~CONNMAN_PROPERTY_FLAG_REFERENCE;
323 if (property->type == DBUS_TYPE_STRING) {
324 dbus_message_iter_get_basic(value, &str);
325 g_free(property->value);
326 property->value = g_strdup(str);
328 property->value = NULL;
331 __connman_element_unlock(element);
334 static void emit_enabled_signal(DBusConnection *conn,
335 struct connman_element *element)
338 DBusMessageIter entry, value;
339 const char *iface, *key;
341 DBG("conn %p", conn);
346 switch (element->type) {
347 case CONNMAN_ELEMENT_TYPE_DEVICE:
348 iface = CONNMAN_DEVICE_INTERFACE;
351 case CONNMAN_ELEMENT_TYPE_NETWORK:
352 iface = CONNMAN_NETWORK_INTERFACE;
355 case CONNMAN_ELEMENT_TYPE_CONNECTION:
356 iface = CONNMAN_CONNECTION_INTERFACE;
363 signal = dbus_message_new_signal(element->path,
364 iface, "PropertyChanged");
368 dbus_message_iter_init_append(signal, &entry);
370 dbus_message_iter_append_basic(&entry, DBUS_TYPE_STRING, &key);
372 dbus_message_iter_open_container(&entry, DBUS_TYPE_VARIANT,
373 DBUS_TYPE_BOOLEAN_AS_STRING, &value);
374 dbus_message_iter_append_basic(&value, DBUS_TYPE_BOOLEAN,
376 dbus_message_iter_close_container(&entry, &value);
378 g_dbus_send_message(conn, signal);
381 static DBusMessage *do_update(DBusConnection *conn,
382 DBusMessage *msg, void *data)
384 struct connman_element *element = data;
386 DBG("conn %p", conn);
388 if (element->enabled == FALSE)
389 return __connman_error_failed(msg);
391 if (element->driver && element->driver->update) {
392 DBG("Calling update callback");
393 if (element->driver->update(element) < 0)
394 return __connman_error_failed(msg);
398 return g_dbus_create_reply(msg, DBUS_TYPE_INVALID);
401 static DBusMessage *do_enable(DBusConnection *conn,
402 DBusMessage *msg, void *data)
404 struct connman_element *element = data;
406 DBG("conn %p", conn);
408 if (element->enabled == TRUE)
409 return __connman_error_failed(msg);
411 if (element->driver && element->driver->enable) {
412 DBG("Calling enable callback");
413 if (element->driver->enable(element) < 0)
414 return __connman_error_failed(msg);
417 element->enabled = TRUE;
419 emit_enabled_signal(connection, element);
421 return g_dbus_create_reply(msg, DBUS_TYPE_INVALID);
424 static DBusMessage *do_disable(DBusConnection *conn,
425 DBusMessage *msg, void *data)
427 struct connman_element *element = data;
429 DBG("conn %p", conn);
431 if (element->enabled == FALSE)
432 return __connman_error_failed(msg);
434 if (element->driver && element->driver->disable) {
435 DBG("Calling disable callback");
436 if (element->driver->disable(element) < 0)
437 return __connman_error_failed(msg);
440 element->enabled = FALSE;
442 emit_enabled_signal(connection, element);
444 return g_dbus_create_reply(msg, DBUS_TYPE_INVALID);
447 static void append_networks(struct connman_element *element,
448 DBusMessageIter *entry)
450 DBusMessageIter value, iter;
451 const char *key = "Networks";
453 dbus_message_iter_append_basic(entry, DBUS_TYPE_STRING, &key);
455 dbus_message_iter_open_container(entry, DBUS_TYPE_VARIANT,
456 DBUS_TYPE_ARRAY_AS_STRING DBUS_TYPE_OBJECT_PATH_AS_STRING,
459 dbus_message_iter_open_container(&value, DBUS_TYPE_ARRAY,
460 DBUS_TYPE_OBJECT_PATH_AS_STRING, &iter);
462 __connman_element_list(element, CONNMAN_ELEMENT_TYPE_NETWORK, &iter);
464 dbus_message_iter_close_container(&value, &iter);
466 dbus_message_iter_close_container(entry, &value);
469 static DBusMessage *device_get_properties(DBusConnection *conn,
470 DBusMessage *msg, void *data)
472 struct connman_element *element = data;
474 DBusMessageIter array, dict, entry;
477 DBG("conn %p", conn);
479 reply = dbus_message_new_method_return(msg);
483 dbus_message_iter_init_append(reply, &array);
485 dbus_message_iter_open_container(&array, DBUS_TYPE_ARRAY,
486 DBUS_DICT_ENTRY_BEGIN_CHAR_AS_STRING
487 DBUS_TYPE_STRING_AS_STRING DBUS_TYPE_VARIANT_AS_STRING
488 DBUS_DICT_ENTRY_END_CHAR_AS_STRING, &dict);
490 str = subtype2description(element->subtype);
491 if (str != NULL && element->devname != NULL) {
492 char *name = g_strdup_printf("%s (%s)", str, element->devname);
494 connman_dbus_dict_append_variant(&dict, "Name",
495 DBUS_TYPE_STRING, &name);
499 str = subtype2string(element->subtype);
501 connman_dbus_dict_append_variant(&dict, "Type",
502 DBUS_TYPE_STRING, &str);
504 str = __connman_element_policy2string(element->policy);
506 connman_dbus_dict_append_variant(&dict, "Policy",
507 DBUS_TYPE_STRING, &str);
509 connman_dbus_dict_append_variant(&dict, "Powered",
510 DBUS_TYPE_BOOLEAN, &element->enabled);
512 if (element->subtype == CONNMAN_ELEMENT_SUBTYPE_WIFI ||
513 element->subtype == CONNMAN_ELEMENT_SUBTYPE_WIMAX) {
514 dbus_message_iter_open_container(&dict, DBUS_TYPE_DICT_ENTRY,
516 append_networks(element, &entry);
517 dbus_message_iter_close_container(&dict, &entry);
520 add_common_properties(element, &dict);
522 dbus_message_iter_close_container(&array, &dict);
527 static DBusMessage *device_set_property(DBusConnection *conn,
528 DBusMessage *msg, void *data)
530 struct connman_element *element = data;
531 DBusMessageIter iter, value;
534 DBG("conn %p", conn);
536 if (dbus_message_iter_init(msg, &iter) == FALSE)
537 return __connman_error_invalid_arguments(msg);
539 dbus_message_iter_get_basic(&iter, &name);
540 dbus_message_iter_next(&iter);
541 dbus_message_iter_recurse(&iter, &value);
543 if (__connman_security_check_privileges(msg) < 0)
544 return __connman_error_permission_denied(msg);
546 if (g_str_equal(name, "Powered") == TRUE) {
549 dbus_message_iter_get_basic(&value, &powered);
552 do_enable(conn, msg, element);
554 do_disable(conn, msg, element);
556 set_common_property(element, name, &value);
558 __connman_element_store(element);
560 return g_dbus_create_reply(msg, DBUS_TYPE_INVALID);
563 static int parse_network_dict(DBusMessageIter *iter, const char **ssid,
564 const char **security, const char **passphrase)
566 while (dbus_message_iter_get_arg_type(iter) == DBUS_TYPE_DICT_ENTRY) {
567 DBusMessageIter entry, value;
570 dbus_message_iter_recurse(iter, &entry);
571 dbus_message_iter_get_basic(&entry, &key);
573 dbus_message_iter_next(&entry);
574 dbus_message_iter_recurse(&entry, &value);
576 switch (dbus_message_iter_get_arg_type(&value)) {
577 case DBUS_TYPE_STRING:
578 if (g_str_equal(key, "WiFi.SSID") == TRUE)
579 dbus_message_iter_get_basic(&value, ssid);
580 else if (g_str_equal(key, "WiFi.Security") == TRUE)
581 dbus_message_iter_get_basic(&value, security);
582 else if (g_str_equal(key, "WiFi.Passphrase") == TRUE)
583 dbus_message_iter_get_basic(&value, passphrase);
587 dbus_message_iter_next(iter);
593 static DBusMessage *device_create_network(DBusConnection *conn,
594 DBusMessage *msg, void *data)
596 struct connman_element *element = data;
597 struct connman_element *network;
598 DBusMessageIter iter, array;
599 const char *ssid = NULL, *security = NULL, *passphrase = NULL;
601 DBG("conn %p", conn);
603 if (dbus_message_iter_init(msg, &iter) == FALSE)
604 return __connman_error_invalid_arguments(msg);
606 dbus_message_iter_recurse(&iter, &array);
607 parse_network_dict(&array, &ssid, &security, &passphrase);
609 return __connman_error_invalid_arguments(msg);
611 DBG("ssid %s security %s passphrase %s", ssid, security, passphrase);
613 network = connman_element_create(ssid);
615 network->type = CONNMAN_ELEMENT_TYPE_NETWORK;
616 network->index = element->index;
618 network->remember = TRUE;
620 connman_element_add_static_property(network, "Name",
621 DBUS_TYPE_STRING, &ssid);
623 connman_element_add_static_array_property(element, "WiFi.SSID",
624 DBUS_TYPE_BYTE, &ssid, strlen(ssid));
626 network->wifi.security = g_strdup(security);
627 network->wifi.passphrase = g_strdup(passphrase);
629 connman_element_register(network, element);
631 return g_dbus_create_reply(msg, DBUS_TYPE_OBJECT_PATH, &network->path,
635 static DBusMessage *device_remove_network(DBusConnection *conn,
636 DBusMessage *msg, void *data)
638 return g_dbus_create_reply(msg, DBUS_TYPE_INVALID);
641 static DBusMessage *network_get_properties(DBusConnection *conn,
642 DBusMessage *msg, void *data)
644 struct connman_element *element = data;
646 DBusMessageIter array, dict;
649 DBG("conn %p", conn);
651 reply = dbus_message_new_method_return(msg);
655 dbus_message_iter_init_append(reply, &array);
657 dbus_message_iter_open_container(&array, DBUS_TYPE_ARRAY,
658 DBUS_DICT_ENTRY_BEGIN_CHAR_AS_STRING
659 DBUS_TYPE_STRING_AS_STRING DBUS_TYPE_VARIANT_AS_STRING
660 DBUS_DICT_ENTRY_END_CHAR_AS_STRING, &dict);
662 str = __connman_element_policy2string(element->policy);
664 connman_dbus_dict_append_variant(&dict, "Policy",
665 DBUS_TYPE_STRING, &str);
667 connman_dbus_dict_append_variant(&dict, "Available",
668 DBUS_TYPE_BOOLEAN, &element->available);
670 connman_dbus_dict_append_variant(&dict, "Connected",
671 DBUS_TYPE_BOOLEAN, &element->enabled);
673 connman_dbus_dict_append_variant(&dict, "Remember",
674 DBUS_TYPE_BOOLEAN, &element->remember);
676 add_common_properties(element, &dict);
678 dbus_message_iter_close_container(&array, &dict);
683 static DBusMessage *network_set_property(DBusConnection *conn,
684 DBusMessage *msg, void *data)
686 struct connman_element *element = data;
687 DBusMessageIter iter;
688 DBusMessageIter value;
691 DBG("conn %p", conn);
693 if (dbus_message_iter_init(msg, &iter) == FALSE)
694 return __connman_error_invalid_arguments(msg);
696 dbus_message_iter_get_basic(&iter, &name);
697 dbus_message_iter_next(&iter);
698 dbus_message_iter_recurse(&iter, &value);
700 if (__connman_security_check_privileges(msg) < 0)
701 return __connman_error_permission_denied(msg);
703 if (g_str_equal(name, "Remember") == TRUE) {
704 dbus_message_iter_get_basic(&value, &element->remember);
705 } else if (g_str_equal(name, "WiFi.Passphrase") == TRUE) {
708 dbus_message_iter_get_basic(&value, &str);
709 g_free(element->wifi.passphrase);
710 element->wifi.passphrase = g_strdup(str);
712 set_common_property(element, name, &value);
714 __connman_element_store(element);
716 return g_dbus_create_reply(msg, DBUS_TYPE_INVALID);
719 static DBusMessage *connection_get_properties(DBusConnection *conn,
720 DBusMessage *msg, void *data)
722 struct connman_element *element = data;
724 DBusMessageIter array, dict;
727 DBG("conn %p", conn);
729 reply = dbus_message_new_method_return(msg);
733 dbus_message_iter_init_append(reply, &array);
735 dbus_message_iter_open_container(&array, DBUS_TYPE_ARRAY,
736 DBUS_DICT_ENTRY_BEGIN_CHAR_AS_STRING
737 DBUS_TYPE_STRING_AS_STRING DBUS_TYPE_VARIANT_AS_STRING
738 DBUS_DICT_ENTRY_END_CHAR_AS_STRING, &dict);
740 str = subtype2string(element->subtype);
742 connman_dbus_dict_append_variant(&dict, "Type",
743 DBUS_TYPE_STRING, &str);
745 if (element->subtype == CONNMAN_ELEMENT_SUBTYPE_WIFI ||
746 element->subtype == CONNMAN_ELEMENT_SUBTYPE_WIMAX)
747 connman_dbus_dict_append_variant(&dict, "Strength",
748 DBUS_TYPE_BYTE, &element->strength);
750 connman_dbus_dict_append_variant(&dict, "Default",
751 DBUS_TYPE_BOOLEAN, &element->enabled);
753 add_common_properties(element, &dict);
755 dbus_message_iter_close_container(&array, &dict);
760 static DBusMessage *connection_set_property(DBusConnection *conn,
761 DBusMessage *msg, void *data)
763 struct connman_element *element = data;
764 DBusMessageIter iter, value;
767 DBG("conn %p", conn);
769 if (dbus_message_iter_init(msg, &iter) == FALSE)
770 return __connman_error_invalid_arguments(msg);
772 dbus_message_iter_get_basic(&iter, &name);
773 dbus_message_iter_next(&iter);
774 dbus_message_iter_recurse(&iter, &value);
776 if (__connman_security_check_privileges(msg) < 0)
777 return __connman_error_permission_denied(msg);
779 if (g_str_equal(name, "Default") == TRUE) {
782 dbus_message_iter_get_basic(&value, &enabled);
785 return do_enable(conn, msg, element);
787 return do_disable(conn, msg, element);
790 return g_dbus_create_reply(msg, DBUS_TYPE_INVALID);
793 static GDBusMethodTable device_methods[] = {
794 { "GetProperties", "", "a{sv}", device_get_properties },
795 { "SetProperty", "sv", "", device_set_property },
796 { "CreateNetwork", "a{sv}", "o", device_create_network },
797 { "RemoveNetwork", "o", "", device_remove_network },
798 { "ProposeScan", "", "", do_update },
802 static GDBusMethodTable network_methods[] = {
803 { "GetProperties", "", "a{sv}", network_get_properties },
804 { "SetProperty", "sv", "", network_set_property },
805 { "Connect", "", "", do_enable },
806 { "Disconnect", "", "", do_disable },
810 static GDBusMethodTable connection_methods[] = {
811 { "GetProperties", "", "a{sv}", connection_get_properties },
812 { "SetProperty", "sv", "", connection_set_property },
816 static GDBusSignalTable element_signals[] = {
817 { "PropertyChanged", "sv" },
821 struct append_filter {
822 enum connman_element_type type;
823 DBusMessageIter *iter;
826 static gboolean append_path(GNode *node, gpointer user_data)
828 struct connman_element *element = node->data;
829 struct append_filter *filter = user_data;
831 DBG("element %p name %s", element, element->name);
833 if (element->type == CONNMAN_ELEMENT_TYPE_ROOT)
836 if (filter->type != CONNMAN_ELEMENT_TYPE_UNKNOWN &&
837 filter->type != element->type)
840 if (filter->type == CONNMAN_ELEMENT_TYPE_DEVICE &&
841 element->subtype == CONNMAN_ELEMENT_SUBTYPE_NETWORK)
844 dbus_message_iter_append_basic(filter->iter,
845 DBUS_TYPE_OBJECT_PATH, &element->path);
850 void __connman_element_list(struct connman_element *element,
851 enum connman_element_type type,
852 DBusMessageIter *iter)
854 struct append_filter filter = { type, iter };
859 if (element != NULL) {
860 node = g_node_find(element_root, G_PRE_ORDER,
861 G_TRAVERSE_ALL, element);
867 g_node_traverse(node, G_PRE_ORDER, G_TRAVERSE_ALL, -1,
868 append_path, &filter);
872 enum connman_element_type type;
876 static gboolean count_element(GNode *node, gpointer user_data)
878 struct connman_element *element = node->data;
879 struct count_data *data = user_data;
881 DBG("element %p name %s", element, element->name);
883 if (element->type == CONNMAN_ELEMENT_TYPE_ROOT)
886 if (data->type != CONNMAN_ELEMENT_TYPE_UNKNOWN &&
887 data->type != element->type)
895 int __connman_element_count(struct connman_element *element,
896 enum connman_element_type type)
898 struct count_data data = { type, 0 };
903 if (element != NULL) {
904 node = g_node_find(element_root, G_PRE_ORDER,
905 G_TRAVERSE_ALL, element);
911 g_node_traverse(node, G_PRE_ORDER, G_TRAVERSE_ALL, -1,
912 count_element, &data);
917 static gint compare_priority(gconstpointer a, gconstpointer b)
919 const struct connman_driver *driver1 = a;
920 const struct connman_driver *driver2 = b;
922 return driver2->priority - driver1->priority;
925 static gboolean match_driver(struct connman_element *element,
926 struct connman_driver *driver)
928 if (element->type == CONNMAN_ELEMENT_TYPE_ROOT)
931 if (element->type != driver->type &&
932 driver->type != CONNMAN_ELEMENT_TYPE_UNKNOWN)
935 if (element->subtype == driver->subtype ||
936 driver->subtype == CONNMAN_ELEMENT_SUBTYPE_UNKNOWN)
942 static void enable_element(struct connman_element *element)
944 if (element->type != CONNMAN_ELEMENT_TYPE_DEVICE)
947 if (element->policy != CONNMAN_ELEMENT_POLICY_AUTO)
950 if (element->driver && element->driver->enable) {
951 if (element->driver->enable(element) == 0) {
952 element->enabled = TRUE;
953 emit_enabled_signal(connection, element);
958 static gboolean probe_driver(GNode *node, gpointer data)
960 struct connman_element *element = node->data;
961 struct connman_driver *driver = data;
963 DBG("element %p name %s", element, element->name);
965 if (!element->driver && match_driver(element, driver) == TRUE) {
966 if (driver->probe(element) < 0)
969 __connman_element_lock(element);
970 element->driver = driver;
971 __connman_element_unlock(element);
973 enable_element(element);
979 void __connman_driver_rescan(struct connman_driver *driver)
981 DBG("driver %p name %s", driver, driver->name);
986 if (element_root != NULL)
987 g_node_traverse(element_root, G_PRE_ORDER,
988 G_TRAVERSE_ALL, -1, probe_driver, driver);
992 * connman_driver_register:
993 * @driver: driver definition
995 * Register a new driver
997 * Returns: %0 on success
999 int connman_driver_register(struct connman_driver *driver)
1001 DBG("driver %p name %s", driver, driver->name);
1003 if (driver->type == CONNMAN_ELEMENT_TYPE_ROOT)
1009 driver_list = g_slist_insert_sorted(driver_list, driver,
1012 if (element_root != NULL)
1013 g_node_traverse(element_root, G_PRE_ORDER,
1014 G_TRAVERSE_ALL, -1, probe_driver, driver);
1019 static void disable_element(struct connman_element *element)
1021 if (element->policy != CONNMAN_ELEMENT_POLICY_AUTO)
1024 if (element->enabled == FALSE)
1027 if (element->driver && element->driver->disable) {
1028 if (element->driver->disable(element) == 0) {
1029 element->enabled = FALSE;
1030 emit_enabled_signal(connection, element);
1035 static gboolean remove_driver(GNode *node, gpointer data)
1037 struct connman_element *element = node->data;
1038 struct connman_driver *driver = data;
1040 DBG("element %p name %s", element, element->name);
1042 if (element->driver == driver) {
1043 disable_element(element);
1046 driver->remove(element);
1048 __connman_element_lock(element);
1049 element->driver = NULL;
1050 __connman_element_unlock(element);
1057 * connman_driver_unregister:
1058 * @driver: driver definition
1060 * Remove a previously registered driver
1062 void connman_driver_unregister(struct connman_driver *driver)
1064 DBG("driver %p name %s", driver, driver->name);
1066 driver_list = g_slist_remove(driver_list, driver);
1068 if (element_root != NULL)
1069 g_node_traverse(element_root, G_POST_ORDER,
1070 G_TRAVERSE_ALL, -1, remove_driver, driver);
1074 * connman_element_create:
1075 * @name: element name
1077 * Allocate a new element and assign the given #name to it. If the name
1078 * is #NULL, it will be later on created based on the element type.
1080 * Returns: a newly-allocated #connman_element structure
1082 struct connman_element *connman_element_create(const char *name)
1084 struct connman_element *element;
1086 element = g_try_new0(struct connman_element, 1);
1087 if (element == NULL)
1090 DBG("element %p", element);
1092 element->refcount = 1;
1094 element->name = g_strdup(name);
1095 element->type = CONNMAN_ELEMENT_TYPE_UNKNOWN;
1096 element->subtype = CONNMAN_ELEMENT_SUBTYPE_UNKNOWN;
1097 element->state = CONNMAN_ELEMENT_STATE_CLOSED;
1098 element->policy = CONNMAN_ELEMENT_POLICY_AUTO;
1099 element->index = -1;
1100 element->enabled = FALSE;
1105 struct connman_element *connman_element_ref(struct connman_element *element)
1107 DBG("element %p name %s refcount %d", element, element->name,
1108 g_atomic_int_get(&element->refcount) + 1);
1110 g_atomic_int_inc(&element->refcount);
1115 static void free_properties(struct connman_element *element)
1119 DBG("element %p name %s", element, element->name);
1121 __connman_element_lock(element);
1123 for (list = element->properties; list; list = list->next) {
1124 struct connman_property *property = list->data;
1126 if (!(property->flags & CONNMAN_PROPERTY_FLAG_REFERENCE))
1127 g_free(property->value);
1129 g_free(property->name);
1133 g_slist_free(element->properties);
1135 element->properties = NULL;
1137 __connman_element_unlock(element);
1140 void connman_element_unref(struct connman_element *element)
1142 DBG("element %p name %s refcount %d", element, element->name,
1143 g_atomic_int_get(&element->refcount) - 1);
1145 if (g_atomic_int_dec_and_test(&element->refcount) == TRUE) {
1146 free_properties(element);
1147 g_free(element->ipv4.address);
1148 g_free(element->ipv4.netmask);
1149 g_free(element->ipv4.gateway);
1150 g_free(element->ipv4.network);
1151 g_free(element->ipv4.broadcast);
1152 g_free(element->ipv4.nameserver);
1153 g_free(element->devname);
1154 g_free(element->path);
1155 g_free(element->name);
1160 int connman_element_add_static_property(struct connman_element *element,
1161 const char *name, int type, const void *value)
1163 struct connman_property *property;
1165 DBG("element %p name %s", element, element->name);
1167 if (type != DBUS_TYPE_STRING && type != DBUS_TYPE_BYTE)
1170 property = g_try_new0(struct connman_property, 1);
1171 if (property == NULL)
1174 property->flags = CONNMAN_PROPERTY_FLAG_STATIC;
1175 property->id = CONNMAN_PROPERTY_ID_INVALID;
1176 property->name = g_strdup(name);
1177 property->type = type;
1179 DBG("name %s type %d value %p", name, type, value);
1182 case DBUS_TYPE_STRING:
1183 property->value = g_strdup(*((const char **) value));
1185 case DBUS_TYPE_BYTE:
1186 property->value = g_try_malloc(1);
1187 if (property->value != NULL)
1188 memcpy(property->value, value, 1);
1192 __connman_element_lock(element);
1193 element->properties = g_slist_append(element->properties, property);
1194 __connman_element_unlock(element);
1199 static void emit_property_changed(DBusConnection *conn,
1200 struct connman_element *element,
1201 const char *name, int type, const void *data)
1203 DBusMessage *signal;
1204 DBusMessageIter entry, value;
1205 const char *iface, *sig;
1207 DBG("conn %p", conn);
1209 switch (element->type) {
1210 case CONNMAN_ELEMENT_TYPE_DEVICE:
1211 iface = CONNMAN_DEVICE_INTERFACE;
1213 case CONNMAN_ELEMENT_TYPE_NETWORK:
1214 iface = CONNMAN_NETWORK_INTERFACE;
1216 case CONNMAN_ELEMENT_TYPE_CONNECTION:
1217 iface = CONNMAN_CONNECTION_INTERFACE;
1223 signal = dbus_message_new_signal(element->path,
1224 iface, "PropertyChanged");
1228 dbus_message_iter_init_append(signal, &entry);
1230 dbus_message_iter_append_basic(&entry, DBUS_TYPE_STRING, &name);
1233 case DBUS_TYPE_STRING:
1234 sig = DBUS_TYPE_STRING_AS_STRING;
1236 case DBUS_TYPE_BYTE:
1237 sig = DBUS_TYPE_BYTE_AS_STRING;
1240 sig = DBUS_TYPE_VARIANT_AS_STRING;
1244 dbus_message_iter_open_container(&entry, DBUS_TYPE_VARIANT,
1246 dbus_message_iter_append_basic(&value, type, data);
1247 dbus_message_iter_close_container(&entry, &value);
1249 g_dbus_send_message(conn, signal);
1252 int connman_element_set_static_property(struct connman_element *element,
1253 const char *name, int type, const void *value)
1257 DBG("element %p name %s", element, element->name);
1259 if (type != DBUS_TYPE_STRING && type != DBUS_TYPE_BYTE)
1262 __connman_element_lock(element);
1264 for (list = element->properties; list; list = list->next) {
1265 struct connman_property *property = list->data;
1267 if (g_str_equal(property->name, name) == FALSE)
1270 if (!(property->flags & CONNMAN_PROPERTY_FLAG_STATIC))
1273 property->type = type;
1274 g_free(property->value);
1277 case DBUS_TYPE_STRING:
1278 property->value = g_strdup(*((const char **) value));
1280 case DBUS_TYPE_BYTE:
1281 property->value = g_try_malloc(1);
1282 if (property->value != NULL)
1283 memcpy(property->value, value, 1);
1288 __connman_element_unlock(element);
1290 emit_property_changed(connection, element, name, type, value);
1295 int connman_element_add_static_array_property(struct connman_element *element,
1296 const char *name, int type, const void *value, int len)
1298 struct connman_property *property;
1300 DBG("element %p name %s", element, element->name);
1302 if (type != DBUS_TYPE_BYTE)
1305 property = g_try_new0(struct connman_property, 1);
1306 if (property == NULL)
1309 property->flags = CONNMAN_PROPERTY_FLAG_STATIC;
1310 property->id = CONNMAN_PROPERTY_ID_INVALID;
1311 property->name = g_strdup(name);
1312 property->type = DBUS_TYPE_ARRAY;
1313 property->subtype = type;
1315 DBG("name %s type %d value %p", name, type, value);
1318 case DBUS_TYPE_BYTE:
1319 property->value = g_try_malloc(len);
1320 if (property->value != NULL) {
1321 memcpy(property->value,
1322 *((const unsigned char **) value), len);
1323 property->size = len;
1328 __connman_element_lock(element);
1329 element->properties = g_slist_append(element->properties, property);
1330 __connman_element_unlock(element);
1335 static void *get_reference_value(struct connman_element *element,
1336 enum connman_property_id id)
1340 DBG("element %p name %s", element, element->name);
1342 for (list = element->properties; list; list = list->next) {
1343 struct connman_property *property = list->data;
1345 if (property->id != id)
1348 if (!(property->flags & CONNMAN_PROPERTY_FLAG_REFERENCE))
1349 return property->value;
1352 if (element->parent == NULL)
1355 return get_reference_value(element->parent, id);
1358 static void set_reference_properties(struct connman_element *element)
1362 DBG("element %p name %s", element, element->name);
1364 for (list = element->properties; list; list = list->next) {
1365 struct connman_property *property = list->data;
1367 if (!(property->flags & CONNMAN_PROPERTY_FLAG_REFERENCE))
1370 property->value = get_reference_value(element->parent,
1375 static struct connman_property *create_property(struct connman_element *element,
1376 enum connman_property_id id)
1378 struct connman_property *property;
1381 DBG("element %p name %s", element, element->name);
1383 __connman_element_lock(element);
1385 for (list = element->properties; list; list = list->next) {
1386 property = list->data;
1388 if (property->id == id)
1392 property = g_try_new0(struct connman_property, 1);
1393 if (property == NULL)
1396 property->flags = CONNMAN_PROPERTY_FLAG_REFERENCE;
1398 property->name = g_strdup(propid2name(id));
1399 property->type = propid2type(id);
1401 if (property->name == NULL) {
1407 element->properties = g_slist_append(element->properties, property);
1410 __connman_element_unlock(element);
1415 static void create_default_properties(struct connman_element *element)
1417 struct connman_property *property;
1420 DBG("element %p name %s", element, element->name);
1422 for (i = 0; propid_table[i].name; i++) {
1423 DBG("property %s", propid_table[i].name);
1425 property = create_property(element, propid_table[i].id);
1427 property->flags &= ~CONNMAN_PROPERTY_FLAG_REFERENCE;
1429 if (propid_table[i].type != DBUS_TYPE_STRING)
1432 if (propid_table[i].value)
1433 property->value = g_strdup(propid_table[i].value);
1435 property->value = g_strdup("");
1439 static int define_properties_valist(struct connman_element *element,
1442 enum connman_property_id id;
1444 DBG("element %p name %s", element, element->name);
1446 id = va_arg(args, enum connman_property_id);
1448 while (id != CONNMAN_PROPERTY_ID_INVALID) {
1450 DBG("property %d", id);
1452 create_property(element, id);
1454 id = va_arg(args, enum connman_property_id);
1461 * connman_element_define_properties:
1462 * @element: an element
1463 * @varargs: list of property identifiers
1465 * Define the valid properties for an element.
1467 * Returns: %0 on success
1469 int connman_element_define_properties(struct connman_element *element, ...)
1474 DBG("element %p name %s", element, element->name);
1476 va_start(args, element);
1478 err = define_properties_valist(element, args);
1485 int connman_element_create_property(struct connman_element *element,
1486 const char *name, int type)
1491 int connman_element_set_property(struct connman_element *element,
1492 enum connman_property_id id, const void *value)
1495 case CONNMAN_PROPERTY_ID_IPV4_ADDRESS:
1496 __connman_element_lock(element);
1497 g_free(element->ipv4.address);
1498 element->ipv4.address = g_strdup(*((const char **) value));
1499 __connman_element_unlock(element);
1501 case CONNMAN_PROPERTY_ID_IPV4_NETMASK:
1502 __connman_element_lock(element);
1503 g_free(element->ipv4.netmask);
1504 element->ipv4.netmask = g_strdup(*((const char **) value));
1505 __connman_element_unlock(element);
1507 case CONNMAN_PROPERTY_ID_IPV4_GATEWAY:
1508 __connman_element_lock(element);
1509 g_free(element->ipv4.gateway);
1510 element->ipv4.gateway = g_strdup(*((const char **) value));
1511 __connman_element_unlock(element);
1513 case CONNMAN_PROPERTY_ID_IPV4_BROADCAST:
1514 __connman_element_lock(element);
1515 g_free(element->ipv4.broadcast);
1516 element->ipv4.broadcast = g_strdup(*((const char **) value));
1517 __connman_element_unlock(element);
1519 case CONNMAN_PROPERTY_ID_IPV4_NAMESERVER:
1520 __connman_element_lock(element);
1521 g_free(element->ipv4.nameserver);
1522 element->ipv4.nameserver = g_strdup(*((const char **) value));
1523 __connman_element_unlock(element);
1525 case CONNMAN_PROPERTY_ID_WIFI_SECURITY:
1526 __connman_element_lock(element);
1527 g_free(element->wifi.security);
1528 element->wifi.security = g_strdup(*((const char **) value));
1529 __connman_element_unlock(element);
1531 case CONNMAN_PROPERTY_ID_WIFI_PASSPHRASE:
1532 __connman_element_lock(element);
1533 g_free(element->wifi.passphrase);
1534 element->wifi.passphrase = g_strdup(*((const char **) value));
1535 __connman_element_unlock(element);
1544 int connman_element_get_value(struct connman_element *element,
1545 enum connman_property_id id, void *value)
1547 if (element->type == CONNMAN_ELEMENT_TYPE_ROOT)
1551 case CONNMAN_PROPERTY_ID_IPV4_ADDRESS:
1552 if (element->ipv4.address == NULL)
1553 return connman_element_get_value(element->parent,
1555 __connman_element_lock(element);
1556 *((char **) value) = element->ipv4.address;
1557 __connman_element_unlock(element);
1559 case CONNMAN_PROPERTY_ID_IPV4_NETMASK:
1560 if (element->ipv4.netmask == NULL)
1561 return connman_element_get_value(element->parent,
1563 __connman_element_lock(element);
1564 *((char **) value) = element->ipv4.netmask;
1565 __connman_element_unlock(element);
1567 case CONNMAN_PROPERTY_ID_IPV4_GATEWAY:
1568 if (element->ipv4.gateway == NULL)
1569 return connman_element_get_value(element->parent,
1571 __connman_element_lock(element);
1572 *((char **) value) = element->ipv4.gateway;
1573 __connman_element_unlock(element);
1575 case CONNMAN_PROPERTY_ID_IPV4_BROADCAST:
1576 if (element->ipv4.broadcast == NULL)
1577 return connman_element_get_value(element->parent,
1579 __connman_element_lock(element);
1580 *((char **) value) = element->ipv4.broadcast;
1581 __connman_element_unlock(element);
1583 case CONNMAN_PROPERTY_ID_IPV4_NAMESERVER:
1584 if (element->ipv4.nameserver == NULL)
1585 return connman_element_get_value(element->parent,
1587 __connman_element_lock(element);
1588 *((char **) value) = element->ipv4.nameserver;
1589 __connman_element_unlock(element);
1591 case CONNMAN_PROPERTY_ID_WIFI_SECURITY:
1592 if (element->wifi.security == NULL)
1593 return connman_element_get_value(element->parent,
1595 __connman_element_lock(element);
1596 *((char **) value) = element->wifi.security;
1597 __connman_element_unlock(element);
1599 case CONNMAN_PROPERTY_ID_WIFI_PASSPHRASE:
1600 if (element->wifi.passphrase == NULL)
1601 return connman_element_get_value(element->parent,
1603 __connman_element_lock(element);
1604 *((char **) value) = element->wifi.passphrase;
1605 __connman_element_unlock(element);
1614 gboolean connman_element_get_static_property(struct connman_element *element,
1615 const char *name, void *value)
1618 gboolean found = FALSE;
1620 DBG("element %p name %s", element, element->name);
1622 __connman_element_lock(element);
1624 for (list = element->properties; list; list = list->next) {
1625 struct connman_property *property = list->data;
1627 if (!(property->flags & CONNMAN_PROPERTY_FLAG_STATIC))
1630 if (g_str_equal(property->name, name) == TRUE) {
1631 switch (property->type) {
1632 case DBUS_TYPE_STRING:
1633 *((char **) value) = property->value;
1641 __connman_element_unlock(element);
1646 gboolean connman_element_get_static_array_property(struct connman_element *element,
1647 const char *name, void *value, int *len)
1650 gboolean found = FALSE;
1652 DBG("element %p name %s", element, element->name);
1654 __connman_element_lock(element);
1656 for (list = element->properties; list; list = list->next) {
1657 struct connman_property *property = list->data;
1659 if (!(property->flags & CONNMAN_PROPERTY_FLAG_STATIC))
1662 if (g_str_equal(property->name, name) == TRUE) {
1663 *((char **) value) = property->value;
1664 *len = property->size;
1670 __connman_element_unlock(element);
1675 gboolean connman_element_match_static_property(struct connman_element *element,
1676 const char *name, const void *value)
1679 gboolean result = FALSE;
1681 DBG("element %p name %s", element, element->name);
1683 __connman_element_lock(element);
1685 for (list = element->properties; list; list = list->next) {
1686 struct connman_property *property = list->data;
1688 if (!(property->flags & CONNMAN_PROPERTY_FLAG_STATIC))
1691 if (g_str_equal(property->name, name) == FALSE)
1694 if (property->type == DBUS_TYPE_STRING)
1695 result = g_str_equal(property->value,
1696 *((const char **) value));
1702 __connman_element_unlock(element);
1707 static void append_devices(DBusMessageIter *entry)
1709 DBusMessageIter value, iter;
1710 const char *key = "Devices";
1712 dbus_message_iter_append_basic(entry, DBUS_TYPE_STRING, &key);
1714 dbus_message_iter_open_container(entry, DBUS_TYPE_VARIANT,
1715 DBUS_TYPE_ARRAY_AS_STRING DBUS_TYPE_OBJECT_PATH_AS_STRING,
1718 dbus_message_iter_open_container(&value, DBUS_TYPE_ARRAY,
1719 DBUS_TYPE_OBJECT_PATH_AS_STRING, &iter);
1720 __connman_element_list(NULL, CONNMAN_ELEMENT_TYPE_DEVICE, &iter);
1721 dbus_message_iter_close_container(&value, &iter);
1723 dbus_message_iter_close_container(entry, &value);
1726 static void emit_devices_signal(DBusConnection *conn)
1728 DBusMessage *signal;
1729 DBusMessageIter entry;
1731 DBG("conn %p", conn);
1733 signal = dbus_message_new_signal(CONNMAN_MANAGER_PATH,
1734 CONNMAN_MANAGER_INTERFACE, "PropertyChanged");
1738 dbus_message_iter_init_append(signal, &entry);
1740 append_devices(&entry);
1742 g_dbus_send_message(conn, signal);
1745 static void emit_networks_signal(DBusConnection *conn,
1746 struct connman_element *device)
1748 DBusMessage *signal;
1749 DBusMessageIter entry;
1751 DBG("conn %p", conn);
1756 signal = dbus_message_new_signal(device->path,
1757 CONNMAN_DEVICE_INTERFACE, "PropertyChanged");
1761 dbus_message_iter_init_append(signal, &entry);
1763 append_networks(device, &entry);
1765 g_dbus_send_message(conn, signal);
1768 static void append_connections(DBusMessageIter *entry)
1770 DBusMessageIter value, iter;
1771 const char *key = "Connections";
1773 dbus_message_iter_append_basic(entry, DBUS_TYPE_STRING, &key);
1775 dbus_message_iter_open_container(entry, DBUS_TYPE_VARIANT,
1776 DBUS_TYPE_ARRAY_AS_STRING DBUS_TYPE_OBJECT_PATH_AS_STRING,
1779 dbus_message_iter_open_container(&value, DBUS_TYPE_ARRAY,
1780 DBUS_TYPE_OBJECT_PATH_AS_STRING, &iter);
1781 __connman_element_list(NULL, CONNMAN_ELEMENT_TYPE_CONNECTION, &iter);
1782 dbus_message_iter_close_container(&value, &iter);
1784 dbus_message_iter_close_container(entry, &value);
1787 static void emit_connections_signal(DBusConnection *conn)
1789 DBusMessage *signal;
1790 DBusMessageIter entry;
1792 DBG("conn %p", conn);
1794 signal = dbus_message_new_signal(CONNMAN_MANAGER_PATH,
1795 CONNMAN_MANAGER_INTERFACE, "PropertyChanged");
1799 dbus_message_iter_init_append(signal, &entry);
1801 append_connections(&entry);
1803 g_dbus_send_message(conn, signal);
1806 static void append_state(DBusMessageIter *entry, const char *state)
1808 DBusMessageIter value;
1809 const char *key = "State";
1811 dbus_message_iter_append_basic(entry, DBUS_TYPE_STRING, &key);
1813 dbus_message_iter_open_container(entry, DBUS_TYPE_VARIANT,
1814 DBUS_TYPE_STRING_AS_STRING, &value);
1815 dbus_message_iter_append_basic(&value, DBUS_TYPE_STRING, &state);
1816 dbus_message_iter_close_container(entry, &value);
1819 static void emit_state_change(DBusConnection *conn, const char *state)
1821 DBusMessage *signal;
1822 DBusMessageIter entry;
1824 DBG("conn %p", conn);
1826 signal = dbus_message_new_signal(CONNMAN_MANAGER_PATH,
1827 CONNMAN_MANAGER_INTERFACE, "PropertyChanged");
1831 dbus_message_iter_init_append(signal, &entry);
1833 append_state(&entry, state);
1835 g_dbus_send_message(conn, signal);
1838 static void set_signal_strength(struct connman_element *connection)
1840 struct connman_element *element = connection;
1842 while (element != NULL) {
1843 if (element->type == CONNMAN_ELEMENT_TYPE_NETWORK) {
1844 connection->strength = element->strength;
1848 element = element->parent;
1852 static void register_element(gpointer data, gpointer user_data)
1854 struct connman_element *element = data;
1855 const gchar *basepath;
1859 __connman_element_lock(element);
1861 if (element->parent) {
1862 node = g_node_find(element_root, G_PRE_ORDER,
1863 G_TRAVERSE_ALL, element->parent);
1864 basepath = element->parent->path;
1866 if (element->subtype == CONNMAN_ELEMENT_SUBTYPE_UNKNOWN)
1867 element->subtype = element->parent->subtype;
1869 element->parent = element_root->data;
1871 node = element_root;
1875 element->path = g_strdup_printf("%s/%s", basepath, element->name);
1877 set_reference_properties(element);
1879 __connman_element_unlock(element);
1881 DBG("element %p path %s", element, element->path);
1883 __connman_element_load(element);
1885 g_node_append_data(node, element);
1887 if (element->type == CONNMAN_ELEMENT_TYPE_DEVICE &&
1888 element->subtype != CONNMAN_ELEMENT_SUBTYPE_NETWORK) {
1889 if (g_dbus_register_interface(connection, element->path,
1890 CONNMAN_DEVICE_INTERFACE,
1891 device_methods, element_signals,
1892 NULL, element, NULL) == FALSE)
1893 connman_error("Failed to register %s device",
1896 emit_devices_signal(connection);
1899 if (element->type == CONNMAN_ELEMENT_TYPE_NETWORK) {
1900 if (g_dbus_register_interface(connection, element->path,
1901 CONNMAN_NETWORK_INTERFACE,
1902 network_methods, element_signals,
1903 NULL, element, NULL) == FALSE)
1904 connman_error("Failed to register %s network",
1907 emit_networks_signal(connection, element->parent);
1910 if (element->type == CONNMAN_ELEMENT_TYPE_CONNECTION) {
1911 if (g_dbus_register_interface(connection, element->path,
1912 CONNMAN_CONNECTION_INTERFACE,
1913 connection_methods, element_signals,
1914 NULL, element, NULL) == FALSE)
1915 connman_error("Failed to register %s connection",
1918 set_signal_strength(element);
1919 emit_connections_signal(connection);
1920 emit_state_change(connection, "online");
1924 __connman_element_store(element);
1926 for (list = driver_list; list; list = list->next) {
1927 struct connman_driver *driver = list->data;
1929 if (match_driver(element, driver) == FALSE)
1932 DBG("driver %p name %s", driver, driver->name);
1934 if (driver->probe(element) == 0) {
1935 __connman_element_lock(element);
1936 element->driver = driver;
1937 __connman_element_unlock(element);
1939 enable_element(element);
1946 * connman_element_register:
1947 * @element: the element to register
1948 * @parent: the parent to register the element with
1950 * Register an element with the core. It will be register under the given
1951 * parent of if %NULL is provided under the root element.
1953 * Returns: %0 on success
1955 int connman_element_register(struct connman_element *element,
1956 struct connman_element *parent)
1958 DBG("element %p name %s parent %p", element, element->name, parent);
1960 if (element->devname == NULL)
1961 element->devname = g_strdup(element->name);
1963 if (device_filter && element->type == CONNMAN_ELEMENT_TYPE_DEVICE &&
1964 element->subtype != CONNMAN_ELEMENT_SUBTYPE_NETWORK) {
1965 if (g_pattern_match_simple(device_filter,
1966 element->devname) == FALSE) {
1967 DBG("ignoring %s [%s] device", element->name,
1973 if (connman_element_ref(element) == NULL)
1976 __connman_element_lock(element);
1978 if (element->name == NULL) {
1979 element->name = g_strdup(type2string(element->type));
1980 if (element->name == NULL) {
1981 __connman_element_unlock(element);
1986 element->parent = parent;
1988 __connman_element_unlock(element);
1990 register_element(element, NULL);
1995 static gboolean remove_element(GNode *node, gpointer user_data)
1997 struct connman_element *element = node->data;
1998 struct connman_element *root = user_data;
2000 DBG("element %p name %s", element, element->name);
2002 if (element == root)
2005 if (element->driver) {
2006 disable_element(element);
2008 if (element->driver->remove)
2009 element->driver->remove(element);
2011 __connman_element_lock(element);
2012 element->driver = NULL;
2013 __connman_element_unlock(element);
2017 g_node_unlink(node);
2018 g_node_destroy(node);
2021 if (element->type == CONNMAN_ELEMENT_TYPE_CONNECTION) {
2022 if (__connman_element_count(NULL,
2023 CONNMAN_ELEMENT_TYPE_CONNECTION) == 0)
2024 emit_state_change(connection, "offline");
2025 emit_connections_signal(connection);
2027 g_dbus_unregister_interface(connection, element->path,
2028 CONNMAN_CONNECTION_INTERFACE);
2031 if (element->type == CONNMAN_ELEMENT_TYPE_NETWORK) {
2032 emit_networks_signal(connection, element->parent);
2034 g_dbus_unregister_interface(connection, element->path,
2035 CONNMAN_NETWORK_INTERFACE);
2038 if (element->type == CONNMAN_ELEMENT_TYPE_DEVICE &&
2039 element->subtype != CONNMAN_ELEMENT_SUBTYPE_NETWORK) {
2040 emit_devices_signal(connection);
2042 g_dbus_unregister_interface(connection, element->path,
2043 CONNMAN_DEVICE_INTERFACE);
2046 connman_element_unref(element);
2051 void connman_element_unregister(struct connman_element *element)
2055 DBG("element %p name %s", element, element->name);
2057 node = g_node_find(element_root, G_PRE_ORDER, G_TRAVERSE_ALL, element);
2060 g_node_traverse(node, G_POST_ORDER,
2061 G_TRAVERSE_ALL, -1, remove_element, NULL);
2064 void connman_element_unregister_children(struct connman_element *element)
2068 DBG("element %p name %s", element, element->name);
2070 node = g_node_find(element_root, G_PRE_ORDER, G_TRAVERSE_ALL, element);
2073 g_node_traverse(node, G_POST_ORDER,
2074 G_TRAVERSE_ALL, -1, remove_element, element);
2077 static gboolean update_element(GNode *node, gpointer user_data)
2079 struct connman_element *element = node->data;
2080 struct connman_element *root = user_data;
2082 DBG("element %p name %s", element, element->name);
2084 if (element->driver && element->driver->update)
2085 element->driver->update(element);
2087 if (element->type == CONNMAN_ELEMENT_TYPE_CONNECTION &&
2088 root->type == CONNMAN_ELEMENT_TYPE_NETWORK) {
2089 if (element->strength != root->strength) {
2090 element->strength = root->strength;
2091 emit_property_changed(connection, element, "Strength",
2092 DBUS_TYPE_BYTE, &element->strength);
2099 void connman_element_update(struct connman_element *element)
2103 DBG("element %p name %s", element, element->name);
2105 node = g_node_find(element_root, G_PRE_ORDER, G_TRAVERSE_ALL, element);
2108 g_node_traverse(node, G_PRE_ORDER,
2109 G_TRAVERSE_ALL, -1, update_element, element);
2112 int connman_element_set_enabled(struct connman_element *element,
2115 if (element->enabled == enabled)
2118 element->enabled = enabled;
2120 emit_enabled_signal(connection, element);
2125 int __connman_element_init(DBusConnection *conn, const char *device)
2127 struct connman_element *element;
2129 DBG("conn %p", conn);
2131 connection = dbus_connection_ref(conn);
2132 if (connection == NULL)
2135 device_filter = g_strdup(device);
2137 element = connman_element_create("root");
2139 element->path = g_strdup("/");
2140 element->type = CONNMAN_ELEMENT_TYPE_ROOT;
2142 create_default_properties(element);
2144 element_root = g_node_new(element);
2146 __connman_device_init();
2151 static gboolean free_driver(GNode *node, gpointer data)
2153 struct connman_element *element = node->data;
2155 DBG("element %p name %s", element, element->name);
2157 if (element->driver) {
2158 disable_element(element);
2160 if (element->driver->remove)
2161 element->driver->remove(element);
2163 __connman_element_lock(element);
2164 element->driver = NULL;
2165 __connman_element_unlock(element);
2171 static gboolean free_node(GNode *node, gpointer data)
2173 struct connman_element *element = node->data;
2175 DBG("element %p name %s", element, element->name);
2177 if (g_node_depth(node) > 1)
2178 connman_element_unregister(element);
2183 void __connman_element_cleanup(void)
2187 __connman_device_cleanup();
2189 g_node_traverse(element_root, G_POST_ORDER, G_TRAVERSE_ALL, -1,
2192 g_node_traverse(element_root, G_POST_ORDER, G_TRAVERSE_ALL, -1,
2195 g_node_destroy(element_root);
2196 element_root = NULL;
2198 g_free(device_filter);
2200 dbus_connection_unref(connection);