summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDenis Kenzior <denkenz@gmail.com>2017-11-01 10:35:13 -0500
committerDenis Kenzior <denkenz@gmail.com>2017-11-08 21:02:22 -0600
commit2ec8ed30bf76690b916461a14422b7483f8e9be9 (patch)
treefedea6977c04278536ed2e49065455edb4d63f0a
parent689ebc808f8642fa5dc5035a74cbe18b797a81a7 (diff)
downloadofono-2ec8ed30bf76690b916461a14422b7483f8e9be9.tar.gz
mbimmodem: Add netreg driver
-rw-r--r--Makefile.am3
-rw-r--r--drivers/mbimmodem/mbimmodem.c2
-rw-r--r--drivers/mbimmodem/mbimmodem.h4
-rw-r--r--drivers/mbimmodem/network-registration.c439
4 files changed, 447 insertions, 1 deletions
diff --git a/Makefile.am b/Makefile.am
index 1e559a7e..0c4eebbe 100644
--- a/Makefile.am
+++ b/Makefile.am
@@ -621,7 +621,8 @@ builtin_sources += $(mbim_sources) \
drivers/mbimmodem/mbimmodem.h \
drivers/mbimmodem/mbimmodem.c \
drivers/mbimmodem/devinfo.c \
- drivers/mbimmodem/sim.c
+ drivers/mbimmodem/sim.c \
+ drivers/mbimmodem/network-registration.c
builtin_modules += mbim
builtin_sources += plugins/mbim.c
diff --git a/drivers/mbimmodem/mbimmodem.c b/drivers/mbimmodem/mbimmodem.c
index d2d09780..a9aaae73 100644
--- a/drivers/mbimmodem/mbimmodem.c
+++ b/drivers/mbimmodem/mbimmodem.c
@@ -32,11 +32,13 @@ static int mbimmodem_init(void)
{
mbim_devinfo_init();
mbim_sim_init();
+ mbim_netreg_init();
return 0;
}
static void mbimmodem_exit(void)
{
+ mbim_netreg_exit();
mbim_sim_exit();
mbim_devinfo_exit();
}
diff --git a/drivers/mbimmodem/mbimmodem.h b/drivers/mbimmodem/mbimmodem.h
index 2f0b365c..de979c88 100644
--- a/drivers/mbimmodem/mbimmodem.h
+++ b/drivers/mbimmodem/mbimmodem.h
@@ -23,6 +23,7 @@
enum MBIM_GROUP {
SIM_GROUP = 1,
+ NETREG_GROUP = 2,
};
extern void mbim_devinfo_init(void);
@@ -30,3 +31,6 @@ extern void mbim_devinfo_exit(void);
extern void mbim_sim_init(void);
extern void mbim_sim_exit(void);
+
+extern void mbim_netreg_init(void);
+extern void mbim_netreg_exit(void);
diff --git a/drivers/mbimmodem/network-registration.c b/drivers/mbimmodem/network-registration.c
new file mode 100644
index 00000000..59407c4a
--- /dev/null
+++ b/drivers/mbimmodem/network-registration.c
@@ -0,0 +1,439 @@
+/*
+ *
+ * oFono - Open Source Telephony
+ *
+ * Copyright (C) 2017 Intel Corporation. All rights reserved.
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 as
+ * published by the Free Software Foundation.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
+ *
+ */
+
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+#endif
+
+#define _GNU_SOURCE
+#include <stdbool.h>
+#include <stdint.h>
+#include <stdlib.h>
+#include <errno.h>
+
+#include <ofono/log.h>
+#include <ofono/modem.h>
+#include <ofono/netreg.h>
+
+#include "src/common.h"
+
+#include "drivers/mbimmodem/mbim.h"
+#include "drivers/mbimmodem/mbim-message.h"
+#include "drivers/mbimmodem/mbimmodem.h"
+
+struct netreg_data {
+ struct mbim_device *device;
+ struct l_idle *delayed_register;
+};
+
+static inline int available_data_classes_to_tech(uint32_t n)
+{
+ if (n & MBIM_DATA_CLASS_LTE)
+ return ACCESS_TECHNOLOGY_EUTRAN;
+
+ if (n & (MBIM_DATA_CLASS_HSUPA | MBIM_DATA_CLASS_HSDPA))
+ return ACCESS_TECHNOLOGY_UTRAN_HSDPA_HSUPA;
+
+ if (n & MBIM_DATA_CLASS_HSUPA)
+ return ACCESS_TECHNOLOGY_UTRAN_HSUPA;
+
+ if (n & MBIM_DATA_CLASS_HSDPA)
+ return ACCESS_TECHNOLOGY_UTRAN_HSDPA;
+
+ if (n & MBIM_DATA_CLASS_UMTS)
+ return ACCESS_TECHNOLOGY_UTRAN;
+
+ if (n & MBIM_DATA_CLASS_EDGE)
+ return ACCESS_TECHNOLOGY_GSM_EGPRS;
+
+ if (n & MBIM_DATA_CLASS_GPRS)
+ return ACCESS_TECHNOLOGY_GSM;
+
+ return -1;
+}
+
+static inline int register_state_to_status(uint32_t register_state)
+{
+ switch (register_state) {
+ case 0: /* MBIMRegisterStateUnknown */
+ return NETWORK_REGISTRATION_STATUS_UNKNOWN;
+ case 1: /* MBIMRegisterStateDeregistered */
+ return NETWORK_REGISTRATION_STATUS_NOT_REGISTERED;
+ case 2: /* MBIMRegisterStateSearching */
+ return NETWORK_REGISTRATION_STATUS_SEARCHING;
+ case 3: /* MBIMRegisterStateHome */
+ return NETWORK_REGISTRATION_STATUS_REGISTERED;
+ case 4: /* MBIMRegisterStateRoaming */
+ case 5: /* MBIMRegisterStatePartner */
+ return NETWORK_REGISTRATION_STATUS_ROAMING;
+ case 6: /* MBIMRegisterStateDenied */
+ return NETWORK_REGISTRATION_STATUS_DENIED;
+ }
+
+ return NETWORK_REGISTRATION_STATUS_UNKNOWN;
+}
+
+static void mbim_register_state_changed(struct mbim_message *message,
+ void *user)
+{
+ struct ofono_netreg *netreg = user;
+ uint32_t nw_error;
+ uint32_t register_state;
+ uint32_t register_mode;
+ uint32_t available_data_classes;
+ int status;
+ int tech;
+
+ DBG("");
+
+ if (!mbim_message_get_arguments(message, "uuuu",
+ &nw_error, &register_state,
+ &register_mode,
+ &available_data_classes))
+ return;
+
+ DBG("NwError: %u, RegisterMode: %u", nw_error, register_mode);
+
+ status = register_state_to_status(register_state);
+ tech = available_data_classes_to_tech(available_data_classes);
+
+ ofono_netreg_status_notify(netreg, status, -1, -1, tech);
+}
+
+static void mbim_registration_status_cb(struct mbim_message *message,
+ void *user)
+{
+ struct cb_data *cbd = user;
+ ofono_netreg_status_cb_t cb = cbd->cb;
+ uint32_t dummy;
+ uint32_t register_state;
+ uint32_t available_data_classes;
+ int status;
+ int tech;
+
+ DBG("");
+
+ if (mbim_message_get_error(message) != 0)
+ goto error;
+
+ if (!mbim_message_get_arguments(message, "uuuu",
+ &dummy, &register_state,
+ &dummy,
+ &available_data_classes))
+ goto error;
+
+ status = register_state_to_status(register_state);
+ tech = available_data_classes_to_tech(available_data_classes);
+
+ CALLBACK_WITH_SUCCESS(cb, status, -1, -1, tech, cbd->data);
+ return;
+error:
+ CALLBACK_WITH_FAILURE(cb, -1, -1, -1, -1, cbd->data);
+}
+
+static void mbim_registration_status(struct ofono_netreg *netreg,
+ ofono_netreg_status_cb_t cb,
+ void *data)
+{
+ struct netreg_data *nd = ofono_netreg_get_data(netreg);
+ struct cb_data *cbd = cb_data_new(cb, data);
+ struct mbim_message *message;
+
+ message = mbim_message_new(mbim_uuid_basic_connect,
+ MBIM_CID_REGISTER_STATE,
+ MBIM_COMMAND_TYPE_QUERY);
+ mbim_message_set_arguments(message, "");
+
+ if (mbim_device_send(nd->device, NETREG_GROUP, message,
+ mbim_registration_status_cb, cbd, l_free) > 0)
+ return;
+
+ l_free(cbd);
+ mbim_message_unref(message);
+ CALLBACK_WITH_FAILURE(cb, -1, -1, -1, -1, data);
+}
+
+static void mbim_current_operator_cb(struct mbim_message *message, void *user)
+{
+ struct cb_data *cbd = user;
+ ofono_netreg_operator_cb_t cb = cbd->cb;
+ struct ofono_network_operator op;
+ uint32_t dummy;
+ uint32_t register_state;
+ uint32_t available_data_classes;
+ L_AUTO_FREE_VAR(char *, provider_id) = NULL;
+ L_AUTO_FREE_VAR(char *, provider_name) = NULL;
+ L_AUTO_FREE_VAR(char *, roaming_text) = NULL;
+
+ DBG("");
+
+ if (mbim_message_get_error(message) != 0)
+ goto error;
+
+ if (!mbim_message_get_arguments(message, "uuuuusss",
+ &dummy, &register_state, &dummy,
+ &available_data_classes, &dummy,
+ &provider_id, &provider_name,
+ &roaming_text))
+ goto error;
+
+ DBG("provider: %s(%s)", provider_name, provider_id);
+
+ /* If MBIMRegisterStateRoaming or MBIMRegisterStatePartner */
+ if (register_state == 4 || register_state == 5)
+ DBG("roaming text: %s", roaming_text);
+
+ strncpy(op.name, provider_name, OFONO_MAX_OPERATOR_NAME_LENGTH);
+ op.name[OFONO_MAX_OPERATOR_NAME_LENGTH] = '\0';
+
+ strncpy(op.mcc, provider_id, OFONO_MAX_MCC_LENGTH);
+ op.mcc[OFONO_MAX_MCC_LENGTH] = '\0';
+
+ strncpy(op.mnc, provider_id + OFONO_MAX_MCC_LENGTH,
+ OFONO_MAX_MNC_LENGTH);
+ op.mnc[OFONO_MAX_MNC_LENGTH] = '\0';
+
+ /* Set to current */
+ op.status = 2;
+ op.tech = available_data_classes_to_tech(available_data_classes);
+
+ CALLBACK_WITH_SUCCESS(cb, &op, cbd->data);
+ return;
+error:
+ CALLBACK_WITH_FAILURE(cb, NULL, cbd->data);
+}
+
+static void mbim_current_operator(struct ofono_netreg *netreg,
+ ofono_netreg_operator_cb_t cb, void *data)
+{
+ struct netreg_data *nd = ofono_netreg_get_data(netreg);
+ struct cb_data *cbd = cb_data_new(cb, data);
+ struct mbim_message *message;
+
+ message = mbim_message_new(mbim_uuid_basic_connect,
+ MBIM_CID_REGISTER_STATE,
+ MBIM_COMMAND_TYPE_QUERY);
+ mbim_message_set_arguments(message, "");
+
+ if (mbim_device_send(nd->device, NETREG_GROUP, message,
+ mbim_current_operator_cb, cbd, l_free) > 0)
+ return;
+
+ l_free(cbd);
+ mbim_message_unref(message);
+ CALLBACK_WITH_FAILURE(cb, NULL, data);
+}
+
+static void mbim_register_state_set_cb(struct mbim_message *message, void *user)
+{
+ struct cb_data *cbd = user;
+ ofono_netreg_register_cb_t cb = cbd->cb;
+
+ DBG("");
+
+ if (mbim_message_get_error(message) != 0)
+ CALLBACK_WITH_FAILURE(cb, cbd->data);
+ else
+ CALLBACK_WITH_SUCCESS(cb, cbd->data);
+}
+
+static void mbim_register_auto(struct ofono_netreg *netreg,
+ ofono_netreg_register_cb_t cb, void *data)
+{
+ static const uint32_t data_class = MBIM_DATA_CLASS_GPRS |
+ MBIM_DATA_CLASS_EDGE |
+ MBIM_DATA_CLASS_UMTS |
+ MBIM_DATA_CLASS_HSDPA |
+ MBIM_DATA_CLASS_HSUPA |
+ MBIM_DATA_CLASS_LTE;
+ struct netreg_data *nd = ofono_netreg_get_data(netreg);
+ struct cb_data *cbd = cb_data_new(cb, data);
+ struct mbim_message *message;
+
+ message = mbim_message_new(mbim_uuid_basic_connect,
+ MBIM_CID_REGISTER_STATE,
+ MBIM_COMMAND_TYPE_SET);
+ mbim_message_set_arguments(message, "suu", NULL, 0, data_class);
+
+ if (mbim_device_send(nd->device, NETREG_GROUP, message,
+ mbim_register_state_set_cb, cbd, l_free) > 0)
+ return;
+
+ l_free(cbd);
+ mbim_message_unref(message);
+ CALLBACK_WITH_FAILURE(cb, data);
+}
+
+static inline int convert_signal_strength(uint32_t strength)
+{
+ if (strength == 99)
+ return -1;
+
+ return strength * 100 / 31;
+}
+
+static void mbim_signal_state_query_cb(struct mbim_message *message, void *user)
+{
+ struct cb_data *cbd = user;
+ ofono_netreg_strength_cb_t cb = cbd->cb;
+ uint32_t strength;
+
+ DBG("");
+
+ if (mbim_message_get_error(message) != 0)
+ goto error;
+
+ if (!mbim_message_get_arguments(message, "u", &strength))
+ goto error;
+
+ CALLBACK_WITH_SUCCESS(cb, convert_signal_strength(strength), cbd->data);
+ return;
+
+error:
+ CALLBACK_WITH_FAILURE(cb, -1, cbd->data);
+}
+
+static void mbim_signal_strength(struct ofono_netreg *netreg,
+ ofono_netreg_strength_cb_t cb, void *data)
+{
+ struct netreg_data *nd = ofono_netreg_get_data(netreg);
+ struct cb_data *cbd = cb_data_new(cb, data);
+ struct mbim_message *message;
+
+ message = mbim_message_new(mbim_uuid_basic_connect,
+ MBIM_CID_SIGNAL_STATE,
+ MBIM_COMMAND_TYPE_QUERY);
+ mbim_message_set_arguments(message, "");
+
+ if (mbim_device_send(nd->device, NETREG_GROUP, message,
+ mbim_signal_state_query_cb, cbd, l_free) > 0)
+ return;
+
+ l_free(cbd);
+ mbim_message_unref(message);
+ CALLBACK_WITH_FAILURE(cb, -1, data);
+}
+
+static void mbim_signal_state_changed(struct mbim_message *message, void *user)
+{
+ struct ofono_netreg *netreg = user;
+ uint32_t strength;
+ uint32_t error_rate;
+ uint32_t signal_strength_interval;
+ uint32_t rssi_threshold;
+
+ DBG("");
+
+ if (!mbim_message_get_arguments(message, "uuuu",
+ &strength, &error_rate,
+ &signal_strength_interval,
+ &rssi_threshold))
+ return;
+
+ DBG("strength: %u, error_rate: %u", strength, error_rate);
+ DBG("strength interval: %u, rssi_threshold: %u",
+ signal_strength_interval, rssi_threshold);
+
+ ofono_netreg_strength_notify(netreg, convert_signal_strength(strength));
+}
+
+static void delayed_register(struct l_idle *idle, void *user_data)
+{
+ struct ofono_netreg *netreg = user_data;
+ struct netreg_data *nd = ofono_netreg_get_data(netreg);
+
+ DBG("");
+
+ l_idle_remove(idle);
+ nd->delayed_register = NULL;
+
+ if (!mbim_device_register(nd->device, NETREG_GROUP,
+ mbim_uuid_basic_connect,
+ MBIM_CID_SIGNAL_STATE,
+ mbim_signal_state_changed,
+ netreg, NULL))
+ goto error;
+
+ if (!mbim_device_register(nd->device, NETREG_GROUP,
+ mbim_uuid_basic_connect,
+ MBIM_CID_REGISTER_STATE,
+ mbim_register_state_changed,
+ netreg, NULL))
+ goto error;
+
+ ofono_netreg_register(netreg);
+ return;
+
+error:
+ ofono_netreg_remove(netreg);
+}
+
+static int mbim_netreg_probe(struct ofono_netreg *netreg, unsigned int vendor,
+ void *data)
+{
+ struct mbim_device *device = data;
+ struct netreg_data *nd = l_new(struct netreg_data, 1);
+
+ DBG("");
+
+ nd->device = mbim_device_ref(device);
+ nd->delayed_register = l_idle_create(delayed_register, netreg, NULL);
+
+ ofono_netreg_set_data(netreg, nd);
+
+ return 0;
+}
+
+static void mbim_netreg_remove(struct ofono_netreg *netreg)
+{
+ struct netreg_data *nd = ofono_netreg_get_data(netreg);
+
+ DBG("");
+
+ ofono_netreg_set_data(netreg, NULL);
+
+ l_idle_remove(nd->delayed_register);
+ mbim_device_cancel_group(nd->device, NETREG_GROUP);
+ mbim_device_unregister_group(nd->device, NETREG_GROUP);
+ mbim_device_unref(nd->device);
+ nd->device = NULL;
+ l_free(nd);
+}
+
+static struct ofono_netreg_driver driver = {
+ .name = "mbim",
+ .probe = mbim_netreg_probe,
+ .remove = mbim_netreg_remove,
+ .registration_status = mbim_registration_status,
+ .current_operator = mbim_current_operator,
+ .register_auto = mbim_register_auto,
+ .strength = mbim_signal_strength,
+};
+
+void mbim_netreg_init(void)
+{
+ ofono_netreg_driver_register(&driver);
+}
+
+void mbim_netreg_exit(void)
+{
+ ofono_netreg_driver_unregister(&driver);
+}