aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--plugins/mm-modem-cinterion-gsm.c422
1 files changed, 421 insertions, 1 deletions
diff --git a/plugins/mm-modem-cinterion-gsm.c b/plugins/mm-modem-cinterion-gsm.c
index 7edf1c87..199d03af 100644
--- a/plugins/mm-modem-cinterion-gsm.c
+++ b/plugins/mm-modem-cinterion-gsm.c
@@ -19,6 +19,7 @@
#include <stdio.h>
#include <string.h>
#include <unistd.h>
+#include <ctype.h>
#include "mm-errors.h"
#include "mm-modem-helpers.h"
@@ -26,7 +27,27 @@
#include "mm-serial-parsers.h"
#include "mm-log.h"
-G_DEFINE_TYPE (MMModemCinterionGsm, mm_modem_cinterion_gsm, MM_TYPE_GENERIC_GSM);
+static void modem_gsm_network_init (MMModemGsmNetwork *gsm_network_class);
+
+G_DEFINE_TYPE_EXTENDED (MMModemCinterionGsm, mm_modem_cinterion_gsm, MM_TYPE_GENERIC_GSM, 0,
+ G_IMPLEMENT_INTERFACE (MM_TYPE_MODEM_GSM_NETWORK, modem_gsm_network_init))
+
+/* Mask of all bands supported in 2G devices */
+#define ALL_2G_BANDS \
+ (MM_MODEM_GSM_BAND_EGSM | \
+ MM_MODEM_GSM_BAND_DCS | \
+ MM_MODEM_GSM_BAND_PCS | \
+ MM_MODEM_GSM_BAND_G850)
+
+/* Mask of all bands supported in 3G devices (including some 2G bands) */
+#define ALL_3G_BANDS \
+ (MM_MODEM_GSM_BAND_EGSM | \
+ MM_MODEM_GSM_BAND_DCS | \
+ MM_MODEM_GSM_BAND_PCS | \
+ MM_MODEM_GSM_BAND_G850 | \
+ MM_MODEM_GSM_BAND_U2100 | \
+ MM_MODEM_GSM_BAND_U1900 | \
+ MM_MODEM_GSM_BAND_U850)
#define MM_MODEM_CINTERION_GSM_GET_PRIVATE(o) \
(G_TYPE_INSTANCE_GET_PRIVATE ((o), MM_TYPE_MODEM_CINTERION_GSM, MMModemCinterionGsmPrivate))
@@ -42,8 +63,54 @@ typedef struct {
/* Current allowed mode */
MMModemGsmAllowedMode allowed_mode;
+
+ /* Bitmask for currently active bands */
+ guint32 current_bands;
} MMModemCinterionGsmPrivate;
+/* Setup relationship between the band bitmask in the modem and the bitmask
+ * in ModemManager. */
+typedef struct {
+ gchar *cinterion_band;
+ guint32 mm_band_mask;
+} CinterionBand2G;
+/* Table checked in both MC75i (GPRS/EDGE) and EGS5 (GPRS) references.
+ * Note that the modem's configuration is also based on a bitmask, but as we
+ * will just support some of the combinations, we just use strings for them.
+ */
+static const CinterionBand2G bands_2g[] = {
+ { "1", MM_MODEM_GSM_BAND_EGSM },
+ { "2", MM_MODEM_GSM_BAND_DCS },
+ { "4", MM_MODEM_GSM_BAND_PCS },
+ { "8", MM_MODEM_GSM_BAND_G850 },
+ { "3", (MM_MODEM_GSM_BAND_EGSM | MM_MODEM_GSM_BAND_DCS) },
+ { "5", (MM_MODEM_GSM_BAND_EGSM | MM_MODEM_GSM_BAND_PCS) },
+ { "10", (MM_MODEM_GSM_BAND_G850 | MM_MODEM_GSM_BAND_DCS) },
+ { "12", (MM_MODEM_GSM_BAND_G850 | MM_MODEM_GSM_BAND_PCS) },
+ { "15", ALL_2G_BANDS }
+};
+
+/* Setup relationship between the 3G band bitmask in the modem and the bitmask
+ * in ModemManager. */
+typedef struct {
+ guint32 cinterion_band_flag;
+ guint32 mm_band_flag;
+} CinterionBand3G;
+/* Table checked in HC25 (3G) reference. This table includes both 2G and 3G
+ * frequencies. Depending on which one is configured, one access technology or
+ * the other will be used. This may conflict with the allowed mode configuration
+ * set, so you shouldn't for example set 3G frequency bands, and then use a
+ * 2G-only allowed mode. */
+static const CinterionBand3G bands_3g[] = {
+ { (1 << 0), MM_MODEM_GSM_BAND_EGSM },
+ { (1 << 1), MM_MODEM_GSM_BAND_DCS },
+ { (1 << 2), MM_MODEM_GSM_BAND_PCS },
+ { (1 << 3), MM_MODEM_GSM_BAND_G850 },
+ { (1 << 4), MM_MODEM_GSM_BAND_U2100 },
+ { (1 << 5), MM_MODEM_GSM_BAND_U1900 },
+ { (1 << 6), MM_MODEM_GSM_BAND_U850 }
+};
+
MMModem *
mm_modem_cinterion_gsm_new (const char *device,
const char *driver,
@@ -64,6 +131,351 @@ mm_modem_cinterion_gsm_new (const char *device,
NULL));
}
+static void
+convert_str_from_ucs2 (gchar **str)
+{
+ const char *p;
+ char *converted;
+ size_t len;
+
+ p = *str;
+ len = strlen (p);
+
+ /* Len needs to be a multiple of 4 for UCS2 */
+ if ((len < 4) || ((len % 4) != 0))
+ return;
+
+ while (*p) {
+ if (!isxdigit (*p++))
+ return;
+ }
+
+ converted = mm_modem_charset_hex_to_utf8 (*str, MM_MODEM_CHARSET_UCS2);
+ if (converted) {
+ g_free (*str);
+ *str = converted;
+ }
+}
+
+static void
+get_2g_band_done (MMAtSerialPort *port,
+ GString *response,
+ GError *error,
+ gpointer user_data)
+{
+ MMCallbackInfo *info = (MMCallbackInfo *) user_data;
+ MMModemCinterionGsmPrivate *priv = MM_MODEM_CINTERION_GSM_GET_PRIVATE (info->modem);
+
+ if (error)
+ info->error = g_error_copy (error);
+ else {
+ guint32 mm_band = MM_MODEM_GSM_BAND_UNKNOWN;
+ GRegex *regex;
+ GMatchInfo *match_info = NULL;
+
+ /* The AT^SCFG? command replies a list of several different config
+ * values. We will only look for 'Radio/Band".
+ *
+ * AT+SCFG="Radio/Band"
+ * ^SCFG: "Radio/Band","0031","0031"
+ *
+ * Note that "0031" is a UCS2-encoded string, as we configured UCS2 as
+ * character set to use.
+ */
+ regex = g_regex_new ("\\^SCFG:\\s*\"Radio/Band\",\\s*\"(.*)\",\\s*\"(.*)\"", 0, 0, NULL);
+ if (regex &&
+ g_regex_match_full (regex, response->str, response->len, 0, 0, &match_info, NULL)) {
+ gchar *current;
+
+ /* The first number given is the current band configuration, the
+ * second number given is the allowed band configuration, which we
+ * don't really need to get here. */
+ current = g_match_info_fetch (match_info, 1);
+ if (current) {
+ guint i;
+
+ /* If in UCS2, convert to UTF-8 */
+ if (mm_generic_gsm_get_charset (MM_GENERIC_GSM (info->modem)) == MM_MODEM_CHARSET_UCS2)
+ convert_str_from_ucs2 (&current);
+
+ for (i = 0; i < G_N_ELEMENTS (bands_2g); i++) {
+ if (strcmp (bands_2g[i].cinterion_band, current) == 0) {
+ mm_band = bands_2g[i].mm_band_mask;
+ break;
+ }
+ }
+
+ g_free (current);
+ }
+ }
+
+ if (mm_band == MM_MODEM_GSM_BAND_UNKNOWN) {
+ g_set_error (&info->error,
+ MM_MODEM_ERROR,
+ MM_MODEM_ERROR_GENERAL,
+ "Couldn't get bands configuration");
+ } else {
+ priv->current_bands = mm_band;
+ mm_callback_info_set_result (info, GUINT_TO_POINTER (mm_band), NULL);
+ }
+
+ if (regex)
+ g_regex_unref (regex);
+ if (match_info)
+ g_match_info_free (match_info);
+ }
+
+ mm_callback_info_schedule (info);
+}
+
+static void
+get_3g_band_done (MMAtSerialPort *port,
+ GString *response,
+ GError *error,
+ gpointer user_data)
+{
+ MMCallbackInfo *info = (MMCallbackInfo *) user_data;
+ MMModemCinterionGsmPrivate *priv = MM_MODEM_CINTERION_GSM_GET_PRIVATE (info->modem);
+
+ if (error)
+ info->error = g_error_copy (error);
+ else {
+ guint32 mm_band = 0;
+ GRegex *regex;
+ GMatchInfo *match_info = NULL;
+
+ /* The AT^SCFG? command replies a list of several different config
+ * values. We will only look for 'Radio/Band".
+ *
+ * AT+SCFG="Radio/Band"
+ * ^SCFG: "Radio/Band",127
+ *
+ * Note that in this case, the <rba> replied is a number, not a string.
+ */
+ regex = g_regex_new ("\\^SCFG:\\s*\"Radio/Band\",\\s*(\\d*)", 0, 0, NULL);
+ if (regex &&
+ g_regex_match_full (regex, response->str, response->len, 0, 0, &match_info, NULL)) {
+ gchar *current;
+
+ current = g_match_info_fetch (match_info, 1);
+ if (current) {
+ guint32 current_int;
+ guint i;
+
+ current_int = (guint32) atoi (current);
+
+ for (i = 0; i < G_N_ELEMENTS (bands_3g); i++) {
+ if (current_int & bands_3g[i].cinterion_band_flag)
+ mm_band |= bands_3g[i].mm_band_flag;
+ }
+
+ g_free (current);
+ }
+ }
+
+ if (mm_band == 0) {
+ g_set_error (&info->error,
+ MM_MODEM_ERROR,
+ MM_MODEM_ERROR_GENERAL,
+ "Couldn't get bands configuration");
+ } else {
+ priv->current_bands = mm_band;
+ mm_callback_info_set_result (info, GUINT_TO_POINTER (mm_band), NULL);
+ }
+
+ if (regex)
+ g_regex_unref (regex);
+ if (match_info)
+ g_match_info_free (match_info);
+ }
+
+ mm_callback_info_schedule (info);
+}
+
+static void
+get_band (MMModemGsmNetwork *self,
+ MMModemUIntFn callback,
+ gpointer user_data)
+{
+ MMModemCinterionGsmPrivate *priv = MM_MODEM_CINTERION_GSM_GET_PRIVATE (self);
+ MMAtSerialPort *port;
+ MMCallbackInfo *info;
+
+ info = mm_callback_info_uint_new (MM_MODEM (self), callback, user_data);
+
+ /* If results are already cached, return them */
+ if (priv->current_bands > 0) {
+ mm_callback_info_set_result (info, GUINT_TO_POINTER (priv->current_bands), NULL);
+ mm_callback_info_schedule (info);
+ return;
+ }
+
+ /* Otherwise ask the modem */
+ port = mm_generic_gsm_get_best_at_port (MM_GENERIC_GSM (self), &info->error);
+ if (!port) {
+ mm_callback_info_schedule (info);
+ return;
+ }
+
+ /* Query the currently used Radio/Band. The query command is the same for
+ * both 2G and 3G devices, but the reply reader is different. */
+ mm_at_serial_port_queue_command (port,
+ "AT^SCFG=\"Radio/Band\"",
+ 3,
+ ((!priv->only_utran && !priv->both_geran_utran) ?
+ get_2g_band_done :
+ get_3g_band_done),
+ info);
+}
+
+static void
+set_band_done (MMAtSerialPort *port,
+ GString *response,
+ GError *error,
+ gpointer user_data)
+{
+ MMCallbackInfo *info = (MMCallbackInfo *) user_data;
+ MMModemCinterionGsmPrivate *priv = MM_MODEM_CINTERION_GSM_GET_PRIVATE (info->modem);
+
+ if (error)
+ info->error = g_error_copy (error);
+ else
+ priv->current_bands = GPOINTER_TO_UINT (mm_callback_info_get_data (info, "new-band"));
+
+ mm_callback_info_schedule (info);
+}
+
+static void
+set_2g_band (MMModemGsmNetwork *self,
+ MMModemGsmBand band,
+ MMAtSerialPort *port,
+ MMCallbackInfo *info)
+{
+ const gchar *cinterion_band = NULL;
+ gchar *cinterion_band_ucs2 = NULL;
+ gchar *cmd;
+ guint i;
+
+ /* If we get ANY, reset to all-2G bands to get the proper value */
+ if (band == MM_MODEM_GSM_BAND_ANY)
+ band = ALL_2G_BANDS;
+
+ /* Loop looking for correct allowed masks */
+ for (i = 0; i < G_N_ELEMENTS (bands_2g); i++) {
+ if (bands_2g[i].mm_band_mask == band) {
+ cinterion_band = bands_2g[i].cinterion_band;
+ break;
+ }
+ }
+
+ /* If we didn't find a match, set an error */
+ if (!cinterion_band) {
+ info->error = g_error_new (MM_MODEM_ERROR,
+ MM_MODEM_ERROR_GENERAL,
+ "Wrong band mask: '%u'", band);
+ mm_callback_info_schedule (info);
+ return;
+ }
+
+ if (mm_generic_gsm_get_charset (MM_GENERIC_GSM (info->modem)) == MM_MODEM_CHARSET_UCS2)
+ cinterion_band_ucs2 = mm_modem_charset_utf8_to_hex (cinterion_band, MM_MODEM_CHARSET_UCS2);
+
+ mm_callback_info_set_data (info,
+ "new-band",
+ GUINT_TO_POINTER ((guint)band),
+ NULL);
+ /* Following the setup:
+ * AT^SCFG="Radion/Band",<rbp>,<rba>
+ * We will set the preferred band equal to the allowed band, so that we force
+ * the modem to connect at that specific frequency only. Note that we will be
+ * passing double-quote enclosed strings here!
+ */
+ cmd = g_strdup_printf ("^SCFG=\"Radio/Band\",\"%s\",\"%s\"",
+ cinterion_band_ucs2 ? cinterion_band_ucs2 : cinterion_band,
+ cinterion_band_ucs2 ? cinterion_band_ucs2 : cinterion_band);
+ mm_at_serial_port_queue_command (port, cmd, 3, set_band_done, info);
+ g_free (cmd);
+ g_free (cinterion_band_ucs2);
+}
+
+static void
+set_3g_band (MMModemGsmNetwork *self,
+ MMModemGsmBand band,
+ MMAtSerialPort *port,
+ MMCallbackInfo *info)
+{
+ guint32 cinterion_band = 0;
+ gchar *cmd;
+ guint i;
+
+ /* If we get ANY, reset to all-3G bands to get the proper value */
+ if (band == MM_MODEM_GSM_BAND_ANY)
+ band = ALL_3G_BANDS;
+
+ /* Loop looking for correct allowed masks */
+ for (i = 0; i < G_N_ELEMENTS (bands_3g); i++) {
+ if (band & bands_3g[i].mm_band_flag) {
+ cinterion_band |= bands_3g[i].cinterion_band_flag;
+ }
+ }
+
+ /* If we didn't find a match, set an error */
+ if (!cinterion_band) {
+ info->error = g_error_new (MM_MODEM_ERROR,
+ MM_MODEM_ERROR_GENERAL,
+ "Wrong band mask: '%u'", band);
+ mm_callback_info_schedule (info);
+ return;
+ }
+
+ mm_callback_info_set_data (info,
+ "new-band",
+ GUINT_TO_POINTER ((guint)band),
+ NULL);
+ /* Following the setup:
+ * AT^SCFG="Radion/Band",<rba>
+ * We will set the preferred band equal to the allowed band, so that we force
+ * the modem to connect at that specific frequency only. Note that we will be
+ * passing a number here!
+ */
+ cmd = g_strdup_printf ("^SCFG=\"Radio/Band\",%u", cinterion_band);
+ mm_at_serial_port_queue_command (port, cmd, 3, set_band_done, info);
+ g_free (cmd);
+}
+
+static void
+set_band (MMModemGsmNetwork *self,
+ MMModemGsmBand band,
+ MMModemFn callback,
+ gpointer user_data)
+{
+ MMModemCinterionGsmPrivate *priv = MM_MODEM_CINTERION_GSM_GET_PRIVATE (self);
+ MMAtSerialPort *port;
+ MMCallbackInfo *info;
+
+ info = mm_callback_info_new (MM_MODEM (self), callback, user_data);
+
+ /* Are we trying to change the band to the same bands currently
+ * being used? if so, we're done */
+ if (priv->current_bands == band) {
+ mm_callback_info_schedule (info);
+ return;
+ }
+
+ /* Otherwise ask the modem */
+ port = mm_generic_gsm_get_best_at_port (MM_GENERIC_GSM (self), &info->error);
+ if (!port) {
+ mm_callback_info_schedule (info);
+ return;
+ }
+
+ if (!priv->only_utran &&
+ !priv->both_geran_utran)
+ set_2g_band (self, band, port, info);
+ else
+ set_3g_band (self, band, port, info);
+}
+
static MMModemGsmAccessTech
get_access_technology_from_smong_gprs_status (const gchar *gprs_status,
GError **error)
@@ -466,6 +878,13 @@ do_enable_power_up_done (MMGenericGsm *gsm,
/*****************************************************************************/
static void
+modem_gsm_network_init (MMModemGsmNetwork *network_class)
+{
+ network_class->set_band = set_band;
+ network_class->get_band = get_band;
+}
+
+static void
mm_modem_cinterion_gsm_init (MMModemCinterionGsm *self)
{
MMModemCinterionGsmPrivate *priv = MM_MODEM_CINTERION_GSM_GET_PRIVATE (self);
@@ -476,6 +895,7 @@ mm_modem_cinterion_gsm_init (MMModemCinterionGsm *self)
priv->only_geran = FALSE;
priv->only_utran = FALSE;
priv->both_geran_utran = FALSE;
+ priv->current_bands = 0; /* This is a bitmask, so empty */
}
static void