summaryrefslogtreecommitdiffstats
path: root/drivers/qmimodem/devinfo.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/qmimodem/devinfo.c')
-rw-r--r--drivers/qmimodem/devinfo.c225
1 files changed, 225 insertions, 0 deletions
diff --git a/drivers/qmimodem/devinfo.c b/drivers/qmimodem/devinfo.c
new file mode 100644
index 00000000..51e2fe34
--- /dev/null
+++ b/drivers/qmimodem/devinfo.c
@@ -0,0 +1,225 @@
+/*
+ *
+ * oFono - Open Source Telephony
+ *
+ * Copyright (C) 2011-2012 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
+
+#include <ofono/log.h>
+#include <ofono/modem.h>
+#include <ofono/devinfo.h>
+
+#include "qmi.h"
+#include "dms.h"
+
+#include "qmimodem.h"
+
+struct devinfo_data {
+ struct qmi_service *dms;
+};
+
+static void string_cb(struct qmi_result *result, void *user_data)
+{
+ struct cb_data *cbd = user_data;
+ ofono_devinfo_query_cb_t cb = cbd->cb;
+ char *str;
+
+ DBG("");
+
+ if (qmi_result_set_error(result, NULL)) {
+ CALLBACK_WITH_FAILURE(cb, NULL, cbd->data);
+ return;
+ }
+
+ str = qmi_result_get_string(result, 0x01);
+ if (!str) {
+ CALLBACK_WITH_FAILURE(cb, NULL, cbd->data);
+ return;
+ }
+
+ CALLBACK_WITH_SUCCESS(cb, str, cbd->data);
+
+ qmi_free(str);
+}
+
+static void qmi_query_manufacturer(struct ofono_devinfo *devinfo,
+ ofono_devinfo_query_cb_t cb, void *user_data)
+{
+ struct devinfo_data *data = ofono_devinfo_get_data(devinfo);
+ struct cb_data *cbd = cb_data_new(cb, user_data);
+
+ DBG("");
+
+ if (qmi_service_send(data->dms, QMI_DMS_GET_MANUFACTURER, NULL,
+ string_cb, cbd, g_free) > 0)
+ return;
+
+ CALLBACK_WITH_FAILURE(cb, NULL, cbd->data);
+
+ g_free(cbd);
+}
+
+static void qmi_query_model(struct ofono_devinfo *devinfo,
+ ofono_devinfo_query_cb_t cb, void *user_data)
+{
+ struct devinfo_data *data = ofono_devinfo_get_data(devinfo);
+ struct cb_data *cbd = cb_data_new(cb, user_data);
+
+ DBG("");
+
+ if (qmi_service_send(data->dms, QMI_DMS_GET_MODEL_ID, NULL,
+ string_cb, cbd, g_free) > 0)
+ return;
+
+ CALLBACK_WITH_FAILURE(cb, NULL, cbd->data);
+
+ g_free(cbd);
+}
+
+static void qmi_query_revision(struct ofono_devinfo *devinfo,
+ ofono_devinfo_query_cb_t cb, void *user_data)
+{
+ struct devinfo_data *data = ofono_devinfo_get_data(devinfo);
+ struct cb_data *cbd = cb_data_new(cb, user_data);
+
+ DBG("");
+
+ if (qmi_service_send(data->dms, QMI_DMS_GET_REV_ID, NULL,
+ string_cb, cbd, g_free) > 0)
+ return;
+
+ CALLBACK_WITH_FAILURE(cb, NULL, cbd->data);
+
+ g_free(cbd);
+}
+
+static void get_ids_cb(struct qmi_result *result, void *user_data)
+{
+ struct cb_data *cbd = user_data;
+ ofono_devinfo_query_cb_t cb = cbd->cb;
+ char *str;
+
+ DBG("");
+
+ if (qmi_result_set_error(result, NULL)) {
+ CALLBACK_WITH_FAILURE(cb, NULL, cbd->data);
+ return;
+ }
+
+ str = qmi_result_get_string(result, QMI_DMS_RESULT_ESN);
+ if (!str) {
+ str = qmi_result_get_string(result, QMI_DMS_RESULT_IMEI);
+ if (!str) {
+ CALLBACK_WITH_FAILURE(cb, NULL, cbd->data);
+ return;
+ }
+ }
+
+ CALLBACK_WITH_SUCCESS(cb, str, cbd->data);
+
+ qmi_free(str);
+}
+
+static void qmi_query_serial(struct ofono_devinfo *devinfo,
+ ofono_devinfo_query_cb_t cb, void *user_data)
+{
+ struct devinfo_data *data = ofono_devinfo_get_data(devinfo);
+ struct cb_data *cbd = cb_data_new(cb, user_data);
+
+ DBG("");
+
+ if (qmi_service_send(data->dms, QMI_DMS_GET_IDS, NULL,
+ get_ids_cb, cbd, g_free) > 0)
+ return;
+
+ CALLBACK_WITH_FAILURE(cb, NULL, cbd->data);
+
+ g_free(cbd);
+}
+
+static void create_dms_cb(struct qmi_service *service, void *user_data)
+{
+ struct ofono_devinfo *devinfo = user_data;
+ struct devinfo_data *data = ofono_devinfo_get_data(devinfo);
+
+ DBG("");
+
+ if (!service) {
+ ofono_error("Failed to request DMS service");
+ ofono_devinfo_remove(devinfo);
+ return;
+ }
+
+ data->dms = qmi_service_ref(service);
+
+ ofono_devinfo_register(devinfo);
+}
+
+static int qmi_devinfo_probe(struct ofono_devinfo *devinfo,
+ unsigned int vendor, void *user_data)
+{
+ struct qmi_device *device = user_data;
+ struct devinfo_data *data;
+
+ DBG("");
+
+ data = g_new0(struct devinfo_data, 1);
+
+ ofono_devinfo_set_data(devinfo, data);
+
+ qmi_service_create(device, QMI_SERVICE_DMS,
+ create_dms_cb, devinfo, NULL);
+
+ return 0;
+}
+
+static void qmi_devinfo_remove(struct ofono_devinfo *devinfo)
+{
+ struct devinfo_data *data = ofono_devinfo_get_data(devinfo);
+
+ DBG("");
+
+ ofono_devinfo_set_data(devinfo, NULL);
+
+ qmi_service_unref(data->dms);
+
+ g_free(data);
+}
+
+static struct ofono_devinfo_driver driver = {
+ .name = "qmimodem",
+ .probe = qmi_devinfo_probe,
+ .remove = qmi_devinfo_remove,
+ .query_manufacturer = qmi_query_manufacturer,
+ .query_model = qmi_query_model,
+ .query_revision = qmi_query_revision,
+ .query_serial = qmi_query_serial,
+};
+
+void qmi_devinfo_init(void)
+{
+ ofono_devinfo_driver_register(&driver);
+}
+
+void qmi_devinfo_exit(void)
+{
+ ofono_devinfo_driver_unregister(&driver);
+}