[PATCH v2] qmicli: support auto/manual network with --nas-set-system-selection-preference
Dan Williams
dcbw at redhat.com
Mon Jun 4 16:13:08 UTC 2018
Support auto/manual network selection by extending the existing
NAS Set System Selection Preference command with some new options.
--nas-set-system-selection-preference=auto
--nas-set-system-selection-preference=manual=310260
--nas-set-system-selection-preference=gsm|umts|lte,manual=310260
---
src/qmicli/qmicli-helpers.c | 74 ++++++++++++++++++++++++++++
src/qmicli/qmicli-helpers.h | 4 ++
src/qmicli/qmicli-nas.c | 114 +++++++++++++++++++++++++++++++-------------
3 files changed, 160 insertions(+), 32 deletions(-)
diff --git a/src/qmicli/qmicli-helpers.c b/src/qmicli/qmicli-helpers.c
index 00f7122..84aef4a 100644
--- a/src/qmicli/qmicli-helpers.c
+++ b/src/qmicli/qmicli-helpers.c
@@ -263,6 +263,80 @@ qmicli_read_rat_mode_pref_from_string (const gchar *str,
return success && set;
}
+static gboolean
+parse_3gpp_mcc_mnc (const gchar *str,
+ guint16 *out_mcc,
+ guint16 *out_mnc)
+{
+ guint len;
+ guint i;
+ gchar aux[4];
+ guint16 tmp;
+
+ len = strlen (str);
+ if (len != 5 && len != 6)
+ return FALSE;
+ for (i = 0; i < len; i++) {
+ if (!g_ascii_isdigit (str[i]))
+ return FALSE;
+ }
+
+ memcpy (&aux[0], str, 3);
+ aux[3] = '\0';
+ tmp = atoi (aux);
+ if (tmp == 0)
+ return FALSE;
+ *out_mcc = tmp;
+
+ if (len == 5) {
+ memcpy (&aux[0], &str[3], 2);
+ aux[2] = '\0';
+ } else
+ memcpy (&aux[0], &str[3], 3);
+ *out_mnc = atoi (aux);
+
+ return TRUE;
+}
+
+gboolean
+qmicli_read_net_selection_pref_from_string (const gchar *str,
+ QmiNasNetworkSelectionPreference *out,
+ guint16 *out_mcc,
+ guint16 *out_mnc)
+{
+ GType type;
+ GEnumClass *enum_class;
+ GEnumValue *enum_value;
+ gchar *copy, *equals;
+ guint16 mcc = 0, mnc = 0;
+
+ copy = g_strdup (str);
+ equals = strchr (copy, '=');
+ if (equals) {
+ /* Parse MCC/MNC */
+ *equals++ = '\0';
+ if (!parse_3gpp_mcc_mnc (equals, &mcc, &mnc)) {
+ g_free (copy);
+ g_printerr ("error: invalid net selection MCC/MNC: '%s'\n", equals);
+ return FALSE;
+ }
+ }
+
+ type = qmi_nas_network_selection_preference_get_type ();
+ enum_class = G_ENUM_CLASS (g_type_class_ref (type));
+ enum_value = g_enum_get_value_by_nick (enum_class, copy);
+ if (enum_value) {
+ *out = (QmiNasNetworkSelectionPreference)enum_value->value;
+ *out_mcc = mcc;
+ *out_mnc = mnc;
+ } else
+ g_printerr ("error: invalid net selection preference value given: '%s'\n", copy);
+
+ g_free (copy);
+ g_type_class_unref (enum_class);
+ return !!enum_value;
+}
+
gboolean
qmicli_read_facility_from_string (const gchar *str,
QmiDmsUimFacility *out)
diff --git a/src/qmicli/qmicli-helpers.h b/src/qmicli/qmicli-helpers.h
index e8fac1d..6fd45d7 100644
--- a/src/qmicli/qmicli-helpers.h
+++ b/src/qmicli/qmicli-helpers.h
@@ -40,6 +40,10 @@ gboolean qmicli_read_operating_mode_from_string (const gchar *str,
QmiDmsOperatingMode *out);
gboolean qmicli_read_rat_mode_pref_from_string (const gchar *str,
QmiNasRatModePreference *out);
+gboolean qmicli_read_net_selection_pref_from_string (const gchar *str,
+ QmiNasNetworkSelectionPreference *out,
+ guint16 *out_mcc,
+ guint16 *out_mnc);
gboolean qmicli_read_facility_from_string (const gchar *str,
QmiDmsUimFacility *out);
gboolean qmicli_read_enable_disable_from_string (const gchar *str,
diff --git a/src/qmicli/qmicli-nas.c b/src/qmicli/qmicli-nas.c
index 2ea5a6b..203aa1e 100644
--- a/src/qmicli/qmicli-nas.c
+++ b/src/qmicli/qmicli-nas.c
@@ -97,7 +97,7 @@ static GOptionEntry entries[] = {
},
{ "nas-set-system-selection-preference", 0, 0, G_OPTION_ARG_STRING, &set_system_selection_preference_str,
"Set system selection preference",
- "[cdma-1x|cdma-1xevdo|gsm|umts|lte|td-scdma]"
+ "[cdma-1x|cdma-1xevdo|gsm|umts|lte|td-scdma][,[auto|manual=MCCMNC]]"
},
{ "nas-network-scan", 0, 0, G_OPTION_ARG_NONE, &network_scan_flag,
"Scan networks",
@@ -2137,24 +2137,81 @@ static QmiMessageNasSetSystemSelectionPreferenceInput *
set_system_selection_preference_input_create (const gchar *str)
{
QmiMessageNasSetSystemSelectionPreferenceInput *input = NULL;
- QmiNasRatModePreference pref;
+ QmiNasRatModePreference pref = 0;
+ QmiNasNetworkSelectionPreference net_pref = 0;
+ gchar *rat_pref_str = NULL;
+ gchar *net_pref_str = NULL;
GError *error = NULL;
+ guint16 mcc = 0, mnc = 0;
- if (!qmicli_read_rat_mode_pref_from_string (str, &pref)) {
- g_printerr ("error: failed to parse mode pref\n");
- return NULL;
+ if (strchr (str, ',')) {
+ gchar **parts;
+
+ /* Both RAT mode preference and network selection preference were given */
+ parts = g_strsplit_set (str, ",", -1);
+ if (g_strv_length (parts) != 2) {
+ g_printerr ("error: failed to parse selection pref: '%s'\n", str);
+ g_strfreev (parts);
+ return NULL;
+ }
+ rat_pref_str = g_strdup (parts[0]);
+ net_pref_str = g_strdup (parts[1]);
+ g_strfreev (parts);
+ } else if (g_str_has_prefix (str, "auto") || g_str_has_prefix (str, "manual")) {
+ /* Only network selection preference was given */
+ net_pref_str = g_strdup (str);
+ } else {
+ /* Only RAT mode preference was given */
+ rat_pref_str = g_strdup (str);
+ }
+
+ if (net_pref_str && !qmicli_read_net_selection_pref_from_string (net_pref_str, &net_pref, &mcc, &mnc)) {
+ g_printerr ("error: failed to parse net pref: '%s'\n", net_pref_str);
+ goto error;
+ }
+
+ if (rat_pref_str && !qmicli_read_rat_mode_pref_from_string (rat_pref_str, &pref)) {
+ g_printerr ("error: failed to parse mode pref: '%s'\n", rat_pref_str);
+ goto error;
}
input = qmi_message_nas_set_system_selection_preference_input_new ();
- if (!qmi_message_nas_set_system_selection_preference_input_set_mode_preference (
- input,
- pref,
- &error)) {
- g_printerr ("error: couldn't create input data bundle: '%s'\n",
- error->message);
- g_error_free (error);
- qmi_message_nas_set_system_selection_preference_input_unref (input);
- return NULL;
+
+ if (rat_pref_str) {
+ if (!qmi_message_nas_set_system_selection_preference_input_set_mode_preference (
+ input,
+ pref,
+ &error)) {
+ g_printerr ("error: couldn't create input data bundle: '%s'\n",
+ error->message);
+ goto error;
+ }
+
+ if (pref & (QMI_NAS_RAT_MODE_PREFERENCE_GSM |
+ QMI_NAS_RAT_MODE_PREFERENCE_UMTS |
+ QMI_NAS_RAT_MODE_PREFERENCE_LTE)) {
+ if (!qmi_message_nas_set_system_selection_preference_input_set_gsm_wcdma_acquisition_order_preference (
+ input,
+ QMI_NAS_GSM_WCDMA_ACQUISITION_ORDER_PREFERENCE_AUTOMATIC,
+ &error)) {
+ g_printerr ("error: couldn't create input data bundle: '%s'\n",
+ error->message);
+ goto error;
+ }
+ }
+ }
+
+ if (net_pref_str) {
+ if (!qmi_message_nas_set_system_selection_preference_input_set_network_selection_preference (
+ input,
+ net_pref,
+ mcc,
+ mnc,
+ &error)) {
+ g_printerr ("error: couldn't create input data bundle: '%s'\n",
+ error->message);
+ goto error;
+ }
}
if (!qmi_message_nas_set_system_selection_preference_input_set_change_duration (
@@ -2163,27 +2220,20 @@ set_system_selection_preference_input_create (const gchar *str)
&error)) {
g_printerr ("error: couldn't create input data bundle: '%s'\n",
error->message);
- g_error_free (error);
- qmi_message_nas_set_system_selection_preference_input_unref (input);
- return NULL;
- }
-
- if (pref & (QMI_NAS_RAT_MODE_PREFERENCE_GSM |
- QMI_NAS_RAT_MODE_PREFERENCE_UMTS |
- QMI_NAS_RAT_MODE_PREFERENCE_LTE)) {
- if (!qmi_message_nas_set_system_selection_preference_input_set_gsm_wcdma_acquisition_order_preference (
- input,
- QMI_NAS_GSM_WCDMA_ACQUISITION_ORDER_PREFERENCE_AUTOMATIC,
- &error)) {
- g_printerr ("error: couldn't create input data bundle: '%s'\n",
- error->message);
- g_error_free (error);
- qmi_message_nas_set_system_selection_preference_input_unref (input);
- return NULL;
- }
+ goto error;
}
+ g_free (rat_pref_str);
+ g_free (net_pref_str);
return input;
+
+error:
+ g_clear_error (&error);
+ if (input)
+ qmi_message_nas_set_system_selection_preference_input_unref (input);
+ g_free (rat_pref_str);
+ g_free (net_pref_str);
+ return NULL;
}
static void
--
2.14.3
More information about the libqmi-devel
mailing list