summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDan Williams <dcbw@redhat.com>2016-05-27 12:39:56 -0500
committerDan Williams <dcbw@redhat.com>2016-05-27 12:39:56 -0500
commit55f3ab80835114342618083664c4fa585455b9e0 (patch)
treeb4444c6b12eeb9233f8f73c71dfd1c9892eb04c7
parent830b6ebca8e218efeecd211b7a058366ef26272e (diff)
parentba097f1a434761a5129c07b5e4533e57bda43f3e (diff)
downloadModemManager-55f3ab80835114342618083664c4fa585455b9e0.tar.gz
merge: fix various mbm-related connect/disconnect issues (fdo #95302) (fdo #95304)
https://bugs.freedesktop.org/show_bug.cgi?id=95302 https://bugs.freedesktop.org/show_bug.cgi?id=95304
-rw-r--r--plugins/mbm/mm-broadband-bearer-mbm.c456
1 files changed, 265 insertions, 191 deletions
diff --git a/plugins/mbm/mm-broadband-bearer-mbm.c b/plugins/mbm/mm-broadband-bearer-mbm.c
index d6e98dbc6..196604644 100644
--- a/plugins/mbm/mm-broadband-bearer-mbm.c
+++ b/plugins/mbm/mm-broadband-bearer-mbm.c
@@ -46,11 +46,44 @@ G_DEFINE_TYPE (MMBroadbandBearerMbm, mm_broadband_bearer_mbm, MM_TYPE_BROADBAND_
struct _MMBroadbandBearerMbmPrivate {
gpointer connect_pending;
- guint connect_pending_id;
- gulong connect_cancellable_id;
+ gpointer disconnect_pending;
};
/*****************************************************************************/
+
+static void dial_3gpp_report_connection_status (gpointer data,
+ MMBearerConnectionStatus status);
+static void disconnect_report_connection_status (gpointer data,
+ MMBearerConnectionStatus status);
+
+static void
+report_connection_status (MMBaseBearer *bearer,
+ MMBearerConnectionStatus status)
+{
+ MMBroadbandBearerMbm *self = MM_BROADBAND_BEARER_MBM (bearer);
+
+ g_assert (status == MM_BEARER_CONNECTION_STATUS_CONNECTED ||
+ status == MM_BEARER_CONNECTION_STATUS_DISCONNECTED);
+
+ mm_dbg ("Received unsolicited *E2NAP (%s)",
+ mm_bearer_connection_status_get_string (status));
+
+ if (self->priv->connect_pending) {
+ /* Save unsolicited status for the pending connection attempt */
+ dial_3gpp_report_connection_status (self->priv->connect_pending, status);
+ } else if (self->priv->disconnect_pending) {
+ /* Save unsolicited status for the pending disconnection attempt */
+ disconnect_report_connection_status (self->priv->disconnect_pending, status);
+ } else {
+ if (status == MM_BEARER_CONNECTION_STATUS_DISCONNECTED) {
+ MM_BASE_BEARER_CLASS (mm_broadband_bearer_mbm_parent_class)->report_connection_status (
+ bearer,
+ status);
+ }
+ }
+}
+
+/*****************************************************************************/
/* 3GPP Dialing (sub-step of the 3GPP Connection sequence) */
typedef struct {
@@ -62,14 +95,21 @@ typedef struct {
MMPort *data;
GSimpleAsyncResult *result;
guint poll_count;
+ guint poll_id;
+ MMBearerConnectionStatus e2nap_status;
} Dial3gppContext;
static void
dial_3gpp_context_complete_and_free (Dial3gppContext *ctx)
{
+ /* Clear bearer object pointer to this connect context */
+ if (ctx->self->priv->connect_pending == ctx)
+ ctx->self->priv->connect_pending = NULL;
+
g_simple_async_result_complete_in_idle (ctx->result);
- if (ctx->data)
- g_object_unref (ctx->data);
+ g_clear_object (&ctx->data);
+ if (ctx->poll_id)
+ g_source_remove (ctx->poll_id);
g_object_unref (ctx->cancellable);
g_object_unref (ctx->result);
g_object_unref (ctx->primary);
@@ -78,33 +118,6 @@ dial_3gpp_context_complete_and_free (Dial3gppContext *ctx)
g_slice_free (Dial3gppContext, ctx);
}
-static gboolean
-dial_3gpp_context_set_error_if_cancelled (Dial3gppContext *ctx,
- GError **error)
-{
- if (!g_cancellable_is_cancelled (ctx->cancellable))
- return FALSE;
-
- g_set_error (error,
- MM_CORE_ERROR,
- MM_CORE_ERROR_CANCELLED,
- "Dial operation has been cancelled");
- return TRUE;
-}
-
-static gboolean
-dial_3gpp_context_complete_and_free_if_cancelled (Dial3gppContext *ctx)
-{
- GError *error = NULL;
-
- if (!dial_3gpp_context_set_error_if_cancelled (ctx, &error))
- return FALSE;
-
- g_simple_async_result_take_error (ctx->result, error);
- dial_3gpp_context_complete_and_free (ctx);
- return TRUE;
-}
-
static MMPort *
dial_3gpp_finish (MMBroadbandBearer *self,
GAsyncResult *res,
@@ -117,127 +130,94 @@ dial_3gpp_finish (MMBroadbandBearer *self,
}
static void
-report_connection_status (MMBaseBearer *bearer,
- MMBearerConnectionStatus status)
+dial_3gpp_report_connection_status (gpointer data,
+ MMBearerConnectionStatus status)
{
- MMBroadbandBearerMbm *self = MM_BROADBAND_BEARER_MBM (bearer);
- Dial3gppContext *ctx;
-
- g_assert (status == MM_BEARER_CONNECTION_STATUS_CONNECTED ||
- status == MM_BEARER_CONNECTION_STATUS_DISCONNECTED);
+ Dial3gppContext *ctx = data;
- /* Recover context (if any) and remove both cancellation and timeout (if any)*/
- ctx = self->priv->connect_pending;
- self->priv->connect_pending = NULL;
-
- /* Connection status reported but no connection attempt? */
- if (!ctx) {
- g_assert (self->priv->connect_pending_id == 0);
-
- mm_dbg ("Received spontaneous *E2NAP (%s)",
- mm_bearer_connection_status_get_string (status));
-
- if (status == MM_BEARER_CONNECTION_STATUS_DISCONNECTED) {
- /* If no connection attempt on-going, make sure we mark ourselves as
- * disconnected */
- MM_BASE_BEARER_CLASS (mm_broadband_bearer_mbm_parent_class)->report_connection_status (
- bearer,
- status);
- }
- return;
- }
+ g_assert (ctx);
+ ctx->e2nap_status = status;
+}
- if (self->priv->connect_pending_id) {
- g_source_remove (self->priv->connect_pending_id);
- self->priv->connect_pending_id = 0;
- }
+static void
+connect_error_disconnect_ready (MMBroadbandBearer *self,
+ GAsyncResult *res,
+ Dial3gppContext *ctx)
+{
+ MM_BROADBAND_BEARER_GET_CLASS (self)->disconnect_3gpp_finish (
+ self,
+ res,
+ NULL);
+ dial_3gpp_context_complete_and_free (ctx);
+}
- if (self->priv->connect_cancellable_id) {
- g_cancellable_disconnect (ctx->cancellable,
- self->priv->connect_cancellable_id);
- self->priv->connect_cancellable_id = 0;
- }
+static void
+connect_error_disconnect_start (Dial3gppContext *ctx)
+{
+ /* We don't care about connect status anymore */
+ if (ctx->self->priv->connect_pending == ctx)
+ ctx->self->priv->connect_pending = NULL;
+
+ MM_BROADBAND_BEARER_GET_CLASS (ctx->self)->disconnect_3gpp (
+ MM_BROADBAND_BEARER (ctx->self),
+ MM_BROADBAND_MODEM (ctx->modem),
+ ctx->primary,
+ NULL,
+ ctx->data,
+ ctx->cid,
+ (GAsyncReadyCallback) connect_error_disconnect_ready,
+ ctx);
+}
- /* Reporting connected */
- if (status == MM_BEARER_CONNECTION_STATUS_CONNECTED) {
+static gboolean
+handle_e2nap_connect_status (Dial3gppContext *ctx)
+{
+ switch (ctx->e2nap_status) {
+ case MM_BEARER_CONNECTION_STATUS_CONNECTED:
+ /* Reporting connected */
+ mm_dbg ("Connected status indicated already by an unsolicited message");
g_simple_async_result_set_op_res_gpointer (ctx->result,
g_object_ref (ctx->data),
(GDestroyNotify)g_object_unref);
dial_3gpp_context_complete_and_free (ctx);
- return;
+ return TRUE;
+ case MM_BEARER_CONNECTION_STATUS_DISCONNECTED:
+ /* Reporting disconnected */
+ mm_dbg ("Connection failure status indicated already by an unsolicited message");
+ g_simple_async_result_set_error (ctx->result,
+ MM_CORE_ERROR,
+ MM_CORE_ERROR_FAILED,
+ "Call setup failed");
+ dial_3gpp_context_complete_and_free (ctx);
+ return TRUE;
+ default:
+ break;
}
- /* Reporting disconnected */
- g_simple_async_result_set_error (ctx->result,
- MM_CORE_ERROR,
- MM_CORE_ERROR_FAILED,
- "Call setup failed");
- dial_3gpp_context_complete_and_free (ctx);
+ return FALSE;
}
-static void
-connect_cancelled_cb (GCancellable *cancellable,
- MMBroadbandBearerMbm *self)
-{
- GError *error = NULL;
- Dial3gppContext *ctx;
-
- /* Recover context and remove timeout */
- ctx = self->priv->connect_pending;
-
- g_source_remove (self->priv->connect_pending_id);
-
- self->priv->connect_pending = NULL;
- self->priv->connect_pending_id = 0;
- self->priv->connect_cancellable_id = 0;
-
- g_assert (dial_3gpp_context_set_error_if_cancelled (ctx, &error));
-
- g_simple_async_result_take_error (ctx->result, error);
- dial_3gpp_context_complete_and_free (ctx);
-}
-
-static gboolean poll_timeout_cb (MMBroadbandBearerMbm *self);
+static gboolean connect_poll_cb (Dial3gppContext *ctx);
static void
-poll_ready (MMBaseModem *modem,
- GAsyncResult *res,
- MMBroadbandBearerMbm *self)
+connect_poll_ready (MMBaseModem *modem,
+ GAsyncResult *res,
+ Dial3gppContext *ctx)
{
- Dial3gppContext *ctx;
GError *error = NULL;
const gchar *response;
guint state;
- /* Try to recover the connection context. If none found, it means the
- * context was already completed and we have nothing else to do. */
- ctx = self->priv->connect_pending;
-
- /* Balance refcount with the extra ref we passed to command_full() */
- g_object_unref (self);
-
- if (!ctx) {
- mm_dbg ("Connection context was finished already by an unsolicited message");
-
- /* Run _finish() to finalize the async call, even if we don't care
- * the result */
- mm_base_modem_at_command_full_finish (modem, res, NULL);
+ response = mm_base_modem_at_command_full_finish (modem, res, &error);
+ if (!response) {
+ g_simple_async_result_take_error (ctx->result, error);
+ connect_error_disconnect_start (ctx);
return;
}
- response = mm_base_modem_at_command_full_finish (modem, res, &error);
- if (response
- && sscanf (response, "*ENAP: %d", &state) == 1
+ if (sscanf (response, "*ENAP: %d", &state) == 1
&& state == 1) {
/* Success! Connected... */
- self->priv->connect_pending = NULL;
-
- if (ctx && self->priv->connect_cancellable_id) {
- g_cancellable_disconnect (ctx->cancellable,
- self->priv->connect_cancellable_id);
- self->priv->connect_cancellable_id = 0;
- }
-
g_simple_async_result_set_op_res_gpointer (ctx->result,
g_object_ref (ctx->data),
(GDestroyNotify)g_object_unref);
@@ -245,33 +225,43 @@ poll_ready (MMBaseModem *modem,
return;
}
- self->priv->connect_pending_id = g_timeout_add_seconds (1,
- (GSourceFunc)poll_timeout_cb,
- self);
+ /* Process any unsolicited E2NAP disconnect notification */
+ if (handle_e2nap_connect_status (ctx))
+ return;
+
+ /* Check again in one second */
+ g_assert (ctx->poll_id == 0);
+ ctx->poll_id = g_timeout_add_seconds (1,
+ (GSourceFunc)connect_poll_cb,
+ ctx);
}
static gboolean
-poll_timeout_cb (MMBroadbandBearerMbm *self)
+connect_poll_cb (Dial3gppContext *ctx)
{
- Dial3gppContext *ctx;
+ ctx->poll_id = 0;
- /* Recover context */
- ctx = self->priv->connect_pending;
+ /* Complete if we were cancelled */
+ if (g_cancellable_is_cancelled (ctx->cancellable)) {
+ g_simple_async_result_set_error (ctx->result,
+ MM_CORE_ERROR,
+ MM_CORE_ERROR_CANCELLED,
+ "Dial operation has been cancelled");
+ connect_error_disconnect_start (ctx);
+ return G_SOURCE_REMOVE;
+ }
+
+ /* Process any unsolicited E2NAP status */
+ if (handle_e2nap_connect_status (ctx))
+ return G_SOURCE_REMOVE;
/* Too many retries... */
if (ctx->poll_count > 50) {
- g_cancellable_disconnect (ctx->cancellable,
- self->priv->connect_cancellable_id);
-
- self->priv->connect_pending = NULL;
- self->priv->connect_pending_id = 0;
- self->priv->connect_cancellable_id = 0;
-
g_simple_async_result_set_error (ctx->result,
MM_MOBILE_EQUIPMENT_ERROR,
MM_MOBILE_EQUIPMENT_ERROR_NETWORK_TIMEOUT,
"Connection attempt timed out");
- dial_3gpp_context_complete_and_free (ctx);
+ connect_error_disconnect_start (ctx);
return G_SOURCE_REMOVE;
}
@@ -282,56 +272,37 @@ poll_timeout_cb (MMBroadbandBearerMbm *self)
3,
FALSE,
FALSE, /* raw */
- NULL, /* cancellable */
- (GAsyncReadyCallback)poll_ready,
- g_object_ref (ctx->self)); /* we pass the bearer object! */
- self->priv->connect_pending_id = 0;
+ ctx->cancellable,
+ (GAsyncReadyCallback)connect_poll_ready,
+ ctx);
return G_SOURCE_REMOVE;
}
static void
activate_ready (MMBaseModem *modem,
GAsyncResult *res,
- MMBroadbandBearerMbm *self)
+ Dial3gppContext *ctx)
{
- Dial3gppContext *ctx;
GError *error = NULL;
- /* Try to recover the connection context. If none found, it means the
- * context was already completed and we have nothing else to do. */
- ctx = self->priv->connect_pending;
-
- /* Balance refcount with the extra ref we passed to command_full() */
- g_object_unref (self);
-
- if (!ctx) {
- mm_dbg ("Connection context was finished already by an unsolicited message");
-
- /* Run _finish() to finalize the async call, even if we don't care
- * the result */
- mm_base_modem_at_command_full_finish (modem, res, NULL);
- return;
- }
-
/* From now on, if we get cancelled, we'll need to run the connection
* reset ourselves just in case */
if (!mm_base_modem_at_command_full_finish (modem, res, &error)) {
- self->priv->connect_pending = NULL;
g_simple_async_result_take_error (ctx->result, error);
- dial_3gpp_context_complete_and_free (ctx);
+ connect_error_disconnect_start (ctx);
return;
}
- /* We will now setup a timeout to poll for the status */
- self->priv->connect_pending_id = g_timeout_add_seconds (1,
- (GSourceFunc)poll_timeout_cb,
- self);
+ /* Process any unsolicited E2NAP status received before the ENAP OK */
+ if (handle_e2nap_connect_status (ctx))
+ return;
- self->priv->connect_cancellable_id = g_cancellable_connect (ctx->cancellable,
- G_CALLBACK (connect_cancelled_cb),
- self,
- NULL);
+ /* No unsolicited E2NAP status yet; wait for it and periodically poll
+ * to handle very old F3507g/MD300 firmware that may not send E2NAP. */
+ ctx->poll_id = g_timeout_add_seconds (1,
+ (GSourceFunc)connect_poll_cb,
+ ctx);
}
static void
@@ -341,25 +312,21 @@ activate (Dial3gppContext *ctx)
/* The unsolicited response to ENAP may come before the OK does.
* We will keep the connection context in the bearer private data so
- * that it is accessible from the unsolicited message handler. Note
- * also that we do NOT pass the ctx to the GAsyncReadyCallback, as it
- * may not be valid any more when the callback is called (it may be
- * already completed in the unsolicited handling) */
+ * that it is accessible from the unsolicited message handler. */
g_assert (ctx->self->priv->connect_pending == NULL);
ctx->self->priv->connect_pending = ctx;
- /* Success, activate the PDP context and start the data session */
- command = g_strdup_printf ("AT*ENAP=1,%d",
- ctx->cid);
+ /* Activate the PDP context and start the data session */
+ command = g_strdup_printf ("AT*ENAP=1,%d", ctx->cid);
mm_base_modem_at_command_full (ctx->modem,
ctx->primary,
command,
3,
FALSE,
FALSE, /* raw */
- NULL, /* cancellable */
+ ctx->cancellable,
(GAsyncReadyCallback)activate_ready,
- g_object_ref (ctx->self)); /* we pass the bearer object! */
+ ctx);
g_free (command);
}
@@ -370,10 +337,6 @@ authenticate_ready (MMBaseModem *modem,
{
GError *error = NULL;
- /* If cancelled, complete */
- if (dial_3gpp_context_complete_and_free_if_cancelled (ctx))
- return;
-
if (!mm_base_modem_at_command_full_finish (modem, res, &error)) {
g_simple_async_result_take_error (ctx->result, error);
dial_3gpp_context_complete_and_free (ctx);
@@ -416,7 +379,7 @@ authenticate (Dial3gppContext *ctx)
3,
FALSE,
FALSE, /* raw */
- NULL, /* cancellable */
+ ctx->cancellable,
(GAsyncReadyCallback)authenticate_ready,
ctx);
g_free (command);
@@ -633,12 +596,21 @@ typedef struct {
MMBaseModem *modem;
MMPortSerialAt *primary;
GSimpleAsyncResult *result;
+ guint poll_count;
+ guint poll_id;
+ MMBearerConnectionStatus e2nap_status;
} DisconnectContext;
static void
disconnect_context_complete_and_free (DisconnectContext *ctx)
{
- g_simple_async_result_complete (ctx->result);
+ /* Clear bearer object pointer to this disconnect context */
+ if (ctx->self->priv->disconnect_pending == ctx)
+ ctx->self->priv->disconnect_pending = NULL;
+
+ g_simple_async_result_complete_in_idle (ctx->result);
+ if (ctx->poll_id)
+ g_source_remove (ctx->poll_id);
g_object_unref (ctx->result);
g_object_unref (ctx->primary);
g_object_unref (ctx->self);
@@ -655,6 +627,95 @@ disconnect_3gpp_finish (MMBroadbandBearer *self,
}
static void
+disconnect_report_connection_status (gpointer data,
+ MMBearerConnectionStatus status)
+{
+ DisconnectContext *ctx = data;
+
+ g_assert (ctx);
+ ctx->e2nap_status = status;
+}
+
+static gboolean
+handle_e2nap_disconnect_status (DisconnectContext *ctx)
+{
+ if (ctx->e2nap_status == MM_BEARER_CONNECTION_STATUS_DISCONNECTED) {
+ /* Reporting disconnected */
+ mm_dbg ("Connection disconnect indicated already by an unsolicited message");
+ g_simple_async_result_set_op_res_gboolean (ctx->result, TRUE);
+ disconnect_context_complete_and_free (ctx);
+ return TRUE;
+ }
+
+ return FALSE;
+}
+
+static gboolean disconnect_poll_cb (DisconnectContext *ctx);
+
+static void
+disconnect_poll_ready (MMBaseModem *modem,
+ GAsyncResult *res,
+ DisconnectContext *ctx)
+{
+ GError *error = NULL;
+ const gchar *response;
+ guint state;
+
+ response = mm_base_modem_at_command_full_finish (modem, res, &error);
+ if (!response) {
+ g_simple_async_result_take_error (ctx->result, error);
+ disconnect_context_complete_and_free (ctx);
+ return;
+ }
+
+ if (sscanf (response, "*ENAP: %d", &state) == 1
+ && state == 0) {
+ /* Disconnected */
+ g_simple_async_result_set_op_res_gboolean (ctx->result, TRUE);
+ disconnect_context_complete_and_free (ctx);
+ return;
+ }
+
+ /* Check again in one second */
+ g_assert (ctx->poll_id == 0);
+ ctx->poll_id = g_timeout_add_seconds (1,
+ (GSourceFunc)disconnect_poll_cb,
+ ctx);
+}
+
+static gboolean
+disconnect_poll_cb (DisconnectContext *ctx)
+{
+ ctx->poll_id = 0;
+
+ /* Process any unsolicited E2NAP status */
+ if (handle_e2nap_disconnect_status (ctx))
+ return G_SOURCE_REMOVE;
+
+ /* Too many retries... */
+ if (ctx->poll_count > 20) {
+ g_simple_async_result_set_error (ctx->result,
+ MM_MOBILE_EQUIPMENT_ERROR,
+ MM_MOBILE_EQUIPMENT_ERROR_NETWORK_TIMEOUT,
+ "Disconnection attempt timed out");
+ disconnect_context_complete_and_free (ctx);
+ return G_SOURCE_REMOVE;
+ }
+
+ ctx->poll_count++;
+ mm_base_modem_at_command_full (ctx->modem,
+ ctx->primary,
+ "AT*ENAP?",
+ 3,
+ FALSE,
+ FALSE, /* raw */
+ NULL, /* cancellable */
+ (GAsyncReadyCallback)disconnect_poll_ready,
+ ctx);
+ return G_SOURCE_REMOVE;
+}
+
+static void
disconnect_enap_ready (MMBaseModem *modem,
GAsyncResult *res,
DisconnectContext *ctx)
@@ -668,8 +729,15 @@ disconnect_enap_ready (MMBaseModem *modem,
g_error_free (error);
}
- g_simple_async_result_set_op_res_gboolean (ctx->result, TRUE);
- disconnect_context_complete_and_free (ctx);
+ /* Process any unsolicited E2NAP status received before the ENAP OK */
+ if (handle_e2nap_disconnect_status (ctx))
+ return;
+
+ /* No unsolicited E2NAP status yet; wait for it and periodically poll
+ * to handle very old F3507g/MD300 firmware that may not send E2NAP. */
+ ctx->poll_id = g_timeout_add_seconds (1,
+ (GSourceFunc)disconnect_poll_cb,
+ ctx);
}
static void
@@ -695,6 +763,12 @@ disconnect_3gpp (MMBroadbandBearer *self,
user_data,
disconnect_3gpp);
+ /* The unsolicited response to ENAP may come before the OK does.
+ * We will keep the disconnection context in the bearer private data so
+ * that it is accessible from the unsolicited message handler. */
+ g_assert (ctx->self->priv->disconnect_pending == NULL);
+ ctx->self->priv->disconnect_pending = ctx;
+
mm_base_modem_at_command_full (MM_BASE_MODEM (modem),
primary,
"*ENAP=0",