5 * Copyright (C) 2007-2010 Intel Corporation. All rights reserved.
6 * Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies).
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License version 2 as
10 * published by the Free Software Foundation.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
32 #define CONNMAN_API_SUBJECT_TO_CHANGE
33 #include <connman/plugin.h>
34 #include <connman/element.h>
35 #include <connman/device.h>
36 #include <connman/network.h>
37 #include <connman/dbus.h>
38 #include <connman/inet.h>
39 #include <connman/log.h>
41 #define OFONO_SERVICE "org.ofono"
43 #define OFONO_MANAGER_INTERFACE OFONO_SERVICE ".Manager"
44 #define OFONO_MODEM_INTERFACE OFONO_SERVICE ".Modem"
45 #define OFONO_GPRS_INTERFACE OFONO_SERVICE ".DataConnectionManager"
46 #define OFONO_SIM_INTERFACE OFONO_SERVICE ".SimManager"
47 #define OFONO_PRI_CONTEXT_INTERFACE OFONO_SERVICE ".PrimaryDataContext"
48 #define OFONO_REGISTRATION_INTERFACE OFONO_SERVICE ".NetworkRegistration"
50 #define PROPERTY_CHANGED "PropertyChanged"
51 #define GET_PROPERTIES "GetProperties"
52 #define SET_PROPERTY "SetProperty"
53 #define CREATE_CONTEXT "CreateContext"
57 #define CONTEXT_NAME "3G Connection"
58 #define CONTEXT_TYPE "internet"
60 static DBusConnection *connection;
62 static GHashTable *modem_hash = NULL;
66 struct connman_device *device;
70 static int modem_probe(struct connman_device *device)
72 DBG("device %p", device);
77 static void modem_remove(struct connman_device *device)
79 DBG("device %p", device);
82 static void set_property_reply(DBusPendingCall *call, void *user_data)
86 char const *name = user_data;
90 dbus_error_init(&error);
92 reply = dbus_pending_call_steal_reply(call);
94 if (dbus_set_error_from_message(&error, reply)) {
95 connman_error("SetProperty(\"%s\") %s", name,
97 dbus_error_free(&error);
100 dbus_message_unref(reply);
102 dbus_pending_call_unref(call);
105 static int set_property(const char *path, const char *interface,
106 const char *property, int type, void *value,
107 DBusPendingCallNotifyFunction notify, void *user_data,
108 DBusFreeFunction free_function)
110 DBusMessage *message;
111 DBusMessageIter iter;
112 DBusPendingCall *call;
114 DBG("path %s %s.%s", path, interface, property);
116 g_assert(notify == NULL ? free_function == NULL : 1);
121 message = dbus_message_new_method_call(OFONO_SERVICE, path,
122 interface, SET_PROPERTY);
126 dbus_message_set_auto_start(message, FALSE);
128 dbus_message_iter_init_append(message, &iter);
129 connman_dbus_property_append_basic(&iter, property, type, value);
131 if (dbus_connection_send_with_reply(connection, message,
132 &call, TIMEOUT) == FALSE) {
133 connman_error("Failed to change \"%s\" property on %s",
134 property, interface);
135 dbus_message_unref(message);
140 connman_error("D-Bus connection not available");
141 dbus_message_unref(message);
145 if (notify == NULL) {
146 notify = set_property_reply;
147 user_data = (void *)property;
148 free_function = NULL;
151 dbus_pending_call_set_notify(call, notify, user_data, free_function);
153 dbus_message_unref(message);
158 static int gprs_change_powered(const char *path, dbus_bool_t powered)
160 DBG("path %s powered %d", path, powered);
162 return set_property(path, OFONO_GPRS_INTERFACE, "Powered",
163 DBUS_TYPE_BOOLEAN, &powered,
167 static int modem_enable(struct connman_device *device)
169 const char *path = connman_device_get_string(device, "Path");
171 DBG("device %p, path, %s", device, path);
173 return gprs_change_powered(path, TRUE);
176 static int modem_disable(struct connman_device *device)
178 const char *path = connman_device_get_string(device, "Path");
180 DBG("device %p, path %s", device, path);
182 return gprs_change_powered(path, FALSE);
185 static struct connman_device_driver modem_driver = {
187 .type = CONNMAN_DEVICE_TYPE_CELLULAR,
188 .probe = modem_probe,
189 .remove = modem_remove,
190 .enable = modem_enable,
191 .disable = modem_disable,
194 static char *get_ident(const char *path)
201 pos = strrchr(path, '/');
205 return g_strdup(pos + 1);
208 static void create_service(struct connman_network *network)
215 path = connman_network_get_string(network, "Path");
217 group = get_ident(path);
219 connman_network_set_group(network, group);
224 static void set_network_name_reply(DBusPendingCall *call, void *user_data)
226 struct connman_network *network = user_data;
228 DBusMessageIter array, dict;
230 DBG("network %p", network);
232 reply = dbus_pending_call_steal_reply(call);
234 if (dbus_message_iter_init(reply, &array) == FALSE)
237 if (dbus_message_iter_get_arg_type(&array) != DBUS_TYPE_ARRAY)
240 dbus_message_iter_recurse(&array, &dict);
242 while (dbus_message_iter_get_arg_type(&dict) == DBUS_TYPE_DICT_ENTRY) {
243 DBusMessageIter entry, value;
246 dbus_message_iter_recurse(&dict, &entry);
247 dbus_message_iter_get_basic(&entry, &key);
249 dbus_message_iter_next(&entry);
250 dbus_message_iter_recurse(&entry, &value);
253 * 'Operator' is deprecated since version 0.20, but
254 * keep it here for backward compatibility reasons.
256 if (g_str_equal(key, "Operator") == TRUE ||
257 g_str_equal(key, "Name") == TRUE) {
260 dbus_message_iter_get_basic(&value, &name);
261 connman_network_set_name(network, name);
262 create_service(network);
263 } else if (g_strcmp0(key, "Status") == 0) {
265 connman_bool_t roaming;
267 dbus_message_iter_get_basic(&value, &status);
268 if (g_strcmp0(status, "roaming") == 0)
270 else if (g_strcmp0(status, "registered") == 0)
273 dbus_message_iter_next(&dict);
277 connman_network_set_roaming(network, roaming);
278 connman_network_update(network);
281 dbus_message_iter_next(&dict);
284 dbus_message_unref(reply);
286 dbus_pending_call_unref(call);
289 static void set_network_name(struct connman_network *network)
291 struct connman_device *device;
292 DBusMessage *message;
293 DBusPendingCall *call;
296 device = connman_network_get_device(network);
298 path = connman_device_get_string(device, "Path");
302 DBG("path %s", path);
304 message = dbus_message_new_method_call(OFONO_SERVICE, path,
305 OFONO_REGISTRATION_INTERFACE, GET_PROPERTIES);
309 dbus_message_set_auto_start(message, FALSE);
311 if (dbus_connection_send_with_reply(connection, message,
312 &call, TIMEOUT) == FALSE) {
313 connman_error("Failed to get operator");
318 connman_error("D-Bus connection not available");
322 dbus_pending_call_set_notify(call, set_network_name_reply,
323 (void *)network, NULL);
326 dbus_message_unref(message);
329 static void config_network_reply(DBusPendingCall *call, void *user_data)
331 struct connman_network *network = user_data;
333 DBusMessageIter array, dict;
334 gboolean internet_type = FALSE;
336 DBG("network %p", network);
338 reply = dbus_pending_call_steal_reply(call);
340 if (dbus_message_iter_init(reply, &array) == FALSE)
343 if (dbus_message_iter_get_arg_type(&array) != DBUS_TYPE_ARRAY)
346 dbus_message_iter_recurse(&array, &dict);
348 while (dbus_message_iter_get_arg_type(&dict) == DBUS_TYPE_DICT_ENTRY) {
349 DBusMessageIter entry, value;
352 dbus_message_iter_recurse(&dict, &entry);
353 dbus_message_iter_get_basic(&entry, &key);
355 dbus_message_iter_next(&entry);
356 dbus_message_iter_recurse(&entry, &value);
358 if (g_str_equal(key, "Type") == TRUE) {
361 dbus_message_iter_get_basic(&value, &type);
362 if (g_strcmp0(type, "internet") == 0) {
363 internet_type = TRUE;
365 connman_network_set_protocol(network,
366 CONNMAN_NETWORK_PROTOCOL_IP);
367 set_network_name(network);
369 internet_type = FALSE;
371 connman_network_set_protocol(network,
372 CONNMAN_NETWORK_PROTOCOL_UNKNOWN);
376 dbus_message_iter_next(&dict);
380 dbus_message_unref(reply);
382 dbus_pending_call_unref(call);
385 static void config_network(struct connman_network *network, const char *path)
387 DBusMessage *message;
388 DBusPendingCall *call;
390 DBG("path %s", path);
392 message = dbus_message_new_method_call(OFONO_SERVICE, path,
393 OFONO_PRI_CONTEXT_INTERFACE, GET_PROPERTIES);
397 dbus_message_set_auto_start(message, FALSE);
399 if (dbus_connection_send_with_reply(connection, message,
400 &call, TIMEOUT) == FALSE) {
401 connman_error("Failed to get Primary Context");
406 connman_error("D-Bus connection not available");
410 dbus_pending_call_set_notify(call, config_network_reply,
411 (void *)network, NULL);
414 dbus_message_unref(message);
417 static gboolean registration_changed(DBusConnection *connection,
418 DBusMessage *message, void *user_data)
420 const char *path = dbus_message_get_path(message);
421 struct connman_network *network = user_data;
422 DBusMessageIter iter, value;
425 DBG("path %s", path);
427 if (dbus_message_iter_init(message, &iter) == FALSE)
430 dbus_message_iter_get_basic(&iter, &key);
434 dbus_message_iter_next(&iter);
435 dbus_message_iter_recurse(&iter, &value);
437 if (g_strcmp0(key, "Name") == 0 ||
438 g_strcmp0(key, "Operator") == 0) {
441 dbus_message_iter_get_basic(&value, &name);
442 DBG("name %s", name);
443 connman_network_set_name(network, name);
444 create_service(network);
445 } else if (g_strcmp0(key, "Strength") == 0) {
446 connman_uint8_t strength;
448 dbus_message_iter_get_basic(&value, &strength);
449 connman_network_set_strength(network, strength);
450 connman_network_update(network);
451 } else if (g_strcmp0(key, "Status") == 0) {
453 connman_bool_t roaming;
455 dbus_message_iter_get_basic(&value, &status);
456 if (g_strcmp0(status, "roaming") == 0)
458 else if (g_strcmp0(status, "registered") == 0)
463 connman_network_set_roaming(network, roaming);
464 connman_network_update(network);
470 static int network_probe(struct connman_network *network)
475 reg_watch = g_dbus_add_signal_watch(connection, NULL, NULL,
476 OFONO_REGISTRATION_INTERFACE,
478 registration_changed,
483 connman_network_set_data(network, GUINT_TO_POINTER(reg_watch));
485 path = connman_network_get_string(network, "Path");
487 DBG("network %p path %s", network, path);
489 config_network(network, path);
494 static struct connman_network *pending_network;
496 static gboolean pending_network_is_available(
497 struct connman_network *pending_network)
499 struct connman_device *device;
500 struct connman_network *network;
501 const char *identifier;
504 /* Modem may be removed during waiting for active reply */
505 device = connman_network_get_device(pending_network);
506 if (device == NULL) {
507 DBG("Modem is removed");
511 identifier = connman_network_get_identifier(pending_network);
513 ident = g_strdup(identifier);
515 connman_network_unref(pending_network);
517 /* network may be removed during waiting for active reply */
518 network = connman_device_get_network(device, ident);
528 static void set_active_reply(DBusPendingCall *call, void *user_data)
532 struct connman_network *network = user_data;
534 DBG("network %p", network);
536 reply = dbus_pending_call_steal_reply(call);
538 if (pending_network_is_available(network) == FALSE)
541 dbus_error_init(&error);
542 if (dbus_set_error_from_message(&error, reply)) {
543 if (connman_network_get_index(network) < 0)
544 connman_network_set_error(network,
545 CONNMAN_NETWORK_ERROR_ASSOCIATE_FAIL);
547 pending_network = NULL;
549 connman_error("%s", error.message);
551 dbus_error_free(&error);
553 pending_network = network;
556 dbus_message_unref(reply);
558 dbus_pending_call_unref(call);
561 static int set_network_active(struct connman_network *network,
566 const char *path = connman_network_get_string(network, "Path");
568 DBG("network %p, path %s, active %d", network, path, active);
570 error = set_property(path, OFONO_PRI_CONTEXT_INTERFACE,
571 "Active", DBUS_TYPE_BOOLEAN, &active,
572 set_active_reply, network, NULL);
573 if (active == FALSE && error == -EINPROGRESS)
579 static void set_apn(struct connman_network *network)
581 const char *apn, *path;
583 apn = connman_network_get_string(network, "Cellular.APN");
587 path = connman_network_get_string(network, "Path");
591 DBG("path %s, apn %s", path, apn);
593 set_property(path, OFONO_PRI_CONTEXT_INTERFACE,
594 "AccessPointName", DBUS_TYPE_STRING, &apn,
598 static int network_connect(struct connman_network *network)
600 if (connman_network_get_index(network) >= 0)
603 return set_network_active(network, TRUE);
606 static int network_disconnect(struct connman_network *network)
608 if (connman_network_get_index(network) < 0)
611 return set_network_active(network, FALSE);
614 static void network_remove(struct connman_network *network)
618 DBG("network %p", network);
620 reg_watch = GPOINTER_TO_UINT(connman_network_get_data(network));
621 g_dbus_remove_watch(connection, reg_watch);
624 static int network_setup(struct connman_network *network, const char *key)
628 if (g_strcmp0(key, "Cellular.APN") == 0)
634 static struct connman_network_driver network_driver = {
636 .type = CONNMAN_NETWORK_TYPE_CELLULAR,
637 .probe = network_probe,
638 .remove = network_remove,
639 .connect = network_connect,
640 .disconnect = network_disconnect,
641 .setup = network_setup,
644 static void add_network(struct connman_device *device, const char *path)
646 struct connman_network *network;
651 DBG("device %p path %s", device, path);
653 network = connman_device_get_network(device, path);
657 ident = get_ident(path);
659 network = connman_network_create(ident,
660 CONNMAN_NETWORK_TYPE_CELLULAR);
666 connman_network_set_string(network, "Path", path);
667 connman_network_set_available(network, TRUE);
668 connman_network_set_index(network, -1);
670 mcc = connman_device_get_string(device, "MCC");
672 connman_network_set_string(network, "Cellular.MCC", mcc);
674 mnc = connman_device_get_string(device, "MNC");
676 connman_network_set_string(network, "Cellular.MNC", mnc);
678 connman_device_add_network(device, network);
681 static void add_networks(struct connman_device *device, DBusMessageIter *array)
683 DBusMessageIter entry;
687 dbus_message_iter_recurse(array, &entry);
689 while (dbus_message_iter_get_arg_type(&entry) ==
690 DBUS_TYPE_OBJECT_PATH) {
693 dbus_message_iter_get_basic(&entry, &path);
695 add_network(device, path);
697 dbus_message_iter_next(&entry);
701 static void create_context_reply(DBusPendingCall *call, void *user_data)
708 dbus_error_init(&error);
710 reply = dbus_pending_call_steal_reply(call);
712 if (dbus_set_error_from_message(&error, reply)) {
713 connman_error("%s", error.message);
714 dbus_error_free(&error);
717 dbus_message_unref(reply);
719 dbus_pending_call_unref(call);
722 static void add_default_context(DBusMessageIter *array,
723 const char *path, const char *name, const char *type)
725 DBusMessageIter entry;
726 DBusMessage *message;
727 DBusPendingCall *call;
734 dbus_message_iter_recurse(array, &entry);
736 if (dbus_message_iter_get_arg_type(&entry) == DBUS_TYPE_OBJECT_PATH)
739 DBG("path %s, name %s, type %s", path, name, type);
741 message = dbus_message_new_method_call(OFONO_SERVICE, path,
742 OFONO_GPRS_INTERFACE, CREATE_CONTEXT);
746 dbus_message_set_auto_start(message, FALSE);
748 dbus_message_append_args(message, DBUS_TYPE_STRING,
749 &name, DBUS_TYPE_STRING,
750 &type, DBUS_TYPE_INVALID);
752 if (dbus_connection_send_with_reply(connection, message,
753 &call, TIMEOUT) == FALSE) {
754 connman_error("Failed to create default context");
755 dbus_message_unref(message);
760 connman_error("D-Bus connection not available");
761 dbus_message_unref(message);
765 dbus_pending_call_set_notify(call, create_context_reply, NULL, NULL);
767 dbus_message_unref(message);
770 static void check_networks_reply(DBusPendingCall *call, void *user_data)
772 struct connman_device *device = user_data;
774 DBusMessageIter array, dict, contexts;
775 dbus_bool_t attached;
777 DBG("device %p", device);
779 reply = dbus_pending_call_steal_reply(call);
781 if (dbus_message_iter_init(reply, &array) == FALSE)
784 if (dbus_message_iter_get_arg_type(&array) != DBUS_TYPE_ARRAY)
787 dbus_message_iter_recurse(&array, &dict);
789 while (dbus_message_iter_get_arg_type(&dict) == DBUS_TYPE_DICT_ENTRY) {
790 DBusMessageIter entry, value;
793 dbus_message_iter_recurse(&dict, &entry);
794 dbus_message_iter_get_basic(&entry, &key);
796 dbus_message_iter_next(&entry);
797 dbus_message_iter_recurse(&entry, &value);
801 if (g_str_equal(key, "Attached") == TRUE) {
802 dbus_message_iter_get_basic(&value, &attached);
803 DBG("Attached %d", attached);
804 } else if (g_str_equal(key, "PrimaryContexts") == TRUE) {
807 path = connman_device_get_string(device, "Path");
809 add_default_context(&contexts, path,
810 CONTEXT_NAME, CONTEXT_TYPE);
811 } else if (g_str_equal(key, "Powered") == TRUE) {
814 dbus_message_iter_get_basic(&value, &powered);
816 connman_device_set_powered(device, powered);
819 dbus_message_iter_next(&dict);
822 if (attached == TRUE)
823 add_networks(device, &contexts);
826 dbus_message_unref(reply);
828 dbus_pending_call_unref(call);
831 static void check_networks(struct modem_data *modem)
833 DBusMessage *message;
834 DBusPendingCall *call;
835 struct connman_device *device;
837 DBG("modem %p", modem);
842 device = modem->device;
846 message = dbus_message_new_method_call(OFONO_SERVICE, modem->path,
847 OFONO_GPRS_INTERFACE, GET_PROPERTIES);
851 dbus_message_set_auto_start(message, FALSE);
853 if (dbus_connection_send_with_reply(connection, message,
854 &call, TIMEOUT) == FALSE) {
855 connman_error("Failed to get ofono GPRS");
860 connman_error("D-Bus connection not available");
864 dbus_pending_call_set_notify(call, check_networks_reply,
865 (void *)device, NULL);
868 dbus_message_unref(message);
871 static void add_device(const char *path, const char *imsi,
872 const char *mcc, const char *mnc)
874 struct modem_data *modem;
875 struct connman_device *device;
877 DBG("path %s imsi %s", path, imsi);
885 modem = g_hash_table_lookup(modem_hash, path);
889 device = connman_device_create(imsi, CONNMAN_DEVICE_TYPE_CELLULAR);
893 connman_device_set_ident(device, imsi);
895 connman_device_set_mode(device, CONNMAN_DEVICE_MODE_NETWORK_MULTIPLE);
897 connman_device_set_string(device, "Path", path);
899 connman_device_set_string(device, "MCC", mcc);
901 connman_device_set_string(device, "MNC", mnc);
903 if (connman_device_register(device) < 0) {
904 connman_device_unref(device);
908 modem->device = device;
910 check_networks(modem);
913 static void sim_properties_reply(DBusPendingCall *call, void *user_data)
915 const char *path = user_data;
919 /* If MobileNetworkCodeLength is not provided, mnc_length is 0 */
920 unsigned char mnc_length = 0;
922 DBusMessageIter array, dict;
924 DBG("path %s", path);
926 reply = dbus_pending_call_steal_reply(call);
928 if (dbus_message_iter_init(reply, &array) == FALSE)
931 if (dbus_message_iter_get_arg_type(&array) != DBUS_TYPE_ARRAY)
934 dbus_message_iter_recurse(&array, &dict);
936 while (dbus_message_iter_get_arg_type(&dict) == DBUS_TYPE_DICT_ENTRY) {
937 DBusMessageIter entry, value;
940 dbus_message_iter_recurse(&dict, &entry);
941 dbus_message_iter_get_basic(&entry, &key);
943 dbus_message_iter_next(&entry);
944 dbus_message_iter_recurse(&entry, &value);
946 if (g_str_equal(key, "SubscriberIdentity") == TRUE)
947 dbus_message_iter_get_basic(&value, &imsi);
949 * 'MobileNetworkCodeLength' is deprecated since version 0.20, but
950 * keep it here for backward compatibility reasons.
952 else if (g_str_equal(key, "MobileNetworkCodeLength") == TRUE)
953 dbus_message_iter_get_basic(&value,
954 (void *) &mnc_length);
955 else if (g_str_equal(key, "MobileCountryCode") == TRUE)
956 dbus_message_iter_get_basic(&value,
958 else if (g_str_equal(key, "MobileNetworkCode") == TRUE)
959 dbus_message_iter_get_basic(&value,
962 dbus_message_iter_next(&dict);
965 if (mnc_length == 2 || mnc_length == 3) {
966 mcc = g_strndup(imsi, 3);
967 mnc = g_strndup(imsi + 3, mnc_length);
970 add_device(path, imsi, mcc, mnc);
972 if (mnc_length == 2 || mnc_length == 3) {
978 dbus_message_unref(reply);
980 dbus_pending_call_unref(call);
983 static void get_imsi(const char *path)
985 DBusMessage *message;
986 DBusPendingCall *call;
988 DBG("path %s", path);
990 message = dbus_message_new_method_call(OFONO_SERVICE, path,
991 OFONO_SIM_INTERFACE, GET_PROPERTIES);
995 dbus_message_set_auto_start(message, FALSE);
997 if (dbus_connection_send_with_reply(connection, message,
998 &call, TIMEOUT) == FALSE) {
999 connman_error("Failed to get ofono modem sim");
1004 connman_error("D-Bus connection not available");
1008 dbus_pending_call_set_notify(call, sim_properties_reply,
1009 (void *)path, NULL);
1012 dbus_message_unref(message);
1015 static int modem_change_powered(const char *path, dbus_bool_t powered)
1017 DBG("path %s powered %d", path, powered);
1019 return set_property(path, OFONO_MODEM_INTERFACE, "Powered",
1020 DBUS_TYPE_BOOLEAN, &powered,
1024 static struct modem_data *add_modem(const char *path)
1026 struct modem_data *modem;
1031 modem = g_hash_table_lookup(modem_hash, path);
1032 if (modem != NULL) {
1033 modem->available = TRUE;
1038 modem = g_try_new0(struct modem_data, 1);
1042 modem->path = g_strdup(path);
1043 modem->device = NULL;
1044 modem->available = TRUE;
1046 g_hash_table_insert(modem_hash, g_strdup(path), modem);
1051 static gboolean modem_has_gprs(DBusMessageIter *array)
1053 DBusMessageIter entry;
1055 dbus_message_iter_recurse(array, &entry);
1057 while (dbus_message_iter_get_arg_type(&entry) == DBUS_TYPE_STRING) {
1058 const char *interface;
1060 dbus_message_iter_get_basic(&entry, &interface);
1062 if (g_strcmp0(OFONO_GPRS_INTERFACE, interface) == 0)
1065 dbus_message_iter_next(&entry);
1071 static void modem_properties_reply(DBusPendingCall *call, void *user_data)
1074 DBusMessageIter array, dict;
1075 const char *path = user_data;
1077 DBG("path %s", path);
1079 reply = dbus_pending_call_steal_reply(call);
1081 if (dbus_message_iter_init(reply, &array) == FALSE)
1084 if (dbus_message_iter_get_arg_type(&array) != DBUS_TYPE_ARRAY)
1087 dbus_message_iter_recurse(&array, &dict);
1089 while (dbus_message_iter_get_arg_type(&dict) == DBUS_TYPE_DICT_ENTRY) {
1090 DBusMessageIter entry, value;
1092 dbus_bool_t powered;
1094 dbus_message_iter_recurse(&dict, &entry);
1095 dbus_message_iter_get_basic(&entry, &key);
1097 dbus_message_iter_next(&entry);
1098 dbus_message_iter_recurse(&entry, &value);
1100 if (g_str_equal(key, "Powered") == TRUE) {
1101 dbus_message_iter_get_basic(&value, &powered);
1103 if (powered == FALSE) {
1104 modem_change_powered(path, TRUE);
1107 } else if (g_str_equal(key, "Interfaces") == TRUE) {
1108 if (modem_has_gprs(&value) == TRUE)
1112 dbus_message_iter_next(&dict);
1116 dbus_message_unref(reply);
1118 dbus_pending_call_unref(call);
1121 static void get_modem_properties(struct modem_data *modem)
1123 DBusMessage *message;
1124 DBusPendingCall *call;
1126 DBG("path %s", modem->path);
1128 if (modem->path == NULL)
1131 message = dbus_message_new_method_call(OFONO_SERVICE, modem->path,
1132 OFONO_MODEM_INTERFACE, GET_PROPERTIES);
1133 if (message == NULL)
1136 dbus_message_set_auto_start(message, FALSE);
1138 if (dbus_connection_send_with_reply(connection, message,
1139 &call, TIMEOUT) == FALSE) {
1140 connman_error("Failed to get ofono modem");
1145 connman_error("D-Bus connection not available");
1149 dbus_pending_call_set_notify(call, modem_properties_reply,
1150 (void *)modem->path, NULL);
1153 dbus_message_unref(message);
1156 static void mask_unavailable(gpointer key, gpointer value, gpointer user_data)
1158 struct modem_data *modem = value;
1160 modem->available = FALSE;
1163 static void modems_set_unavailable()
1165 g_hash_table_foreach(modem_hash, mask_unavailable, NULL);
1168 static void cleanup_modem(gpointer key, gpointer value, gpointer user_data)
1170 struct modem_data *modem = value;
1172 if (modem->available == FALSE)
1173 g_hash_table_remove(modem_hash, key);
1176 static void cleanup_modems()
1178 g_hash_table_foreach(modem_hash, cleanup_modem, NULL);
1181 static void update_modems(DBusMessageIter *array)
1183 DBusMessageIter entry;
1185 dbus_message_iter_recurse(array, &entry);
1187 modems_set_unavailable();
1189 while (dbus_message_iter_get_arg_type(&entry) ==
1190 DBUS_TYPE_OBJECT_PATH) {
1192 struct modem_data *modem;
1194 dbus_message_iter_get_basic(&entry, &path);
1196 modem = add_modem(path);
1198 get_modem_properties(modem);
1200 dbus_message_iter_next(&entry);
1206 static void manager_properties_reply(DBusPendingCall *call, void *user_data)
1209 DBusMessageIter array, dict;
1213 reply = dbus_pending_call_steal_reply(call);
1215 if (dbus_message_iter_init(reply, &array) == FALSE)
1218 if (dbus_message_iter_get_arg_type(&array) != DBUS_TYPE_ARRAY)
1221 dbus_message_iter_recurse(&array, &dict);
1223 while (dbus_message_iter_get_arg_type(&dict) == DBUS_TYPE_DICT_ENTRY) {
1224 DBusMessageIter entry, value;
1227 dbus_message_iter_recurse(&dict, &entry);
1228 dbus_message_iter_get_basic(&entry, &key);
1230 dbus_message_iter_next(&entry);
1231 dbus_message_iter_recurse(&entry, &value);
1233 if (g_str_equal(key, "Modems") == TRUE) {
1234 update_modems(&value);
1238 dbus_message_iter_next(&dict);
1242 dbus_message_unref(reply);
1244 dbus_pending_call_unref(call);
1247 static void modem_remove_device(struct modem_data *modem)
1249 if (modem->device == NULL)
1252 connman_device_unregister(modem->device);
1253 connman_device_unref(modem->device);
1255 modem->device = NULL;
1258 static void remove_modem(gpointer data)
1260 struct modem_data *modem = data;
1262 g_free(modem->path);
1264 modem_remove_device(modem);
1269 static void ofono_connect(DBusConnection *connection, void *user_data)
1271 DBusMessage *message;
1272 DBusPendingCall *call;
1274 DBG("connection %p", connection);
1276 modem_hash = g_hash_table_new_full(g_str_hash, g_str_equal,
1277 g_free, remove_modem);
1279 message = dbus_message_new_method_call(OFONO_SERVICE, "/",
1280 OFONO_MANAGER_INTERFACE, GET_PROPERTIES);
1281 if (message == NULL)
1284 dbus_message_set_auto_start(message, FALSE);
1286 if (dbus_connection_send_with_reply(connection, message,
1287 &call, TIMEOUT) == FALSE) {
1288 connman_error("Failed to get ofono modems");
1293 connman_error("D-Bus connection not available");
1297 dbus_pending_call_set_notify(call, manager_properties_reply,
1301 dbus_message_unref(message);
1305 static void ofono_disconnect(DBusConnection *connection, void *user_data)
1307 DBG("connection %p", connection);
1309 if (modem_hash == NULL)
1312 g_hash_table_destroy(modem_hash);
1317 static gboolean modem_changed(DBusConnection *connection, DBusMessage *message,
1320 const char *path = dbus_message_get_path(message);
1321 struct modem_data *modem;
1322 DBusMessageIter iter, value;
1325 DBG("path %s", path);
1327 modem = g_hash_table_lookup(modem_hash, path);
1331 if (dbus_message_iter_init(message, &iter) == FALSE)
1334 dbus_message_iter_get_basic(&iter, &key);
1336 dbus_message_iter_next(&iter);
1337 dbus_message_iter_recurse(&iter, &value);
1339 if (g_str_equal(key, "Powered") == TRUE) {
1340 dbus_bool_t powered;
1342 dbus_message_iter_get_basic(&value, &powered);
1343 if (powered == TRUE)
1346 modem_remove_device(modem);
1347 } else if (g_str_equal(key, "Interfaces") == TRUE) {
1348 if (modem_has_gprs(&value) == TRUE) {
1349 if (modem->device == NULL)
1350 get_imsi(modem->path);
1351 } else if (modem->device != NULL)
1352 modem_remove_device(modem);
1358 static gboolean gprs_changed(DBusConnection *connection, DBusMessage *message,
1361 const char *path = dbus_message_get_path(message);
1362 struct modem_data *modem;
1363 DBusMessageIter iter, value;
1366 DBG("path %s", path);
1368 modem = g_hash_table_lookup(modem_hash, path);
1372 if (dbus_message_iter_init(message, &iter) == FALSE)
1375 dbus_message_iter_get_basic(&iter, &key);
1377 dbus_message_iter_next(&iter);
1378 dbus_message_iter_recurse(&iter, &value);
1380 if (g_str_equal(key, "Attached") == TRUE) {
1381 dbus_bool_t attached;
1383 dbus_message_iter_get_basic(&value, &attached);
1385 DBG("Attached %d", attached);
1387 if (attached == TRUE)
1388 check_networks(modem);
1389 else if (modem->device != NULL)
1390 connman_device_remove_all_networks(modem->device);
1392 } else if (g_str_equal(key, "PrimaryContexts") == TRUE) {
1393 check_networks(modem);
1394 } else if (g_str_equal(key, "Powered") == TRUE) {
1395 dbus_bool_t powered;
1397 if (modem->device == NULL)
1400 dbus_message_iter_get_basic(&value, &powered);
1401 connman_device_set_powered(modem->device, powered);
1407 static gboolean manager_changed(DBusConnection *connection,
1408 DBusMessage *message, void *user_data)
1410 const char *path = dbus_message_get_path(message);
1411 DBusMessageIter iter, value;
1414 DBG("path %s", path);
1416 if (dbus_message_iter_init(message, &iter) == FALSE)
1419 dbus_message_iter_get_basic(&iter, &key);
1421 dbus_message_iter_next(&iter);
1422 dbus_message_iter_recurse(&iter, &value);
1424 if (g_str_equal(key, "Modems") == TRUE)
1425 update_modems(&value);
1430 static void get_dns(DBusMessageIter *array, struct connman_element *parent)
1432 DBusMessageIter entry;
1433 gchar *nameserver = NULL, *nameserver_old = NULL;
1437 dbus_message_iter_recurse(array, &entry);
1439 while (dbus_message_iter_get_arg_type(&entry) == DBUS_TYPE_STRING) {
1442 dbus_message_iter_get_basic(&entry, &dns);
1446 if (nameserver == NULL) {
1448 nameserver = g_strdup(dns);
1451 nameserver_old = nameserver;
1452 nameserver = g_strdup_printf("%s %s",
1453 nameserver_old, dns);
1454 g_free(nameserver_old);
1457 dbus_message_iter_next(&entry);
1460 parent->ipv4.nameserver = nameserver;
1463 static void update_settings(DBusMessageIter *array,
1464 struct connman_element *parent)
1466 DBusMessageIter dict;
1467 const char *interface = NULL;
1471 if (dbus_message_iter_get_arg_type(array) != DBUS_TYPE_ARRAY)
1474 dbus_message_iter_recurse(array, &dict);
1476 while (dbus_message_iter_get_arg_type(&dict) == DBUS_TYPE_DICT_ENTRY) {
1477 DBusMessageIter entry, value;
1480 dbus_message_iter_recurse(&dict, &entry);
1481 dbus_message_iter_get_basic(&entry, &key);
1483 dbus_message_iter_next(&entry);
1484 dbus_message_iter_recurse(&entry, &value);
1486 if (g_str_equal(key, "Interface") == TRUE) {
1489 dbus_message_iter_get_basic(&value, &interface);
1491 DBG("interface %s", interface);
1493 index = connman_inet_ifindex(interface);
1495 connman_network_set_index(
1496 pending_network, index);
1498 connman_error("Can not find interface %s",
1502 } else if (g_str_equal(key, "Method") == TRUE) {
1505 dbus_message_iter_get_basic(&value, &method);
1506 if (g_strcmp0(method, "static") == 0) {
1508 parent->ipv4.method =
1509 CONNMAN_IPCONFIG_METHOD_FIXED;
1510 } else if (g_strcmp0(method, "dhcp") == 0) {
1512 parent->ipv4.method =
1513 CONNMAN_IPCONFIG_METHOD_DHCP;
1516 } else if (g_str_equal(key, "Address") == TRUE) {
1517 const char *address;
1519 dbus_message_iter_get_basic(&value, &address);
1521 DBG("address %s", address);
1523 parent->ipv4.address = g_strdup(address);
1524 } else if (g_str_equal(key, "Netmask") == TRUE) {
1525 const char *netmask;
1527 dbus_message_iter_get_basic(&value, &netmask);
1529 DBG("netmask %s", netmask);
1531 parent->ipv4.netmask = g_strdup(netmask);
1532 } else if (g_str_equal(key, "DomainNameServers") == TRUE) {
1534 get_dns(&value, parent);
1535 } else if (g_str_equal(key, "Gateway") == TRUE) {
1536 const char *gateway;
1538 dbus_message_iter_get_basic(&value, &gateway);
1540 DBG("gateway %s", gateway);
1542 parent->ipv4.gateway = g_strdup(gateway);
1545 dbus_message_iter_next(&dict);
1548 /* deactive, oFono send NULL inteface before deactive signal */
1549 if (interface == NULL)
1550 connman_network_set_index(pending_network, -1);
1553 static void cleanup_ipconfig(struct connman_element *parent)
1555 g_free(parent->ipv4.address);
1556 parent->ipv4.address = NULL;
1558 g_free(parent->ipv4.netmask);
1559 parent->ipv4.netmask = NULL;
1561 g_free(parent->ipv4.nameserver);
1562 parent->ipv4.nameserver = NULL;
1564 g_free(parent->ipv4.gateway);
1565 parent->ipv4.gateway = NULL;
1567 parent->ipv4.method = CONNMAN_IPCONFIG_METHOD_UNKNOWN;
1570 static int static_network_set_connected(
1571 struct connman_network *pending_network,
1572 struct connman_element *parent,
1573 connman_bool_t connected)
1575 if (connected == FALSE)
1576 cleanup_ipconfig(parent);
1578 connman_network_set_connected(pending_network, connected);
1583 static gboolean pri_context_changed(DBusConnection *connection,
1584 DBusMessage *message, void *user_data)
1586 const char *path = dbus_message_get_path(message);
1587 struct connman_element *parent;
1588 const char *pending_path;
1589 DBusMessageIter iter, value;
1592 DBG("pending_network %p, path %s", pending_network, path);
1594 if (pending_network == NULL)
1597 pending_path = connman_network_get_string(pending_network, "Path");
1598 if (g_strcmp0(pending_path, path) != 0)
1601 parent = connman_network_get_element(pending_network);
1603 if (dbus_message_iter_init(message, &iter) == FALSE)
1606 dbus_message_iter_get_basic(&iter, &key);
1608 dbus_message_iter_next(&iter);
1609 dbus_message_iter_recurse(&iter, &value);
1611 if (g_str_equal(key, "Settings") == TRUE) {
1613 update_settings(&value, parent);
1614 } else if (g_str_equal(key, "Active") == TRUE) {
1617 dbus_message_iter_get_basic(&value, &active);
1619 switch (parent->ipv4.method) {
1620 case CONNMAN_IPCONFIG_METHOD_UNKNOWN:
1621 case CONNMAN_IPCONFIG_METHOD_OFF:
1622 case CONNMAN_IPCONFIG_METHOD_MANUAL:
1624 case CONNMAN_IPCONFIG_METHOD_FIXED:
1625 connman_network_set_method(pending_network,
1626 CONNMAN_IPCONFIG_METHOD_FIXED);
1628 if (static_network_set_connected(
1629 pending_network, parent, active) < 0)
1630 set_network_active(pending_network, FALSE);
1632 case CONNMAN_IPCONFIG_METHOD_DHCP:
1633 connman_network_set_method(pending_network,
1634 CONNMAN_IPCONFIG_METHOD_DHCP);
1635 connman_network_set_connected(pending_network, active);
1639 pending_network = NULL;
1646 static guint gprs_watch;
1647 static guint modem_watch;
1648 static guint manager_watch;
1649 static guint context_watch;
1651 static int ofono_init(void)
1655 connection = connman_dbus_get_connection();
1656 if (connection == NULL)
1659 watch = g_dbus_add_service_watch(connection, OFONO_SERVICE,
1660 ofono_connect, ofono_disconnect, NULL, NULL);
1662 gprs_watch = g_dbus_add_signal_watch(connection, NULL, NULL,
1663 OFONO_GPRS_INTERFACE,
1668 modem_watch = g_dbus_add_signal_watch(connection, NULL, NULL,
1669 OFONO_MODEM_INTERFACE,
1674 manager_watch = g_dbus_add_signal_watch(connection, NULL, NULL,
1675 OFONO_MANAGER_INTERFACE,
1680 context_watch = g_dbus_add_signal_watch(connection, NULL, NULL,
1681 OFONO_PRI_CONTEXT_INTERFACE,
1683 pri_context_changed,
1686 if (watch == 0 || gprs_watch == 0 || modem_watch == 0 ||
1687 manager_watch == 0 || context_watch == 0) {
1692 err = connman_network_driver_register(&network_driver);
1696 err = connman_device_driver_register(&modem_driver);
1698 connman_network_driver_unregister(&network_driver);
1705 g_dbus_remove_watch(connection, watch);
1706 g_dbus_remove_watch(connection, gprs_watch);
1707 g_dbus_remove_watch(connection, modem_watch);
1708 g_dbus_remove_watch(connection, manager_watch);
1709 g_dbus_remove_watch(connection, context_watch);
1711 dbus_connection_unref(connection);
1716 static void ofono_exit(void)
1718 g_dbus_remove_watch(connection, watch);
1719 g_dbus_remove_watch(connection, gprs_watch);
1720 g_dbus_remove_watch(connection, modem_watch);
1721 g_dbus_remove_watch(connection, manager_watch);
1722 g_dbus_remove_watch(connection, context_watch);
1724 ofono_disconnect(connection, NULL);
1726 connman_device_driver_unregister(&modem_driver);
1727 connman_network_driver_unregister(&network_driver);
1729 dbus_connection_unref(connection);
1732 CONNMAN_PLUGIN_DEFINE(ofono, "oFono telephony plugin", VERSION,
1733 CONNMAN_PLUGIN_PRIORITY_DEFAULT, ofono_init, ofono_exit)