5 * Copyright (C) 2007-2012 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
31 #include <sys/ioctl.h>
32 #include <net/ethernet.h>
37 static GSList *device_list = NULL;
38 static gchar **device_filter = NULL;
39 static gchar **nodevice_filter = NULL;
41 enum connman_pending_type {
47 struct connman_device {
49 enum connman_device_type type;
50 enum connman_pending_type powered_pending; /* Indicates a pending
51 enable/disable request */
52 connman_bool_t powered;
53 connman_bool_t scanning;
54 connman_bool_t disconnected;
55 connman_bool_t reconnect;
64 guint pending_timeout;
66 struct connman_device_driver *driver;
70 struct connman_network *network;
74 static void clear_pending_trigger(struct connman_device *device)
76 if (device->pending_timeout > 0) {
77 g_source_remove(device->pending_timeout);
78 device->pending_timeout = 0;
82 static const char *type2description(enum connman_device_type type)
85 case CONNMAN_DEVICE_TYPE_UNKNOWN:
86 case CONNMAN_DEVICE_TYPE_VENDOR:
88 case CONNMAN_DEVICE_TYPE_ETHERNET:
90 case CONNMAN_DEVICE_TYPE_WIFI:
92 case CONNMAN_DEVICE_TYPE_BLUETOOTH:
94 case CONNMAN_DEVICE_TYPE_GPS:
96 case CONNMAN_DEVICE_TYPE_CELLULAR:
98 case CONNMAN_DEVICE_TYPE_GADGET:
106 static const char *type2string(enum connman_device_type type)
109 case CONNMAN_DEVICE_TYPE_UNKNOWN:
110 case CONNMAN_DEVICE_TYPE_VENDOR:
112 case CONNMAN_DEVICE_TYPE_ETHERNET:
114 case CONNMAN_DEVICE_TYPE_WIFI:
116 case CONNMAN_DEVICE_TYPE_BLUETOOTH:
118 case CONNMAN_DEVICE_TYPE_GPS:
120 case CONNMAN_DEVICE_TYPE_CELLULAR:
122 case CONNMAN_DEVICE_TYPE_GADGET:
130 enum connman_service_type __connman_device_get_service_type(struct connman_device *device)
132 enum connman_device_type type = connman_device_get_type(device);
135 case CONNMAN_DEVICE_TYPE_UNKNOWN:
136 case CONNMAN_DEVICE_TYPE_VENDOR:
137 case CONNMAN_DEVICE_TYPE_GPS:
139 case CONNMAN_DEVICE_TYPE_ETHERNET:
140 return CONNMAN_SERVICE_TYPE_ETHERNET;
141 case CONNMAN_DEVICE_TYPE_WIFI:
142 return CONNMAN_SERVICE_TYPE_WIFI;
143 case CONNMAN_DEVICE_TYPE_BLUETOOTH:
144 return CONNMAN_SERVICE_TYPE_BLUETOOTH;
145 case CONNMAN_DEVICE_TYPE_CELLULAR:
146 return CONNMAN_SERVICE_TYPE_CELLULAR;
147 case CONNMAN_DEVICE_TYPE_GADGET:
148 return CONNMAN_SERVICE_TYPE_GADGET;
152 return CONNMAN_SERVICE_TYPE_UNKNOWN;
155 static gboolean device_pending_reset(gpointer user_data)
157 struct connman_device *device = user_data;
159 DBG("device %p", device);
161 /* Power request timedout, reset power pending state. */
162 device->pending_timeout = 0;
163 device->powered_pending = PENDING_NONE;
168 int __connman_device_enable(struct connman_device *device)
172 DBG("device %p", device);
174 if (!device->driver || !device->driver->enable)
177 /* There is an ongoing power disable request. */
178 if (device->powered_pending == PENDING_DISABLE)
181 if (device->powered_pending == PENDING_ENABLE)
184 if (device->powered_pending == PENDING_NONE && device->powered == TRUE)
187 device->powered_pending = PENDING_ENABLE;
189 err = device->driver->enable(device);
191 * device gets enabled right away.
192 * Invoke the callback
195 connman_device_set_powered(device, TRUE);
199 if (err == -EALREADY) {
200 /* If device is already powered, but connman is not updated */
201 connman_device_set_powered(device, TRUE);
205 * if err == -EINPROGRESS, then the DBus call to the respective daemon
206 * was successful. We set a 4 sec timeout so if the daemon never
207 * returns a reply, we would reset the pending request.
209 if (err == -EINPROGRESS)
210 device->pending_timeout = g_timeout_add_seconds(4,
211 device_pending_reset, device);
216 int __connman_device_disable(struct connman_device *device)
220 DBG("device %p", device);
222 if (!device->driver || !device->driver->disable)
225 /* Ongoing power enable request */
226 if (device->powered_pending == PENDING_ENABLE)
229 if (device->powered_pending == PENDING_DISABLE)
232 if (device->powered_pending == PENDING_NONE && device->powered == FALSE)
235 device->powered_pending = PENDING_DISABLE;
236 device->reconnect = FALSE;
238 if (device->network) {
239 struct connman_service *service =
240 connman_service_lookup_from_network(device->network);
243 __connman_service_disconnect(service);
245 connman_network_set_connected(device->network, FALSE);
248 err = device->driver->disable(device);
249 if (err == 0 || err == -EALREADY) {
250 connman_device_set_powered(device, FALSE);
254 if (err == -EINPROGRESS)
255 device->pending_timeout = g_timeout_add_seconds(4,
256 device_pending_reset, device);
261 static void probe_driver(struct connman_device_driver *driver)
265 DBG("driver %p name %s", driver, driver->name);
267 for (list = device_list; list != NULL; list = list->next) {
268 struct connman_device *device = list->data;
270 if (device->driver != NULL)
273 if (driver->type != device->type)
276 if (driver->probe(device) < 0)
279 device->driver = driver;
281 __connman_technology_add_device(device);
285 static void remove_device(struct connman_device *device)
287 DBG("device %p", device);
289 __connman_device_disable(device);
291 __connman_technology_remove_device(device);
293 if (device->driver->remove)
294 device->driver->remove(device);
296 device->driver = NULL;
299 static void remove_driver(struct connman_device_driver *driver)
303 DBG("driver %p name %s", driver, driver->name);
305 for (list = device_list; list != NULL; list = list->next) {
306 struct connman_device *device = list->data;
308 if (device->driver == driver)
309 remove_device(device);
313 connman_bool_t __connman_device_has_driver(struct connman_device *device)
315 if (device == NULL || device->driver == NULL)
321 static GSList *driver_list = NULL;
323 static gint compare_priority(gconstpointer a, gconstpointer b)
325 const struct connman_device_driver *driver1 = a;
326 const struct connman_device_driver *driver2 = b;
328 return driver2->priority - driver1->priority;
332 * connman_device_driver_register:
333 * @driver: device driver definition
335 * Register a new device driver
337 * Returns: %0 on success
339 int connman_device_driver_register(struct connman_device_driver *driver)
341 DBG("driver %p name %s", driver, driver->name);
343 driver_list = g_slist_insert_sorted(driver_list, driver,
345 probe_driver(driver);
351 * connman_device_driver_unregister:
352 * @driver: device driver definition
354 * Remove a previously registered device driver
356 void connman_device_driver_unregister(struct connman_device_driver *driver)
358 DBG("driver %p name %s", driver, driver->name);
360 driver_list = g_slist_remove(driver_list, driver);
362 remove_driver(driver);
365 static void free_network(gpointer data)
367 struct connman_network *network = data;
369 DBG("network %p", network);
371 __connman_network_set_device(network, NULL);
373 connman_network_unref(network);
376 static void device_destruct(struct connman_device *device)
378 DBG("device %p name %s", device, device->name);
380 clear_pending_trigger(device);
382 g_free(device->ident);
383 g_free(device->node);
384 g_free(device->name);
385 g_free(device->address);
386 g_free(device->interface);
387 g_free(device->path);
388 g_free(device->devname);
390 g_free(device->last_network);
392 g_hash_table_destroy(device->networks);
393 device->networks = NULL;
399 * connman_device_create:
400 * @node: device node name (for example an address)
403 * Allocate a new device of given #type and assign the #node name to it.
405 * Returns: a newly-allocated #connman_device structure
407 struct connman_device *connman_device_create(const char *node,
408 enum connman_device_type type)
410 struct connman_device *device;
412 DBG("node %s type %d", node, type);
414 device = g_try_new0(struct connman_device, 1);
418 DBG("device %p", device);
420 device->refcount = 1;
423 device->name = g_strdup(type2description(device->type));
425 device->networks = g_hash_table_new_full(g_str_hash, g_str_equal,
426 g_free, free_network);
428 device_list = g_slist_prepend(device_list, device);
434 * connman_device_ref:
435 * @device: device structure
437 * Increase reference counter of device
439 struct connman_device *connman_device_ref_debug(struct connman_device *device,
440 const char *file, int line, const char *caller)
442 DBG("%p ref %d by %s:%d:%s()", device, device->refcount + 1,
445 __sync_fetch_and_add(&device->refcount, 1);
451 * connman_device_unref:
452 * @device: device structure
454 * Decrease reference counter of device
456 void connman_device_unref_debug(struct connman_device *device,
457 const char *file, int line, const char *caller)
459 DBG("%p ref %d by %s:%d:%s()", device, device->refcount - 1,
462 if (__sync_fetch_and_sub(&device->refcount, 1) != 1)
465 if (device->driver) {
466 device->driver->remove(device);
467 device->driver = NULL;
470 device_list = g_slist_remove(device_list, device);
472 device_destruct(device);
475 const char *__connman_device_get_type(struct connman_device *device)
477 return type2string(device->type);
481 * connman_device_get_type:
482 * @device: device structure
486 enum connman_device_type connman_device_get_type(struct connman_device *device)
492 * connman_device_set_index:
493 * @device: device structure
494 * @index: index number
496 * Set index number of device
498 void connman_device_set_index(struct connman_device *device, int index)
500 device->index = index;
504 * connman_device_get_index:
505 * @device: device structure
507 * Get index number of device
509 int connman_device_get_index(struct connman_device *device)
511 return device->index;
515 * connman_device_set_interface:
516 * @device: device structure
517 * @interface: interface name
519 * Set interface name of device
521 void connman_device_set_interface(struct connman_device *device,
522 const char *interface)
524 g_free(device->devname);
525 device->devname = g_strdup(interface);
527 g_free(device->interface);
528 device->interface = g_strdup(interface);
530 if (device->name == NULL) {
531 const char *str = type2description(device->type);
532 if (str != NULL && device->interface != NULL)
533 device->name = g_strdup_printf("%s (%s)", str,
539 * connman_device_set_ident:
540 * @device: device structure
541 * @ident: unique identifier
543 * Set unique identifier of device
545 void connman_device_set_ident(struct connman_device *device,
548 g_free(device->ident);
549 device->ident = g_strdup(ident);
552 const char *connman_device_get_ident(struct connman_device *device)
554 return device->ident;
558 * connman_device_set_powered:
559 * @device: device structure
560 * @powered: powered state
562 * Change power state of device
564 int connman_device_set_powered(struct connman_device *device,
565 connman_bool_t powered)
567 enum connman_service_type type;
569 DBG("driver %p powered %d", device, powered);
571 if (device->powered == powered)
574 clear_pending_trigger(device);
576 device->powered_pending = PENDING_NONE;
578 device->powered = powered;
580 type = __connman_device_get_service_type(device);
582 if (device->powered == FALSE) {
583 __connman_technology_disabled(type);
587 __connman_technology_enabled(type);
589 connman_device_set_disconnected(device, FALSE);
590 device->scanning = FALSE;
592 if (device->driver && device->driver->scan)
593 device->driver->scan(device, NULL, 0, NULL, NULL, NULL);
598 connman_bool_t connman_device_get_powered(struct connman_device *device)
600 return device->powered;
603 static int device_scan(struct connman_device *device)
605 if (!device->driver || !device->driver->scan)
608 if (device->powered == FALSE)
611 return device->driver->scan(device, NULL, 0, NULL, NULL, NULL);
614 int __connman_device_disconnect(struct connman_device *device)
619 DBG("device %p", device);
621 connman_device_set_disconnected(device, TRUE);
623 g_hash_table_iter_init(&iter, device->networks);
625 while (g_hash_table_iter_next(&iter, &key, &value) == TRUE) {
626 struct connman_network *network = value;
628 if (connman_network_get_connecting(network) == TRUE) {
630 * Skip network in the process of connecting.
631 * This is a workaround for WiFi networks serviced
632 * by the supplicant plugin that hold a reference
633 * to the network. If we disconnect the network
634 * here then the referenced object will not be
635 * registered and usage (like launching DHCP client)
636 * will fail. There is nothing to be gained by
637 * removing the network here anyway.
639 connman_warn("Skipping disconnect of %s, network is connecting.",
640 connman_network_get_identifier(network));
644 __connman_network_disconnect(network);
650 int connman_device_disconnect_service(struct connman_device *device)
652 DBG("device %p", device);
654 device->reconnect = FALSE;
656 if (device->network) {
657 struct connman_service *service =
658 connman_service_lookup_from_network(device->network);
661 __connman_service_disconnect(service);
663 connman_network_set_connected(device->network, FALSE);
669 int connman_device_reconnect_service(struct connman_device *device)
671 DBG("device %p", device);
673 device->reconnect = TRUE;
675 __connman_service_auto_connect();
680 static void mark_network_available(gpointer key, gpointer value,
683 struct connman_network *network = value;
685 connman_network_set_available(network, TRUE);
688 static void mark_network_unavailable(gpointer key, gpointer value,
691 struct connman_network *network = value;
693 if (connman_network_get_connected(network) == TRUE ||
694 connman_network_get_connecting(network) == TRUE)
697 connman_network_set_available(network, FALSE);
700 static gboolean remove_unavailable_network(gpointer key, gpointer value,
703 struct connman_network *network = value;
705 if (connman_network_get_connected(network) == TRUE)
708 if (connman_network_get_available(network) == TRUE)
714 void __connman_device_cleanup_networks(struct connman_device *device)
716 g_hash_table_foreach_remove(device->networks,
717 remove_unavailable_network, NULL);
720 connman_bool_t connman_device_get_scanning(struct connman_device *device)
722 return device->scanning;
725 void connman_device_reset_scanning(struct connman_device *device)
727 g_hash_table_foreach(device->networks,
728 mark_network_available, NULL);
732 * connman_device_set_scanning:
733 * @device: device structure
734 * @scanning: scanning state
736 * Change scanning state of device
738 int connman_device_set_scanning(struct connman_device *device,
739 connman_bool_t scanning)
741 DBG("device %p scanning %d", device, scanning);
743 if (!device->driver || !device->driver->scan)
746 if (device->scanning == scanning)
749 device->scanning = scanning;
751 if (scanning == TRUE) {
752 __connman_technology_scan_started(device);
754 g_hash_table_foreach(device->networks,
755 mark_network_unavailable, NULL);
760 __connman_device_cleanup_networks(device);
762 __connman_technology_scan_stopped(device);
764 __connman_service_auto_connect();
770 * connman_device_set_disconnected:
771 * @device: device structure
772 * @disconnected: disconnected state
774 * Change disconnected state of device (only for device with networks)
776 int connman_device_set_disconnected(struct connman_device *device,
777 connman_bool_t disconnected)
779 DBG("device %p disconnected %d", device, disconnected);
781 if (device->disconnected == disconnected)
784 device->disconnected = disconnected;
790 * connman_device_get_disconnected:
791 * @device: device structure
793 * Get device disconnected state
795 connman_bool_t connman_device_get_disconnected(struct connman_device *device)
797 return device->disconnected;
801 * connman_device_set_string:
802 * @device: device structure
803 * @key: unique identifier
804 * @value: string value
806 * Set string value for specific key
808 int connman_device_set_string(struct connman_device *device,
809 const char *key, const char *value)
811 DBG("device %p key %s value %s", device, key, value);
813 if (g_str_equal(key, "Address") == TRUE) {
814 g_free(device->address);
815 device->address = g_strdup(value);
816 } else if (g_str_equal(key, "Name") == TRUE) {
817 g_free(device->name);
818 device->name = g_strdup(value);
819 } else if (g_str_equal(key, "Node") == TRUE) {
820 g_free(device->node);
821 device->node = g_strdup(value);
822 } else if (g_str_equal(key, "Path") == TRUE) {
823 g_free(device->path);
824 device->path = g_strdup(value);
833 * connman_device_get_string:
834 * @device: device structure
835 * @key: unique identifier
837 * Get string value for specific key
839 const char *connman_device_get_string(struct connman_device *device,
842 DBG("device %p key %s", device, key);
844 if (g_str_equal(key, "Address") == TRUE)
845 return device->address;
846 else if (g_str_equal(key, "Name") == TRUE)
848 else if (g_str_equal(key, "Node") == TRUE)
850 else if (g_str_equal(key, "Interface") == TRUE)
851 return device->interface;
852 else if (g_str_equal(key, "Path") == TRUE)
859 * connman_device_add_network:
860 * @device: device structure
861 * @network: network structure
863 * Add new network to the device
865 int connman_device_add_network(struct connman_device *device,
866 struct connman_network *network)
868 const char *identifier = connman_network_get_identifier(network);
870 DBG("device %p network %p", device, network);
872 if (identifier == NULL)
875 connman_network_ref(network);
877 __connman_network_set_device(network, device);
879 g_hash_table_replace(device->networks, g_strdup(identifier),
886 * connman_device_get_network:
887 * @device: device structure
888 * @identifier: network identifier
890 * Get network for given identifier
892 struct connman_network *connman_device_get_network(struct connman_device *device,
893 const char *identifier)
895 DBG("device %p identifier %s", device, identifier);
897 return g_hash_table_lookup(device->networks, identifier);
901 * connman_device_remove_network:
902 * @device: device structure
903 * @identifier: network identifier
905 * Remove network for given identifier
907 int connman_device_remove_network(struct connman_device *device,
908 struct connman_network *network)
910 const char *identifier;
912 DBG("device %p network %p", device, network);
917 identifier = connman_network_get_identifier(network);
918 g_hash_table_remove(device->networks, identifier);
923 void connman_device_remove_all_networks(struct connman_device *device)
925 g_hash_table_remove_all(device->networks);
928 void __connman_device_set_network(struct connman_device *device,
929 struct connman_network *network)
936 if (device->network == network)
939 if (network != NULL) {
940 name = connman_network_get_string(network, "Name");
941 g_free(device->last_network);
942 device->last_network = g_strdup(name);
944 device->network = network;
946 g_free(device->last_network);
947 device->last_network = NULL;
949 device->network = NULL;
953 void __connman_device_set_reconnect(struct connman_device *device,
954 connman_bool_t reconnect)
956 device->reconnect = reconnect;
959 connman_bool_t __connman_device_get_reconnect(
960 struct connman_device *device)
962 return device->reconnect;
965 static gboolean match_driver(struct connman_device *device,
966 struct connman_device_driver *driver)
968 if (device->type == driver->type ||
969 driver->type == CONNMAN_DEVICE_TYPE_UNKNOWN)
976 * connman_device_register:
977 * @device: device structure
979 * Register device with the system
981 int connman_device_register(struct connman_device *device)
985 DBG("device %p name %s", device, device->name);
987 if (device->driver != NULL)
990 for (list = driver_list; list; list = list->next) {
991 struct connman_device_driver *driver = list->data;
993 if (match_driver(device, driver) == FALSE)
996 DBG("driver %p name %s", driver, driver->name);
998 if (driver->probe(device) == 0) {
999 device->driver = driver;
1004 if (device->driver == NULL)
1007 return __connman_technology_add_device(device);
1011 * connman_device_unregister:
1012 * @device: device structure
1014 * Unregister device with the system
1016 void connman_device_unregister(struct connman_device *device)
1018 DBG("device %p name %s", device, device->name);
1020 if (device->driver == NULL)
1023 remove_device(device);
1027 * connman_device_get_data:
1028 * @device: device structure
1030 * Get private device data pointer
1032 void *connman_device_get_data(struct connman_device *device)
1034 return device->driver_data;
1038 * connman_device_set_data:
1039 * @device: device structure
1040 * @data: data pointer
1042 * Set private device data pointer
1044 void connman_device_set_data(struct connman_device *device, void *data)
1046 device->driver_data = data;
1049 struct connman_device *__connman_device_find_device(
1050 enum connman_service_type type)
1054 for (list = device_list; list != NULL; list = list->next) {
1055 struct connman_device *device = list->data;
1056 enum connman_service_type service_type =
1057 __connman_device_get_service_type(device);
1059 if (service_type != type)
1068 struct connman_device *connman_device_find_by_index(int index)
1072 for (list = device_list; list != NULL; list = list->next) {
1073 struct connman_device *device = list->data;
1074 if (device->index == index)
1082 * connman_device_set_regdom
1083 * @device: device structure
1084 * @alpha2: string representing regulatory domain
1086 * Set regulatory domain on device basis
1088 int connman_device_set_regdom(struct connman_device *device,
1091 if (device->driver == NULL || device->driver->set_regdom == NULL)
1094 if (device->powered == FALSE)
1097 return device->driver->set_regdom(device, alpha2);
1101 * connman_device_regdom_notify
1102 * @device: device structure
1103 * @alpha2: string representing regulatory domain
1105 * Notify on setting regulatory domain on device basis
1107 void connman_device_regdom_notify(struct connman_device *device,
1108 int result, const char *alpha2)
1110 __connman_technology_notify_regdom_by_device(device, result, alpha2);
1113 int __connman_device_request_scan(enum connman_service_type type)
1115 connman_bool_t success = FALSE;
1116 int last_err = -ENOSYS;
1121 case CONNMAN_SERVICE_TYPE_UNKNOWN:
1122 case CONNMAN_SERVICE_TYPE_SYSTEM:
1123 case CONNMAN_SERVICE_TYPE_ETHERNET:
1124 case CONNMAN_SERVICE_TYPE_BLUETOOTH:
1125 case CONNMAN_SERVICE_TYPE_CELLULAR:
1126 case CONNMAN_SERVICE_TYPE_GPS:
1127 case CONNMAN_SERVICE_TYPE_VPN:
1128 case CONNMAN_SERVICE_TYPE_GADGET:
1130 case CONNMAN_SERVICE_TYPE_WIFI:
1134 for (list = device_list; list != NULL; list = list->next) {
1135 struct connman_device *device = list->data;
1136 enum connman_service_type service_type =
1137 __connman_device_get_service_type(device);
1139 if (service_type != CONNMAN_SERVICE_TYPE_UNKNOWN &&
1140 service_type != type) {
1144 err = device_scan(device);
1145 if (err == 0 || err == -EALREADY || err == -EINPROGRESS) {
1149 DBG("device %p err %d", device, err);
1153 if (success == TRUE)
1159 int __connman_device_request_hidden_scan(struct connman_device *device,
1160 const char *ssid, unsigned int ssid_len,
1161 const char *identity, const char *passphrase,
1164 DBG("device %p", device);
1166 if (device == NULL || device->driver == NULL ||
1167 device->driver->scan == NULL)
1170 if (device->scanning == TRUE)
1173 return device->driver->scan(device, ssid, ssid_len,
1174 identity, passphrase, user_data);
1177 static char *index2ident(int index, const char *prefix)
1180 struct ether_addr eth;
1187 sk = socket(PF_INET, SOCK_DGRAM | SOCK_CLOEXEC, 0);
1191 memset(&ifr, 0, sizeof(ifr));
1192 ifr.ifr_ifindex = index;
1194 err = ioctl(sk, SIOCGIFNAME, &ifr);
1197 err = ioctl(sk, SIOCGIFHWADDR, &ifr);
1204 len = prefix ? strlen(prefix) + 18 : 18;
1206 str = g_malloc(len);
1210 memcpy(ð, &ifr.ifr_hwaddr.sa_data, sizeof(eth));
1211 snprintf(str, len, "%s%02x%02x%02x%02x%02x%02x",
1212 prefix ? prefix : "",
1213 eth.ether_addr_octet[0],
1214 eth.ether_addr_octet[1],
1215 eth.ether_addr_octet[2],
1216 eth.ether_addr_octet[3],
1217 eth.ether_addr_octet[4],
1218 eth.ether_addr_octet[5]);
1223 static char *index2addr(int index)
1226 struct ether_addr eth;
1233 sk = socket(PF_INET, SOCK_DGRAM | SOCK_CLOEXEC, 0);
1237 memset(&ifr, 0, sizeof(ifr));
1238 ifr.ifr_ifindex = index;
1240 err = ioctl(sk, SIOCGIFNAME, &ifr);
1243 err = ioctl(sk, SIOCGIFHWADDR, &ifr);
1254 memcpy(ð, &ifr.ifr_hwaddr.sa_data, sizeof(eth));
1255 snprintf(str, 18, "%02X:%02X:%02X:%02X:%02X:%02X",
1256 eth.ether_addr_octet[0],
1257 eth.ether_addr_octet[1],
1258 eth.ether_addr_octet[2],
1259 eth.ether_addr_octet[3],
1260 eth.ether_addr_octet[4],
1261 eth.ether_addr_octet[5]);
1266 struct connman_device *connman_device_create_from_index(int index)
1268 enum connman_device_type type;
1269 struct connman_device *device;
1270 char *devname, *ident = NULL;
1271 char *addr = NULL, *name = NULL;
1276 devname = connman_inet_ifname(index);
1277 if (devname == NULL)
1280 if (__connman_device_isfiltered(devname) == TRUE) {
1281 connman_info("Ignoring interface %s (filtered)", devname);
1286 type = __connman_rtnl_get_device_type(index);
1289 case CONNMAN_DEVICE_TYPE_UNKNOWN:
1290 connman_info("Ignoring interface %s (type unknown)", devname);
1293 case CONNMAN_DEVICE_TYPE_ETHERNET:
1294 case CONNMAN_DEVICE_TYPE_GADGET:
1295 case CONNMAN_DEVICE_TYPE_WIFI:
1296 name = index2ident(index, "");
1297 addr = index2addr(index);
1299 case CONNMAN_DEVICE_TYPE_BLUETOOTH:
1300 case CONNMAN_DEVICE_TYPE_CELLULAR:
1301 case CONNMAN_DEVICE_TYPE_GPS:
1302 case CONNMAN_DEVICE_TYPE_VENDOR:
1303 name = g_strdup(devname);
1307 device = connman_device_create(name, type);
1312 case CONNMAN_DEVICE_TYPE_UNKNOWN:
1313 case CONNMAN_DEVICE_TYPE_VENDOR:
1314 case CONNMAN_DEVICE_TYPE_GPS:
1316 case CONNMAN_DEVICE_TYPE_ETHERNET:
1317 case CONNMAN_DEVICE_TYPE_GADGET:
1318 ident = index2ident(index, NULL);
1320 case CONNMAN_DEVICE_TYPE_WIFI:
1321 ident = index2ident(index, NULL);
1323 case CONNMAN_DEVICE_TYPE_BLUETOOTH:
1325 case CONNMAN_DEVICE_TYPE_CELLULAR:
1326 ident = index2ident(index, NULL);
1330 connman_device_set_index(device, index);
1331 connman_device_set_interface(device, devname);
1333 if (ident != NULL) {
1334 connman_device_set_ident(device, ident);
1338 connman_device_set_string(device, "Address", addr);
1348 connman_bool_t __connman_device_isfiltered(const char *devname)
1351 char **blacklisted_interfaces;
1354 if (device_filter == NULL)
1357 for (pattern = device_filter, match = FALSE; *pattern; pattern++) {
1358 if (g_pattern_match_simple(*pattern, devname) == TRUE) {
1364 if (match == FALSE) {
1365 DBG("ignoring device %s (match)", devname);
1370 if (g_pattern_match_simple("dummy*", devname) == TRUE) {
1371 DBG("ignoring dummy networking devices");
1375 if (nodevice_filter == NULL)
1378 for (pattern = nodevice_filter; *pattern; pattern++) {
1379 if (g_pattern_match_simple(*pattern, devname) == TRUE) {
1380 DBG("ignoring device %s (no match)", devname);
1386 blacklisted_interfaces =
1387 connman_setting_get_string_list("NetworkInterfaceBlacklist");
1388 if (blacklisted_interfaces == NULL)
1391 for (pattern = blacklisted_interfaces; *pattern; pattern++) {
1392 if (g_str_has_prefix(devname, *pattern) == TRUE) {
1393 DBG("ignoring device %s (blacklist)", devname);
1401 static void cleanup_devices(void)
1404 * Check what interfaces are currently up and if connman is
1405 * suppose to handle the interface, then cleanup the mess
1406 * related to that interface. There might be weird routes etc
1407 * that are related to that interface and that might confuse
1408 * connmand. So in this case we just turn the interface down
1409 * so that kernel removes routes/addresses automatically and
1410 * then proceed the startup.
1412 * Note that this cleanup must be done before rtnl/detect code
1413 * has activated interface watches.
1419 interfaces = __connman_inet_get_running_interfaces();
1421 if (interfaces == NULL)
1424 for (i = 0; interfaces[i] != NULL; i++) {
1425 connman_bool_t filtered;
1428 filtered = __connman_device_isfiltered(interfaces[i]);
1429 if (filtered == TRUE)
1432 index = connman_inet_ifindex(interfaces[i]);
1436 DBG("cleaning up %s index %d", interfaces[i], index);
1438 connman_inet_ifdown(index);
1441 * ConnMan will turn the interface UP automatically so
1442 * no need to do it here.
1446 g_strfreev(interfaces);
1449 int __connman_device_init(const char *device, const char *nodevice)
1454 device_filter = g_strsplit(device, ",", -1);
1456 if (nodevice != NULL)
1457 nodevice_filter = g_strsplit(nodevice, ",", -1);
1464 void __connman_device_cleanup(void)
1468 g_strfreev(nodevice_filter);
1469 g_strfreev(device_filter);