[PATCH 4/4] cinterion: use ^SIND unsolicited messages for access tech reporting

Dan Williams dcbw at redhat.com
Mon May 22 16:02:08 UTC 2017


On Sun, 2017-05-21 at 21:49 +0200, Aleksander Morgado wrote:
> If the modem supports ^SIND psinfo reporting, we enable the URC and
> flag the access technology polling unsupported, so that we only update
> access technology via the +CIEV URCs.
> 
> E.g.:
> 
>     (ttyACM1): --> 'AT^SIND="psinfo",1<CR>'
>     (ttyACM1): <-- '<CR><LF>^SIND: psinfo,1,10<CR><LF><CR><LF>OK<CR><LF>'
>     Reporting initial access technologies...
>     Modem /org/freedesktop/ModemManager1/Modem/0: access technology changed (unknown -> hsdpa, hsupa)
>     ...
>     (ttyACM1): <-- '<CR><LF>+CIEV: psinfo,4<CR><LF>'
>     Modem /org/freedesktop/ModemManager1/Modem/0: access technology changed (hsdpa, hsupa -> edge)
>     ...
> ---
>  plugins/cinterion/mm-broadband-modem-cinterion.c | 439 ++++++++++++++++-------
>  plugins/cinterion/mm-modem-helpers-cinterion.c   |  33 ++
>  plugins/cinterion/mm-modem-helpers-cinterion.h   |   5 +
>  3 files changed, 339 insertions(+), 138 deletions(-)
> 
> diff --git a/plugins/cinterion/mm-broadband-modem-cinterion.c b/plugins/cinterion/mm-broadband-modem-cinterion.c
> index 735f15bc..06f4357f 100644
> --- a/plugins/cinterion/mm-broadband-modem-cinterion.c
> +++ b/plugins/cinterion/mm-broadband-modem-cinterion.c
> @@ -61,7 +61,7 @@ typedef enum {
>  } FeatureSupport;
>  
>  struct _MMBroadbandModemCinterionPrivate {
> -    /* Flag to know if we should try AT^SIND or not to get psinfo */
> +    /* Flag to know if AT^SIND indications are supported */
>      gboolean sind_psinfo;
>  
>      /* Command to go into sleep mode */
> @@ -80,6 +80,9 @@ struct _MMBroadbandModemCinterionPrivate {
>      GArray *cnmi_supported_ds;
>      GArray *cnmi_supported_bfr;
>  
> +    /* +CIEV 'psinfo' indications */
> +    GRegex *ciev_psinfo_regex;
> +
>      /*Flags for SWWAN support*/
>      FeatureSupport swwan_support;
>  };
> @@ -572,185 +575,329 @@ modem_power_off (MMIfaceModem *self,
>  }
>  
>  /*****************************************************************************/
> -/* ACCESS TECHNOLOGIES */
> +/* Access technologies polling */
>  
>  static gboolean
> -load_access_technologies_finish (MMIfaceModem *self,
> -                                 GAsyncResult *res,
> -                                 MMModemAccessTechnology *access_technologies,
> -                                 guint *mask,
> -                                 GError **error)
> +load_access_technologies_finish (MMIfaceModem             *self,
> +                                 GAsyncResult             *res,
> +                                 MMModemAccessTechnology  *access_technologies,
> +                                 guint                    *mask,
> +                                 GError                  **error)
>  {
> -    if (g_simple_async_result_propagate_error (G_SIMPLE_ASYNC_RESULT (res), error))
> +    gssize val;
> +
> +    if ((val = g_task_propagate_int (G_TASK (res), error)) < 0)
>          return FALSE;
>  
> -    *access_technologies = (MMModemAccessTechnology) GPOINTER_TO_UINT (
> -        g_simple_async_result_get_op_res_gpointer (
> -            G_SIMPLE_ASYNC_RESULT (res)));
> +    *access_technologies = (MMModemAccessTechnology) val;
>      *mask = MM_MODEM_ACCESS_TECHNOLOGY_ANY;
>      return TRUE;
>  }
>  
>  static void
> -smong_query_ready (MMBroadbandModemCinterion *self,
> +smong_query_ready (MMBaseModem  *self,
>                     GAsyncResult *res,
> -                   GSimpleAsyncResult *operation_result)
> +                   GTask        *task)
>  {
>      const gchar             *response;
>      GError                  *error = NULL;
>      MMModemAccessTechnology  access_tech;
>  
> -    response = mm_base_modem_at_command_finish (MM_BASE_MODEM (self), res, &error);
> -    if (!response) {
> -        /* Let the error be critical. */
> -        g_simple_async_result_take_error (operation_result, error);
> -    } else if (!mm_cinterion_parse_smong_response (response, &access_tech, &error)) {
> -        /* We'll reset here the flag to try to use SIND/psinfo the next time */
> -        self->priv->sind_psinfo = TRUE;
> -        g_simple_async_result_take_error (operation_result, error);
> -    } else {
> -        /* We'll default to use SMONG then */
> -        self->priv->sind_psinfo = FALSE;
> -        g_simple_async_result_set_op_res_gpointer (operation_result, GUINT_TO_POINTER (access_tech), NULL);
> +    response = mm_base_modem_at_command_finish (self, res, &error);
> +    if (!response || !mm_cinterion_parse_smong_response (response, &access_tech, &error))
> +        g_task_return_error (task, error);
> +    else
> +        g_task_return_int (task, (gssize) access_tech);
> +    g_object_unref (task);
> +}
> +
> +static void
> +load_access_technologies (MMIfaceModem        *_self,
> +                          GAsyncReadyCallback  callback,
> +                          gpointer             user_data)
> +{
> +    MMBroadbandModemCinterion *self = MM_BROADBAND_MODEM_CINTERION (_self);
> +    GTask                     *task;
> +
> +    task = g_task_new (self, NULL, callback, user_data);
> +
> +    /* Abort access technology polling if ^SIND psinfo URCs are enabled */
> +    if (self->priv->sind_psinfo) {
> +        g_task_return_new_error (task, MM_CORE_ERROR, MM_CORE_ERROR_UNSUPPORTED, "No need to poll access technologies");
> +        g_object_unref (task);
> +        return;
>      }
>  
> -    g_simple_async_result_complete (operation_result);
> -    g_object_unref (operation_result);
> +    mm_base_modem_at_command (
> +        MM_BASE_MODEM (self),
> +        "^SMONG",
> +        3,
> +        FALSE,
> +        (GAsyncReadyCallback)smong_query_ready,
> +        task);
>  }
>  
> -static MMModemAccessTechnology
> -get_access_technology_from_psinfo (const gchar *psinfo,
> -                                   GError **error)
> -{
> -    guint psinfoval;
> -
> -    if (mm_get_uint_from_str (psinfo, &psinfoval)) {
> -        switch (psinfoval) {
> -        case 0:
> -            return MM_MODEM_ACCESS_TECHNOLOGY_UNKNOWN;
> -        case 1:
> -        case 2:
> -            return MM_MODEM_ACCESS_TECHNOLOGY_GPRS;
> -        case 3:
> -        case 4:
> -            return MM_MODEM_ACCESS_TECHNOLOGY_EDGE;
> -        case 5:
> -        case 6:
> -            return MM_MODEM_ACCESS_TECHNOLOGY_UMTS;
> -        case 7:
> -        case 8:
> -            return MM_MODEM_ACCESS_TECHNOLOGY_HSDPA;
> -        case 9:
> -        case 10:
> -            return (MM_MODEM_ACCESS_TECHNOLOGY_HSDPA | MM_MODEM_ACCESS_TECHNOLOGY_HSUPA);
> -        case 16:
> -        case 17:
> -            return MM_MODEM_ACCESS_TECHNOLOGY_LTE;
> -        default:
> -            mm_dbg ("Unable to identify access technology in case:%i", psinfoval);
> -            break;
> -        }
> +/*****************************************************************************/
> +/* Disable unsolicited events (3GPP interface) */
> +
> +static gboolean
> +modem_3gpp_disable_unsolicited_events_finish (MMIfaceModem3gpp  *self,
> +                                              GAsyncResult      *res,
> +                                              GError           **error)
> +{
> +    return g_task_propagate_boolean (G_TASK (res), error);
> +}
> +
> +static void
> +parent_disable_unsolicited_events_ready (MMIfaceModem3gpp *self,
> +                                         GAsyncResult     *res,
> +                                         GTask            *task)
> +{
> +    GError *error = NULL;
> +
> +    if (!iface_modem_3gpp_parent->disable_unsolicited_events_finish (self, res, &error)) {
> +        mm_warn ("Couldn't disable parent 3GPP unsolicited events: %s", error->message);
> +        g_error_free (error);
>      }
> -    else
> -        mm_err ("FAILED get_access_technology_from_psinfo-int");
>  
> -    g_set_error (error,
> -                 MM_CORE_ERROR,
> -                 MM_CORE_ERROR_INVALID_ARGS,
> -                 "Couldn't get network capabilities, "
> -                 "invalid psinfo value: '%s'",
> -                 psinfo);
> -    return MM_MODEM_ACCESS_TECHNOLOGY_UNKNOWN;
> +    g_task_return_boolean (task, TRUE);
> +    g_object_unref (task);
>  }
>  
>  static void
> -sind_query_ready (MMBroadbandModemCinterion *self,
> -                  GAsyncResult *res,
> -                  GSimpleAsyncResult *operation_result)
> +sind_psinfo_disable_ready (MMBaseModem  *self,
> +                           GAsyncResult *res,
> +                           GTask        *task)
>  {
> -    const gchar *response;
>      GError *error = NULL;
> -    GMatchInfo *match_info = NULL;
> -    GRegex *regex;
>  
> -    response = mm_base_modem_at_command_finish (MM_BASE_MODEM (self), res, &error);
> -    if (!response) {
> -        /* Let the error be critical. */
> -        g_simple_async_result_take_error (operation_result, error);
> -        g_simple_async_result_complete (operation_result);
> -        g_object_unref (operation_result);
> -        return;
> +    if (!mm_base_modem_at_command_finish (self, res, &error)) {
> +        mm_warn ("Couldn't disable ^SIND psinfo notifications: %s", error->message);
> +        g_error_free (error);
>      }
>  
> -    /* The AT^SIND? command replies a list of several different indicators.
> -     * We will only look for 'psinfo' which is the one which may tell us
> -     * the available network access technology. Note that only 3G-enabled
> -     * devices seem to have this indicator.
> -     *
> -     * AT+SIND?
> -     * ^SIND: battchg,1,1
> -     * ^SIND: signal,1,99
> -     * ...
> -     */
> -    regex = g_regex_new ("\\r\\n\\^SIND:\\s*psinfo,\\s*(\\d*),\\s*(\\d*)", 0, 0, NULL);
> -    if (g_regex_match_full (regex, response, strlen (response), 0, 0, &match_info, NULL)) {
> -        MMModemAccessTechnology act;
> -        gchar *ind_value;
> -
> -        ind_value = g_match_info_fetch (match_info, 2);
> -        act = get_access_technology_from_psinfo (ind_value, &error);
> -        g_free (ind_value);
> -        g_simple_async_result_set_op_res_gpointer (operation_result, GUINT_TO_POINTER (act), NULL);
> -        g_simple_async_result_complete (operation_result);
> -        g_object_unref (operation_result);
> +    /* Chain up parent's disabñe */
> +    iface_modem_3gpp_parent->disable_unsolicited_events (
> +        MM_IFACE_MODEM_3GPP (self),
> +        (GAsyncReadyCallback)parent_disable_unsolicited_events_ready,
> +        task);
> +}
> +
> +static void
> +modem_3gpp_disable_unsolicited_events (MMIfaceModem3gpp    *self,
> +                                       GAsyncReadyCallback  callback,
> +                                       gpointer             user_data)
> +{
> +    GTask *task;
> +
> +    task = g_task_new (self, NULL, callback, user_data);
> +
> +    /* Disable access technology update reporting */
> +    mm_base_modem_at_command (MM_BASE_MODEM (self),
> +                              "AT^SIND=\"psinfo\",0",
> +                              3,
> +                              FALSE,
> +                              (GAsyncReadyCallback)sind_psinfo_disable_ready,
> +                              task);

Should we even bother disabling if self->priv->sind_psinfo==FALSE? 
Could just jump to parent instead.  That'll save a useless log message
on unsolicited disable if we couldn't enable it in the first place.

> +}
> +
> +/*****************************************************************************/
> +/* Enable unsolicited events (3GPP interface) */
> +
> +static gboolean
> +modem_3gpp_enable_unsolicited_events_finish (MMIfaceModem3gpp  *self,
> +                                             GAsyncResult      *res,
> +                                             GError           **error)
> +{
> +    return g_task_propagate_boolean (G_TASK (res), error);
> +}
> +
> +static void
> +sind_psinfo_enable_ready (MMBaseModem  *_self,
> +                          GAsyncResult *res,
> +                          GTask        *task)
> +{
> +    MMBroadbandModemCinterion *self;
> +    GError                    *error = NULL;
> +    const gchar               *response;
> +    guint                      mode;
> +    guint                      val;
> +
> +    self = MM_BROADBAND_MODEM_CINTERION (_self);
> +    if (!(response = mm_base_modem_at_command_finish (_self, res, &error))) {
> +        mm_warn ("Couldn't enable ^SIND psinfo notifications: %s", error->message);
> +        g_error_free (error);
> +    } else if (!mm_cinterion_parse_sind_response (response, NULL, &mode, &val, &error)) {
> +        mm_warn ("Couldn't parse ^SIND psinfo response: %s", error->message);
> +        g_error_free (error);
>      } else {
> -        /* If there was no 'psinfo' indicator, we'll try AT^SMONG and read the cell
> -         * info table. */
> -        mm_base_modem_at_command (
> -            MM_BASE_MODEM (self),
> -            "^SMONG",
> -            3,
> -            FALSE,
> -            (GAsyncReadyCallback)smong_query_ready,
> -            operation_result);
> +        /* If enabled the indications, flag ^SIND psinfo supported so that we don't poll */
> +        self->priv->sind_psinfo = TRUE;
> +
> +        /* Report initial access technology gathered right away */
> +        mm_dbg ("Reporting initial access technologies...");
> +        mm_iface_modem_update_access_technologies (MM_IFACE_MODEM (self),
> +                                                   mm_cinterion_get_access_technology_from_sind_psinfo (val),
> +                                                   MM_IFACE_MODEM_3GPP_ALL_ACCESS_TECHNOLOGIES_MASK);
>      }
>  
> -    g_match_info_free (match_info);
> -    g_regex_unref (regex);
> +    g_task_return_boolean (task, TRUE);
> +    g_object_unref (task);
>  }
>  
>  static void
> -load_access_technologies (MMIfaceModem *self,
> -                          GAsyncReadyCallback callback,
> -                          gpointer user_data)
> +parent_enable_unsolicited_events_ready (MMIfaceModem3gpp *self,
> +                                        GAsyncResult     *res,
> +                                        GTask            *task)
>  {
> -    MMBroadbandModemCinterion *broadband = MM_BROADBAND_MODEM_CINTERION (self);
> -    GSimpleAsyncResult *result;
> +    GError *error = NULL;
>  
> -    result = g_simple_async_result_new (G_OBJECT (self),
> -                                        callback,
> -                                        user_data,
> -                                        load_access_technologies);
> +    if (!iface_modem_3gpp_parent->enable_unsolicited_events_finish (self, res, &error)) {
> +        mm_warn ("Couldn't enable parent 3GPP unsolicited events: %s", error->message);
> +        g_error_free (error);
> +    }
>  
> -    if (broadband->priv->sind_psinfo) {
> -        /* TODO: Trigger off psinfo URC instead of this polling. */
> -        mm_base_modem_at_command (
> -            MM_BASE_MODEM (self),
> -            "^SIND?",
> -            3,
> -            FALSE,
> -            (GAsyncReadyCallback)sind_query_ready,
> -            result);
> +    /* Enable access technology update reporting */
> +    mm_base_modem_at_command (MM_BASE_MODEM (self),
> +                              "AT^SIND=\"psinfo\",1",
> +                              3,
> +                              FALSE,
> +                              (GAsyncReadyCallback)sind_psinfo_enable_ready,
> +                              task);
> +}
> +
> +static void
> +modem_3gpp_enable_unsolicited_events (MMIfaceModem3gpp    *self,
> +                                      GAsyncReadyCallback  callback,
> +                                      gpointer             user_data)
> +{
> +    GTask *task;
> +
> +    task = g_task_new (self, NULL, callback, user_data);
> +
> +    /* Chain up parent's enable */
> +    iface_modem_3gpp_parent->enable_unsolicited_events (
> +        self,
> +        (GAsyncReadyCallback)parent_enable_unsolicited_events_ready,
> +        task);
> +}
> +
> +/*****************************************************************************/
> +/* Setup/Cleanup unsolicited events (3GPP interface) */
> +
> +static void
> +sind_psinfo_received (MMPortSerialAt            *port,
> +                      GMatchInfo                *match_info,
> +                      MMBroadbandModemCinterion *self)
> +{
> +    guint val;
> +
> +    if (!mm_get_uint_from_match_info (match_info, 1, &val)) {
> +        mm_dbg ("Failed to convert psinfo value");
>          return;
>      }
>  
> -    mm_base_modem_at_command (
> -        MM_BASE_MODEM (self),
> -        "^SMONG",
> -        3,
> -        FALSE,
> -        (GAsyncReadyCallback)smong_query_ready,
> -        result);
> +    mm_iface_modem_update_access_technologies (MM_IFACE_MODEM (self),
> +                                               mm_cinterion_get_access_technology_from_sind_psinfo (val),
> +                                               MM_IFACE_MODEM_3GPP_ALL_ACCESS_TECHNOLOGIES_MASK);
> +}
> +
> +static void
> +set_unsolicited_events_handlers (MMBroadbandModemCinterion *self,
> +                                 gboolean                   enable)
> +{
> +    MMPortSerialAt *ports[2];
> +    guint           i;
> +
> +    ports[0] = mm_base_modem_peek_port_primary   (MM_BASE_MODEM (self));
> +    ports[1] = mm_base_modem_peek_port_secondary (MM_BASE_MODEM (self));
> +
> +    /* Enable unsolicited events in given port */
> +    for (i = 0; i < 2; i++) {

G_N_ELEMENTS()?

Dan

> +        if (!ports[i])
> +            continue;
> +
> +        mm_port_serial_at_add_unsolicited_msg_handler (
> +            ports[i],
> +            self->priv->ciev_psinfo_regex,
> +            enable ? (MMPortSerialAtUnsolicitedMsgFn)sind_psinfo_received : NULL,
> +            enable ? self : NULL,
> +            NULL);
> +    }
> +}
> +
> +static gboolean
> +modem_3gpp_setup_cleanup_unsolicited_events_finish (MMIfaceModem3gpp  *self,
> +                                                    GAsyncResult      *res,
> +                                                    GError           **error)
> +{
> +    return g_task_propagate_boolean (G_TASK (res), error);
> +}
> +
> +static void
> +parent_setup_unsolicited_events_ready (MMIfaceModem3gpp *self,
> +                                       GAsyncResult     *res,
> +                                       GTask            *task)
> +{
> +    GError *error = NULL;
> +
> +    if (!iface_modem_3gpp_parent->setup_unsolicited_events_finish (self, res, &error))
> +        g_task_return_error (task, error);
> +    else {
> +        /* Our own setup now */
> +        set_unsolicited_events_handlers (MM_BROADBAND_MODEM_CINTERION (self), TRUE);
> +        g_task_return_boolean (task, TRUE);
> +    }
> +    g_object_unref (task);
> +}
> +
> +static void
> +modem_3gpp_setup_unsolicited_events (MMIfaceModem3gpp    *self,
> +                                     GAsyncReadyCallback  callback,
> +                                     gpointer             user_data)
> +{
> +    GTask *task;
> +
> +    task = g_task_new (self, NULL, callback, user_data);
> +
> +    /* Chain up parent's setup */
> +    iface_modem_3gpp_parent->setup_unsolicited_events (
> +        self,
> +        (GAsyncReadyCallback)parent_setup_unsolicited_events_ready,
> +        task);
> +}
> +
> +static void
> +parent_cleanup_unsolicited_events_ready (MMIfaceModem3gpp *self,
> +                                         GAsyncResult     *res,
> +                                         GTask            *task)
> +{
> +    GError *error = NULL;
> +
> +    if (!iface_modem_3gpp_parent->cleanup_unsolicited_events_finish (self, res, &error))
> +        g_task_return_error (task, error);
> +    else
> +        g_task_return_boolean (task, TRUE);
> +    g_object_unref (task);
> +}
> +
> +static void
> +modem_3gpp_cleanup_unsolicited_events (MMIfaceModem3gpp    *self,
> +                                       GAsyncReadyCallback  callback,
> +                                       gpointer             user_data)
> +{
> +    GTask *task;
> +
> +    task = g_task_new (self, NULL, callback, user_data);
> +
> +    /* Our own cleanup first */
> +    set_unsolicited_events_handlers (MM_BROADBAND_MODEM_CINTERION (self), FALSE);
> +
> +    /* And now chain up parent's cleanup */
> +    iface_modem_3gpp_parent->cleanup_unsolicited_events (
> +        self,
> +        (GAsyncReadyCallback)parent_cleanup_unsolicited_events_ready,
> +        task);
>  }
>  
>  /*****************************************************************************/
> @@ -1733,6 +1880,9 @@ mm_broadband_modem_cinterion_init (MMBroadbandModemCinterion *self)
>      /* Initialize private variables */
>      self->priv->sind_psinfo = TRUE; /* Initially, always try to get psinfo */
>      self->priv->swwan_support = FEATURE_SUPPORT_UNKNOWN;
> +
> +    self->priv->ciev_psinfo_regex = g_regex_new ("\\r\\n\\+CIEV: psinfo,(\\d+)\\r\\n",
> +                                                 G_REGEX_RAW | G_REGEX_OPTIMIZE, 0, NULL);
>  }
>  
>  static void
> @@ -1754,6 +1904,8 @@ finalize (GObject *object)
>      if (self->priv->cnmi_supported_bfr)
>          g_array_unref (self->priv->cnmi_supported_bfr);
>  
> +    g_regex_unref (self->priv->ciev_psinfo_regex);
> +
>      G_OBJECT_CLASS (mm_broadband_modem_cinterion_parent_class)->finalize (object);
>  }
>  
> @@ -1792,6 +1944,17 @@ static void
>  iface_modem_3gpp_init (MMIfaceModem3gpp *iface)
>  {
>      iface_modem_3gpp_parent = g_type_interface_peek_parent (iface);
> +
> +    iface->enable_unsolicited_events = modem_3gpp_enable_unsolicited_events;
> +    iface->enable_unsolicited_events_finish = modem_3gpp_enable_unsolicited_events_finish;
> +    iface->disable_unsolicited_events = modem_3gpp_disable_unsolicited_events;
> +    iface->disable_unsolicited_events_finish = modem_3gpp_disable_unsolicited_events_finish;
> +
> +    iface->setup_unsolicited_events = modem_3gpp_setup_unsolicited_events;
> +    iface->setup_unsolicited_events_finish = modem_3gpp_setup_cleanup_unsolicited_events_finish;
> +    iface->cleanup_unsolicited_events = modem_3gpp_cleanup_unsolicited_events;
> +    iface->cleanup_unsolicited_events_finish = modem_3gpp_setup_cleanup_unsolicited_events_finish;
> +
>      iface->register_in_network = register_in_network;
>      iface->register_in_network_finish = register_in_network_finish;
>  }
> diff --git a/plugins/cinterion/mm-modem-helpers-cinterion.c b/plugins/cinterion/mm-modem-helpers-cinterion.c
> index da1e7787..3c7a0a0c 100644
> --- a/plugins/cinterion/mm-modem-helpers-cinterion.c
> +++ b/plugins/cinterion/mm-modem-helpers-cinterion.c
> @@ -624,3 +624,36 @@ mm_cinterion_parse_smong_response (const gchar              *response,
>      g_assert (access_tech != MM_MODEM_ACCESS_TECHNOLOGY_UNKNOWN);
>      return TRUE;
>  }
> +
> +/*****************************************************************************/
> +/* ^SIND psinfo helper */
> +
> +MMModemAccessTechnology
> +mm_cinterion_get_access_technology_from_sind_psinfo (guint val)
> +{
> +    switch (val) {
> +    case 0:
> +        return MM_MODEM_ACCESS_TECHNOLOGY_UNKNOWN;
> +    case 1:
> +    case 2:
> +        return MM_MODEM_ACCESS_TECHNOLOGY_GPRS;
> +    case 3:
> +    case 4:
> +        return MM_MODEM_ACCESS_TECHNOLOGY_EDGE;
> +    case 5:
> +    case 6:
> +        return MM_MODEM_ACCESS_TECHNOLOGY_UMTS;
> +    case 7:
> +    case 8:
> +        return MM_MODEM_ACCESS_TECHNOLOGY_HSDPA;
> +    case 9:
> +    case 10:
> +        return (MM_MODEM_ACCESS_TECHNOLOGY_HSDPA | MM_MODEM_ACCESS_TECHNOLOGY_HSUPA);
> +    case 16:
> +    case 17:
> +        return MM_MODEM_ACCESS_TECHNOLOGY_LTE;
> +    default:
> +        mm_dbg ("Unable to identify access technology from psinfo reported value: %u", val);
> +        return MM_MODEM_ACCESS_TECHNOLOGY_UNKNOWN;
> +    }
> +}
> diff --git a/plugins/cinterion/mm-modem-helpers-cinterion.h b/plugins/cinterion/mm-modem-helpers-cinterion.h
> index bcf02f9d..2ec05157 100644
> --- a/plugins/cinterion/mm-modem-helpers-cinterion.h
> +++ b/plugins/cinterion/mm-modem-helpers-cinterion.h
> @@ -82,4 +82,9 @@ gboolean mm_cinterion_parse_smong_response (const gchar              *response,
>                                              MMModemAccessTechnology  *access_tech,
>                                              GError                  **error);
>  
> +/*****************************************************************************/
> +/* ^SIND psinfo helper */
> +
> +MMModemAccessTechnology mm_cinterion_get_access_technology_from_sind_psinfo (guint val);
> +
>  #endif  /* MM_MODEM_HELPERS_CINTERION_H */


More information about the ModemManager-devel mailing list