+/*
+ * uqmi -- tiny QMI support implementation
+ *
+ * Copyright (C) 2014-2015 Felix Fietkau <nbd@openwrt.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
+ * Boston, MA 02110-1301 USA.
+ */
#include "qmi-message.h"
static struct {
char* puk;
} dms_req_data;
-static const char *get_pin_status(int status)
+const char *oper_modes[] = {
+ [QMI_DMS_OPERATING_MODE_ONLINE] = "online",
+ [QMI_DMS_OPERATING_MODE_LOW_POWER] = "low_power",
+ [QMI_DMS_OPERATING_MODE_FACTORY_TEST] = "factory_test",
+ [QMI_DMS_OPERATING_MODE_OFFLINE] = "offline",
+ [QMI_DMS_OPERATING_MODE_RESET] = "reset",
+ [QMI_DMS_OPERATING_MODE_SHUTTING_DOWN] = "shutting_down",
+ [QMI_DMS_OPERATING_MODE_PERSISTENT_LOW_POWER] = "persistent_low_power",
+ [QMI_DMS_OPERATING_MODE_MODE_ONLY_LOW_POWER] = "mode_only_low_power",
+};
+
+static void cmd_dms_get_capabilities_cb(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg)
+{
+ void *t, *networks;
+ int i;
+ struct qmi_dms_get_capabilities_response res;
+ const char *radio_cap[] = {
+ [QMI_DMS_RADIO_INTERFACE_CDMA20001X] = "cdma1x",
+ [QMI_DMS_RADIO_INTERFACE_EVDO] = "cdma1xevdo",
+ [QMI_DMS_RADIO_INTERFACE_GSM] = "gsm",
+ [QMI_DMS_RADIO_INTERFACE_UMTS] = "umts",
+ [QMI_DMS_RADIO_INTERFACE_LTE] = "lte",
+ };
+ const char *service_cap[] = {
+ [QMI_DMS_DATA_SERVICE_CAPABILITY_NONE] = "none",
+ [QMI_DMS_DATA_SERVICE_CAPABILITY_CS] = "cs",
+ [QMI_DMS_DATA_SERVICE_CAPABILITY_PS] = "ps",
+ [QMI_DMS_DATA_SERVICE_CAPABILITY_SIMULTANEOUS_CS_PS] = "simultaneous_cs_ps",
+ [QMI_DMS_DATA_SERVICE_CAPABILITY_NON_SIMULTANEOUS_CS_PS] = "non_simultaneous_cs_ps",
+ };
+
+ qmi_parse_dms_get_capabilities_response(msg, &res);
+
+ t = blobmsg_open_table(&status, NULL);
+
+ blobmsg_add_u32(&status, "max_tx_channel_rate", (int32_t) res.data.info.max_tx_channel_rate);
+ blobmsg_add_u32(&status, "max_rx_channel_rate", (int32_t) res.data.info.max_rx_channel_rate);
+ if ((int)res.data.info.data_service_capability >= 0 && res.data.info.data_service_capability < ARRAY_SIZE(service_cap))
+ blobmsg_add_string(&status, "data_service", service_cap[res.data.info.data_service_capability]);
+
+ if(res.data.info.sim_capability == QMI_DMS_SIM_CAPABILITY_NOT_SUPPORTED)
+ blobmsg_add_string(&status, "sim", "not supported");
+ else if(res.data.info.sim_capability == QMI_DMS_SIM_CAPABILITY_SUPPORTED)
+ blobmsg_add_string(&status, "sim", "supported");
+
+ networks = blobmsg_open_array(&status, "networks");
+ for (i = 0; i < res.data.info.radio_interface_list_n; i++) {
+ if ((int)res.data.info.radio_interface_list[i] >= 0 && res.data.info.radio_interface_list[i] < ARRAY_SIZE(radio_cap))
+ blobmsg_add_string(&status, NULL, radio_cap[res.data.info.radio_interface_list[i]]);
+ else
+ blobmsg_add_string(&status, NULL, "unknown");
+ }
+ blobmsg_close_array(&status, networks);
+
+ blobmsg_close_table(&status, t);
+}
+
+static enum qmi_cmd_result
+cmd_dms_get_capabilities_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
+{
+ qmi_set_dms_get_capabilities_request(msg);
+ return QMI_CMD_REQUEST;
+}
+
+const char *get_pin_status(int status)
{
static const char *pin_status[] = {
[QMI_DMS_UIM_PIN_STATUS_NOT_INITIALIZED] = "not_initialized",
return QMI_CMD_REQUEST;
}
+static void cmd_dms_get_imei_cb(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg)
+{
+ struct qmi_dms_get_ids_response res;
+
+ qmi_parse_dms_get_ids_response(msg, &res);
+ if (res.data.imei)
+ blobmsg_add_string(&status, NULL, res.data.imei);
+}
+
+static enum qmi_cmd_result
+cmd_dms_get_imei_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
+{
+ qmi_set_dms_get_ids_request(msg);
+ return QMI_CMD_REQUEST;
+}
+
#define cmd_dms_reset_cb no_cb
static enum qmi_cmd_result
cmd_dms_reset_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
return QMI_CMD_REQUEST;
}
+static void
+cmd_dms_get_operating_mode_cb(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg)
+{
+ struct qmi_dms_get_operating_mode_response res;
+
+ qmi_parse_dms_get_operating_mode_response(msg, &res);
+ if (res.data.mode < ARRAY_SIZE(oper_modes))
+ blobmsg_add_string(&status, NULL, oper_modes[res.data.mode]);
+ else
+ blobmsg_add_string(&status, NULL, "unknown");
+}
+
+static enum qmi_cmd_result
+cmd_dms_get_operating_mode_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
+{
+ qmi_set_dms_get_operating_mode_request(msg);
+ return QMI_CMD_REQUEST;
+}
+
#define cmd_dms_set_operating_mode_cb no_cb
static enum qmi_cmd_result
cmd_dms_set_operating_mode_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
{
- static const char *modes[] = {
- [QMI_DMS_OPERATING_MODE_ONLINE] = "online",
- [QMI_DMS_OPERATING_MODE_LOW_POWER] = "low_power",
- [QMI_DMS_OPERATING_MODE_FACTORY_TEST] = "factory_test",
- [QMI_DMS_OPERATING_MODE_OFFLINE] = "offline",
- [QMI_DMS_OPERATING_MODE_RESET] = "reset",
- [QMI_DMS_OPERATING_MODE_SHUTTING_DOWN] = "shutting_down",
- [QMI_DMS_OPERATING_MODE_PERSISTENT_LOW_POWER] = "persistent_low_power",
- [QMI_DMS_OPERATING_MODE_MODE_ONLY_LOW_POWER] = "mode_only_low_power",
- };
static struct qmi_dms_set_operating_mode_request sreq = {
QMI_INIT(mode, QMI_DMS_OPERATING_MODE_ONLINE),
};
int i;
- for (i = 0; i < ARRAY_SIZE(modes); i++) {
- if (!modes[i])
+ for (i = 0; i < ARRAY_SIZE(oper_modes); i++) {
+ if (!oper_modes[i])
continue;
- if (strcmp(arg, modes[i]) != 0)
+ if (strcmp(arg, oper_modes[i]) != 0)
continue;
sreq.data.mode = i;
return uqmi_add_error("Invalid argument");
}
+
+#define cmd_dms_set_fcc_authentication_cb no_cb
+static enum qmi_cmd_result
+cmd_dms_set_fcc_authentication_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
+{
+ qmi_set_dms_set_fcc_authentication_request(msg);
+ return QMI_CMD_REQUEST;
+}