diff options
Diffstat (limited to 'src/plugins/sierra/mm-common-sierra.c')
-rw-r--r-- | src/plugins/sierra/mm-common-sierra.c | 516 |
1 files changed, 516 insertions, 0 deletions
diff --git a/src/plugins/sierra/mm-common-sierra.c b/src/plugins/sierra/mm-common-sierra.c new file mode 100644 index 00000000..72cbc34f --- /dev/null +++ b/src/plugins/sierra/mm-common-sierra.c @@ -0,0 +1,516 @@ +/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ +/* + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program 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 General Public License for more details: + * + * Copyright (C) 2008 - 2009 Novell, Inc. + * Copyright (C) 2009 - 2012 Red Hat, Inc. + * Copyright (C) 2012 Lanedo GmbH + */ + +#include <stdlib.h> +#include <string.h> + +#include "mm-common-sierra.h" +#include "mm-base-modem-at.h" +#include "mm-log.h" +#include "mm-modem-helpers.h" +#include "mm-sim-sierra.h" + +static MMIfaceModem *iface_modem_parent; + +/*****************************************************************************/ +/* Custom init and port type hints */ + +#define TAG_SIERRA_APP_PORT "sierra-app-port" +#define TAG_SIERRA_APP1_PPP_OK "sierra-app1-ppp-ok" + +gboolean +mm_common_sierra_grab_port (MMPlugin *self, + MMBaseModem *modem, + MMPortProbe *probe, + GError **error) +{ + MMPortSerialAtFlag pflags = MM_PORT_SERIAL_AT_FLAG_NONE; + MMPortType ptype; + + ptype = mm_port_probe_get_port_type (probe); + + /* Is it a GSM secondary port? */ + if (g_object_get_data (G_OBJECT (probe), TAG_SIERRA_APP_PORT)) { + if (g_object_get_data (G_OBJECT (probe), TAG_SIERRA_APP1_PPP_OK)) + pflags = MM_PORT_SERIAL_AT_FLAG_PPP; + else + pflags = MM_PORT_SERIAL_AT_FLAG_SECONDARY; + } else if (ptype == MM_PORT_TYPE_AT) + pflags = MM_PORT_SERIAL_AT_FLAG_PRIMARY; + + return mm_base_modem_grab_port (modem, + mm_port_probe_peek_port (probe), + ptype, + pflags, + error); +} + +gboolean +mm_common_sierra_port_probe_list_is_icera (GList *probes) +{ + GList *l; + + for (l = probes; l; l = g_list_next (l)) { + /* Only assume the Icera probing check is valid IF the port is not + * secondary. This will skip the stupid ports which reply OK to every + * AT command, even the one we use to check for Icera support */ + if (mm_port_probe_is_icera (MM_PORT_PROBE (l->data)) && + !g_object_get_data (G_OBJECT (l->data), TAG_SIERRA_APP_PORT)) + return TRUE; + } + + return FALSE; +} + +typedef struct { + MMPortSerialAt *port; + guint retries; +} SierraCustomInitContext; + +static void +sierra_custom_init_context_free (SierraCustomInitContext *ctx) +{ + g_object_unref (ctx->port); + g_slice_free (SierraCustomInitContext, ctx); +} + +gboolean +mm_common_sierra_custom_init_finish (MMPortProbe *probe, + GAsyncResult *result, + GError **error) +{ + return g_task_propagate_boolean (G_TASK (result), error); +} + +static void sierra_custom_init_step (GTask *task); + +static void +gcap_ready (MMPortSerialAt *port, + GAsyncResult *res, + GTask *task) +{ + MMPortProbe *probe; + SierraCustomInitContext *ctx; + const gchar *response; + GError *error = NULL; + + probe = g_task_get_source_object (task); + ctx = g_task_get_task_data (task); + + response = mm_port_serial_at_command_finish (port, res, &error); + if (error) { + /* If consumed all tries and the last error was a timeout, assume the + * port is not AT */ + if (ctx->retries == 0 && + g_error_matches (error, MM_SERIAL_ERROR, MM_SERIAL_ERROR_RESPONSE_TIMEOUT)) { + mm_port_probe_set_result_at (probe, FALSE); + } + /* If reported a hard parse error, this port is definitely not an AT + * port, skip trying. */ + else if (g_error_matches (error, MM_SERIAL_ERROR, MM_SERIAL_ERROR_PARSE_FAILED)) { + mm_port_probe_set_result_at (probe, FALSE); + ctx->retries = 0; + } + /* Some Icera-based devices (eg, USB305) have an AT-style port that + * replies to everything with ERROR, so tag as unsupported; sometimes + * the real AT ports do this too, so let a retry tag the port as + * supported if it responds correctly later. */ + else if (g_error_matches (error, MM_MOBILE_EQUIPMENT_ERROR, MM_MOBILE_EQUIPMENT_ERROR_UNKNOWN)) { + mm_port_probe_set_result_at (probe, FALSE); + } + + /* Just retry... */ + sierra_custom_init_step (task); + goto out; + } + + /* A valid reply to ATI tells us this is an AT port already */ + mm_port_probe_set_result_at (probe, TRUE); + + /* Sierra APPx ports have limited AT command parsers that just reply with + * "OK" to most commands. These can sometimes be used for PPP while the + * main port is used for status and control, but older modems tend to crash + * or fail PPP. So we allowlist modems that are known to allow PPP on the + * secondary APP ports. + */ + if (strstr (response, "APP1")) { + g_object_set_data (G_OBJECT (probe), TAG_SIERRA_APP_PORT, GUINT_TO_POINTER (TRUE)); + + /* PPP-on-APP1-port allowlist */ + if (strstr (response, "C885") || + strstr (response, "USB 306") || + strstr (response, "MC8790")) + g_object_set_data (G_OBJECT (probe), TAG_SIERRA_APP1_PPP_OK, GUINT_TO_POINTER (TRUE)); + + /* For debugging: let users figure out if their device supports PPP + * on the APP1 port or not. + */ + if (getenv ("MM_SIERRA_APP1_PPP_OK")) { + mm_obj_dbg (probe, "APP1 PPP OK '%s'", response); + g_object_set_data (G_OBJECT (probe), TAG_SIERRA_APP1_PPP_OK, GUINT_TO_POINTER (TRUE)); + } + } else if (strstr (response, "APP2") || + strstr (response, "APP3") || + strstr (response, "APP4")) { + /* Additional APP ports don't support most AT commands, so they cannot + * be used as the primary port. + */ + g_object_set_data (G_OBJECT (probe), TAG_SIERRA_APP_PORT, GUINT_TO_POINTER (TRUE)); + } + + g_task_return_boolean (task, TRUE); + g_object_unref (task); + +out: + if (error) + g_error_free (error); +} + +static void +sierra_custom_init_step (GTask *task) +{ + MMPortProbe *probe; + SierraCustomInitContext *ctx; + GCancellable *cancellable; + + probe = g_task_get_source_object (task); + ctx = g_task_get_task_data (task); + cancellable = g_task_get_cancellable (task); + + /* If cancelled, end */ + if (g_cancellable_is_cancelled (cancellable)) { + mm_obj_dbg (probe, "no need to keep on running custom init"); + g_task_return_boolean (task, TRUE); + g_object_unref (task); + return; + } + + if (ctx->retries == 0) { + mm_obj_dbg (probe, "couldn't get port type hints"); + g_task_return_boolean (task, TRUE); + g_object_unref (task); + return; + } + + ctx->retries--; + mm_port_serial_at_command ( + ctx->port, + "ATI", + 3, + FALSE, /* raw */ + FALSE, /* allow_cached */ + cancellable, + (GAsyncReadyCallback)gcap_ready, + task); +} + +void +mm_common_sierra_custom_init (MMPortProbe *probe, + MMPortSerialAt *port, + GCancellable *cancellable, + GAsyncReadyCallback callback, + gpointer user_data) +{ + SierraCustomInitContext *ctx; + GTask *task; + + ctx = g_slice_new (SierraCustomInitContext); + ctx->port = g_object_ref (port); + ctx->retries = 3; + + task = g_task_new (probe, cancellable, callback, user_data); + g_task_set_check_cancellable (task, FALSE); + g_task_set_task_data (task, ctx, (GDestroyNotify)sierra_custom_init_context_free); + + sierra_custom_init_step (task); +} + +/*****************************************************************************/ +/* Modem power up (Modem interface) */ + +gboolean +mm_common_sierra_modem_power_up_finish (MMIfaceModem *self, + GAsyncResult *res, + GError **error) +{ + return g_task_propagate_boolean (G_TASK (res), error); +} + +static gboolean +sierra_power_up_wait_cb (GTask *task) +{ + g_task_return_boolean (task, TRUE); + g_object_unref (task); + return G_SOURCE_REMOVE; +} + +static void +cfun_enable_ready (MMBaseModem *self, + GAsyncResult *res, + GTask *task) +{ + GError *error = NULL; + guint i; + const gchar **drivers; + gboolean is_new_sierra = FALSE; + + if (!mm_base_modem_at_command_finish (MM_BASE_MODEM (self), res, &error)) { + g_task_return_error (task, error); + g_object_unref (task); + return; + } + + /* Many Sierra devices return OK immediately in response to CFUN=1 but + * need some time to finish powering up, otherwise subsequent commands + * may return failure or even crash the modem. Give more time for older + * devices like the AC860 and C885, which aren't driven by the 'sierra_net' + * driver. Assume any DirectIP (ie, sierra_net) device is new enough + * to allow a lower timeout. + */ + drivers = mm_base_modem_get_drivers (MM_BASE_MODEM (self)); + for (i = 0; drivers[i]; i++) { + if (g_str_equal (drivers[i], "sierra_net")) { + is_new_sierra = TRUE; + break; + } + } + + /* The modem object will be valid in the callback as 'task' keeps a + * reference to it. */ + g_timeout_add_seconds (is_new_sierra ? 5 : 10, (GSourceFunc)sierra_power_up_wait_cb, task); +} + +static void +pcstate_enable_ready (MMBaseModem *self, + GAsyncResult *res, + GTask *task) +{ + /* Ignore errors for now; we're not sure if all Sierra CDMA devices support + * at!pcstate. + */ + mm_base_modem_at_command_finish (MM_BASE_MODEM (self), res, NULL); + + g_task_return_boolean (task, TRUE); + g_object_unref (task); +} + +void +mm_common_sierra_modem_power_up (MMIfaceModem *self, + GAsyncReadyCallback callback, + gpointer user_data) +{ + GTask *task; + + task = g_task_new (self, NULL, callback, user_data); + + /* For CDMA modems, run !pcstate */ + if (mm_iface_modem_is_cdma_only (self)) { + mm_base_modem_at_command (MM_BASE_MODEM (self), + "!pcstate=1", + 5, + FALSE, + (GAsyncReadyCallback)pcstate_enable_ready, + task); + return; + } + + mm_obj_warn (self, "not in full functionality status, power-up command is needed"); + mm_obj_warn (self, "device may be rebooted"); + + /* Try to go to full functionality mode without rebooting the system. + * Works well if we previously switched off the power with CFUN=4 + */ + mm_base_modem_at_command (MM_BASE_MODEM (self), + "+CFUN=1,0", /* ",0" requests no reset */ + 10, + FALSE, + (GAsyncReadyCallback)cfun_enable_ready, + task); +} + +/*****************************************************************************/ +/* Power state loading (Modem interface) */ + +MMModemPowerState +mm_common_sierra_load_power_state_finish (MMIfaceModem *self, + GAsyncResult *res, + GError **error) +{ + GError *inner_error = NULL; + gssize value; + + value = g_task_propagate_int (G_TASK (res), &inner_error); + if (inner_error) { + g_propagate_error (error, inner_error); + return MM_MODEM_POWER_STATE_UNKNOWN; + } + return (MMModemPowerState)value; +} + +static void +parent_load_power_state_ready (MMIfaceModem *self, + GAsyncResult *res, + GTask *task) +{ + GError *error = NULL; + MMModemPowerState state; + + state = iface_modem_parent->load_power_state_finish (self, res, &error); + if (error) + g_task_return_error (task, error); + else + g_task_return_int (task, state); + + g_object_unref (task); +} + +static void +pcstate_query_ready (MMBaseModem *self, + GAsyncResult *res, + GTask *task) +{ + const gchar *result; + guint state; + GError *error = NULL; + + result = mm_base_modem_at_command_finish (MM_BASE_MODEM (self), res, &error); + if (!result) { + g_task_return_error (task, error); + g_object_unref (task); + return; + } + + /* Parse power state reply */ + result = mm_strip_tag (result, "!PCSTATE:"); + if (!mm_get_uint_from_str (result, &state)) { + g_task_return_new_error (task, + MM_CORE_ERROR, + MM_CORE_ERROR_FAILED, + "Failed to parse !PCSTATE response '%s'", + result); + } else { + switch (state) { + case 0: + g_task_return_int (task, MM_MODEM_POWER_STATE_LOW); + break; + case 1: + g_task_return_int (task, MM_MODEM_POWER_STATE_ON); + break; + default: + g_task_return_new_error (task, + MM_CORE_ERROR, + MM_CORE_ERROR_FAILED, + "Unhandled power state: '%u'", + state); + break; + } + } + + g_object_unref (task); +} + +void +mm_common_sierra_load_power_state (MMIfaceModem *self, + GAsyncReadyCallback callback, + gpointer user_data) +{ + GTask *task; + + task = g_task_new (self, NULL, callback, user_data); + + /* Check power state with AT!PCSTATE? */ + if (mm_iface_modem_is_cdma_only (self)) { + mm_base_modem_at_command (MM_BASE_MODEM (self), + "!pcstate?", + 3, + FALSE, + (GAsyncReadyCallback)pcstate_query_ready, + task); + return; + } + + /* Otherwise run parent's */ + iface_modem_parent->load_power_state (self, + (GAsyncReadyCallback)parent_load_power_state_ready, + task); +} + +/*****************************************************************************/ +/* Create SIM (Modem interface) */ + +MMBaseSim * +mm_common_sierra_create_sim_finish (MMIfaceModem *self, + GAsyncResult *res, + GError **error) +{ + return mm_sim_sierra_new_finish (res, error); +} + +void +mm_common_sierra_create_sim (MMIfaceModem *self, + GAsyncReadyCallback callback, + gpointer user_data) +{ + /* New Sierra SIM */ + mm_sim_sierra_new (MM_BASE_MODEM (self), + NULL, /* cancellable */ + callback, + user_data); +} + +/*****************************************************************************/ +/* Setup ports */ + +void +mm_common_sierra_setup_ports (MMBroadbandModem *self) +{ + MMPortSerialAt *ports[2]; + guint i; + g_autoptr(GRegex) pacsp_regex = NULL; + + pacsp_regex = g_regex_new ("\\r\\n\\+PACSP.*\\r\\n", G_REGEX_RAW | G_REGEX_OPTIMIZE, 0, NULL); + + ports[0] = mm_base_modem_peek_port_primary (MM_BASE_MODEM (self)); + ports[1] = mm_base_modem_peek_port_secondary (MM_BASE_MODEM (self)); + + for (i = 0; i < G_N_ELEMENTS (ports); i++) { + if (!ports[i]) + continue; + + if (i == 1) { + /* Built-in echo removal conflicts with the APP1 port's limited AT + * parser, which doesn't always prefix responses with <CR><LF>. + */ + g_object_set (ports[i], + MM_PORT_SERIAL_AT_REMOVE_ECHO, FALSE, + NULL); + } + + mm_port_serial_at_add_unsolicited_msg_handler ( + ports[i], + pacsp_regex, + NULL, NULL, NULL); + } +} + +/*****************************************************************************/ + +void +mm_common_sierra_peek_parent_interfaces (MMIfaceModem *iface) +{ + iface_modem_parent = g_type_interface_peek_parent (iface); +} |