+/*
+ * 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 qmi_nas_set_system_selection_preference_request sel_req;
+static struct {
+ bool mcc_is_set;
+ bool mnc_is_set;
+} plmn_code_flag;
+
+#define cmd_nas_do_set_system_selection_cb no_cb
+static enum qmi_cmd_result
+cmd_nas_do_set_system_selection_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
+{
+ qmi_set_nas_set_system_selection_preference_request(msg, &sel_req);
+ return QMI_CMD_REQUEST;
+}
+
+static enum qmi_cmd_result
+do_sel_network(void)
+{
+ static bool use_sel_req = false;
+
+ if (!use_sel_req) {
+ use_sel_req = true;
+ uqmi_add_command(NULL, __UQMI_COMMAND_nas_do_set_system_selection);
+ }
+
+ return QMI_CMD_DONE;
+}
#define cmd_nas_set_network_modes_cb no_cb
static enum qmi_cmd_result
}
qmi_set(&sel_req, mode_preference, val);
- qmi_set_nas_set_system_selection_preference_request(msg, &sel_req);
- return QMI_CMD_REQUEST;
+ return do_sel_network();
}
#define cmd_nas_set_network_preference_cb no_cb
pref = QMI_NAS_GSM_WCDMA_ACQUISITION_ORDER_PREFERENCE_WCDMA;
qmi_set(&sel_req, gsm_wcdma_acquisition_order_preference, pref);
- qmi_set_nas_set_system_selection_preference_request(msg, &sel_req);
- return QMI_CMD_REQUEST;
+ return do_sel_network();
+}
+
+#define cmd_nas_set_roaming_cb no_cb
+static enum qmi_cmd_result
+cmd_nas_set_roaming_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
+{
+ QmiNasRoamingPreference pref;
+
+ if (!strcmp(arg, "any"))
+ pref = QMI_NAS_ROAMING_PREFERENCE_ANY;
+ else if (!strcmp(arg, "only"))
+ pref = QMI_NAS_ROAMING_PREFERENCE_NOT_OFF;
+ else if (!strcmp(arg, "off"))
+ pref = QMI_NAS_ROAMING_PREFERENCE_OFF;
+ else
+ return uqmi_add_error("Invalid argument");
+
+ qmi_set(&sel_req, roaming_preference, pref);
+ return do_sel_network();
+}
+
+#define cmd_nas_set_mcc_cb no_cb
+static enum qmi_cmd_result
+cmd_nas_set_mcc_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
+{
+ char *err;
+ int value = strtoul(arg, &err, 10);
+ if (err && *err) {
+ uqmi_add_error("Invalid MCC value");
+ return QMI_CMD_EXIT;
+ }
+
+ sel_req.data.network_selection_preference.mcc = value;
+ plmn_code_flag.mcc_is_set = true;
+ return QMI_CMD_DONE;
+}
+
+#define cmd_nas_set_mnc_cb no_cb
+static enum qmi_cmd_result
+cmd_nas_set_mnc_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
+{
+ char *err;
+ int value = strtoul(arg, &err, 10);
+ if (err && *err) {
+ uqmi_add_error("Invalid MNC value");
+ return QMI_CMD_EXIT;
+ }
+
+ sel_req.data.network_selection_preference.mnc = value;
+ plmn_code_flag.mnc_is_set = true;
+ return QMI_CMD_DONE;
+}
+
+#define cmd_nas_set_plmn_cb no_cb
+static enum qmi_cmd_result
+cmd_nas_set_plmn_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
+{
+ sel_req.set.network_selection_preference = 1;
+ sel_req.data.network_selection_preference.mode = QMI_NAS_NETWORK_SELECTION_PREFERENCE_AUTOMATIC;
+
+ if (!plmn_code_flag.mcc_is_set && plmn_code_flag.mnc_is_set) {
+ uqmi_add_error("No MCC value");
+ return QMI_CMD_EXIT;
+ }
+
+ if (plmn_code_flag.mcc_is_set && sel_req.data.network_selection_preference.mcc) {
+ if (!plmn_code_flag.mnc_is_set) {
+ uqmi_add_error("No MNC value");
+ return QMI_CMD_EXIT;
+ } else {
+ sel_req.data.network_selection_preference.mode = QMI_NAS_NETWORK_SELECTION_PREFERENCE_MANUAL;
+ }
+ }
+
+ return do_sel_network();
}
#define cmd_nas_initiate_network_register_cb no_cb
static enum qmi_cmd_result
cmd_nas_initiate_network_register_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
{
- static struct qmi_nas_initiate_network_register_request register_req;
+ static struct qmi_nas_initiate_network_register_request register_req = {
+ QMI_INIT(action, QMI_NAS_NETWORK_REGISTER_TYPE_AUTOMATIC)
+ };
+
qmi_set_nas_initiate_network_register_request(msg, ®ister_req);
return QMI_CMD_REQUEST;
}
blobmsg_add_u32(&status, "snr", (int32_t) res.data.lte_signal_strength.snr);
}
+ if (res.set.tdma_signal_strength) {
+ blobmsg_add_string(&status, "type", "tdma");
+ blobmsg_add_u32(&status, "signal", (int32_t) res.data.tdma_signal_strength);
+ }
+
blobmsg_close_table(&status, c);
}
static enum qmi_cmd_result
cmd_nas_network_scan_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
{
- struct qmi_nas_network_scan_request sreq = {};
+ struct qmi_nas_network_scan_request sreq = {
+ QMI_INIT(network_type,
+ QMI_NAS_NETWORK_SCAN_TYPE_GSM |
+ QMI_NAS_NETWORK_SCAN_TYPE_UMTS |
+ QMI_NAS_NETWORK_SCAN_TYPE_LTE |
+ QMI_NAS_NETWORK_SCAN_TYPE_TD_SCDMA),
+ };
qmi_set_nas_network_scan_request(msg, &sreq);
return QMI_CMD_REQUEST;