| /* -*- 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) 2013-2021 Aleksander Morgado <aleksander@gnu.org> |
| */ |
| |
| #include <config.h> |
| |
| #include <stdio.h> |
| #include <stdlib.h> |
| |
| #if defined WITH_QMI |
| # include <libqmi-glib.h> |
| #endif |
| |
| #include <ModemManager.h> |
| #include <mm-errors-types.h> |
| |
| #include "mm-port-mbim.h" |
| #include "mm-port-net.h" |
| #include "mm-log-object.h" |
| |
| G_DEFINE_TYPE (MMPortMbim, mm_port_mbim, MM_TYPE_PORT) |
| |
| enum { |
| SIGNAL_NOTIFICATION, |
| SIGNAL_LAST |
| }; |
| |
| static guint signals[SIGNAL_LAST] = { 0 }; |
| |
| struct _MMPortMbimPrivate { |
| gboolean in_progress; |
| MbimDevice *mbim_device; |
| |
| /* monitoring */ |
| gulong notification_monitoring_id; |
| gulong timeout_monitoring_id; |
| gulong removed_monitoring_id; |
| |
| /* supported services info */ |
| MbimDeviceServiceElement **device_services; |
| guint32 device_services_count; |
| |
| #if defined WITH_QMI && QMI_MBIM_QMUX_SUPPORTED |
| QmiDevice *qmi_device; |
| GList *qmi_clients; |
| #endif |
| }; |
| |
| /*****************************************************************************/ |
| |
| #if defined WITH_QMI && QMI_MBIM_QMUX_SUPPORTED |
| |
| gboolean |
| mm_port_mbim_supports_qmi (MMPortMbim *self) |
| { |
| return !!self->priv->qmi_device; |
| } |
| |
| QmiClient * |
| mm_port_mbim_peek_qmi_client (MMPortMbim *self, |
| QmiService service) |
| { |
| GList *l; |
| |
| for (l = self->priv->qmi_clients; l; l = g_list_next (l)) { |
| QmiClient *qmi_client = QMI_CLIENT (l->data); |
| |
| if (qmi_client_get_service (qmi_client) == service) |
| return qmi_client; |
| } |
| |
| return NULL; |
| } |
| |
| QmiClient * |
| mm_port_mbim_get_qmi_client (MMPortMbim *self, |
| QmiService service) |
| { |
| QmiClient *client; |
| |
| client = mm_port_mbim_peek_qmi_client (self, service); |
| return (client ? g_object_ref (client) : NULL); |
| } |
| |
| gboolean |
| mm_port_mbim_allocate_qmi_client_finish (MMPortMbim *self, |
| GAsyncResult *res, |
| GError **error) |
| { |
| return g_task_propagate_boolean (G_TASK (res), error); |
| } |
| |
| static void |
| allocate_client_ready (QmiDevice *qmi_device, |
| GAsyncResult *res, |
| GTask *task) |
| { |
| MMPortMbim *self; |
| QmiClient *qmi_client; |
| GError *error = NULL; |
| |
| self = g_task_get_source_object (task); |
| qmi_client = qmi_device_allocate_client_finish (qmi_device, res, &error); |
| if (!qmi_client) { |
| g_prefix_error (&error, |
| "Couldn't create QMI client for service '%s': ", |
| qmi_service_get_string ((QmiService) GPOINTER_TO_INT (g_task_get_task_data (task)))); |
| g_task_return_error (task, error); |
| } else { |
| /* Store the client in our internal list */ |
| self->priv->qmi_clients = g_list_prepend (self->priv->qmi_clients, qmi_client); |
| g_task_return_boolean (task, TRUE); |
| } |
| g_object_unref (task); |
| } |
| |
| void |
| mm_port_mbim_allocate_qmi_client (MMPortMbim *self, |
| QmiService service, |
| GCancellable *cancellable, |
| GAsyncReadyCallback callback, |
| gpointer user_data) |
| { |
| GTask *task; |
| |
| task = g_task_new (self, cancellable, callback, user_data); |
| |
| if (!mm_port_mbim_is_open (self)) { |
| g_task_return_new_error (task, MM_CORE_ERROR, MM_CORE_ERROR_WRONG_STATE, |
| "Port is closed"); |
| g_object_unref (task); |
| return; |
| } |
| |
| if (!mm_port_mbim_supports_qmi (self)) { |
| g_task_return_new_error (task, MM_CORE_ERROR, MM_CORE_ERROR_UNSUPPORTED, |
| "Port doesn't support QMI over MBIM"); |
| g_object_unref (task); |
| return; |
| } |
| |
| if (!!mm_port_mbim_peek_qmi_client (self, service)) { |
| g_task_return_new_error (task, MM_CORE_ERROR, MM_CORE_ERROR_EXISTS, |
| "Client for service '%s' already allocated", |
| qmi_service_get_string (service)); |
| g_object_unref (task); |
| return; |
| } |
| |
| g_task_set_task_data (task, GINT_TO_POINTER (service), NULL); |
| qmi_device_allocate_client (self->priv->qmi_device, |
| service, |
| QMI_CID_NONE, |
| 10, |
| cancellable, |
| (GAsyncReadyCallback)allocate_client_ready, |
| task); |
| } |
| |
| #endif |
| |
| /*****************************************************************************/ |
| |
| typedef struct { |
| gchar *link_name; |
| guint session_id; |
| } SetupLinkResult; |
| |
| static void |
| setup_link_result_free (SetupLinkResult *ctx) |
| { |
| g_free (ctx->link_name); |
| g_slice_free (SetupLinkResult, ctx); |
| } |
| |
| gchar * |
| mm_port_mbim_setup_link_finish (MMPortMbim *self, |
| GAsyncResult *res, |
| guint *session_id, |
| GError **error) |
| { |
| SetupLinkResult *result; |
| gchar *link_name; |
| |
| result = g_task_propagate_pointer (G_TASK (res), error); |
| if (!result) |
| return NULL; |
| |
| if (session_id) |
| *session_id = result->session_id; |
| link_name = g_steal_pointer (&result->link_name); |
| setup_link_result_free (result); |
| |
| return link_name; |
| } |
| |
| static void |
| device_add_link_ready (MbimDevice *device, |
| GAsyncResult *res, |
| GTask *task) |
| { |
| SetupLinkResult *result; |
| GError *error = NULL; |
| |
| result = g_slice_new0 (SetupLinkResult); |
| |
| result->link_name = mbim_device_add_link_finish (device, res, &result->session_id, &error); |
| if (!result->link_name) { |
| g_prefix_error (&error, "failed to add link for device: "); |
| g_task_return_error (task, error); |
| setup_link_result_free (result); |
| } else |
| g_task_return_pointer (task, result, (GDestroyNotify)setup_link_result_free); |
| g_object_unref (task); |
| } |
| |
| void |
| mm_port_mbim_setup_link (MMPortMbim *self, |
| MMPort *data, |
| const gchar *link_prefix_hint, |
| GAsyncReadyCallback callback, |
| gpointer user_data) |
| { |
| GTask *task; |
| |
| task = g_task_new (self, NULL, callback, user_data); |
| |
| if (!self->priv->mbim_device) { |
| g_task_return_new_error (task, MM_CORE_ERROR, MM_CORE_ERROR_WRONG_STATE, "Port is not open"); |
| g_object_unref (task); |
| return; |
| } |
| |
| mbim_device_add_link (self->priv->mbim_device, |
| MBIM_DEVICE_SESSION_ID_AUTOMATIC, |
| mm_kernel_device_get_name (mm_port_peek_kernel_device (data)), |
| link_prefix_hint, |
| NULL, |
| (GAsyncReadyCallback) device_add_link_ready, |
| task); |
| } |
| |
| /*****************************************************************************/ |
| |
| gboolean |
| mm_port_mbim_cleanup_link_finish (MMPortMbim *self, |
| GAsyncResult *res, |
| GError **error) |
| { |
| return g_task_propagate_boolean (G_TASK (res), error); |
| } |
| |
| static void |
| device_delete_link_ready (MbimDevice *device, |
| GAsyncResult *res, |
| GTask *task) |
| { |
| GError *error = NULL; |
| |
| if (!mbim_device_delete_link_finish (device, res, &error)) |
| g_task_return_error (task, error); |
| else |
| g_task_return_boolean (task, TRUE); |
| g_object_unref (task); |
| } |
| |
| void |
| mm_port_mbim_cleanup_link (MMPortMbim *self, |
| const gchar *link_name, |
| GAsyncReadyCallback callback, |
| gpointer user_data) |
| { |
| GTask *task; |
| |
| task = g_task_new (self, NULL, callback, user_data); |
| |
| if (!self->priv->mbim_device) { |
| g_task_return_new_error (task, MM_CORE_ERROR, MM_CORE_ERROR_WRONG_STATE, "Port is not open"); |
| g_object_unref (task); |
| return; |
| } |
| |
| mbim_device_delete_link (self->priv->mbim_device, |
| link_name, |
| NULL, |
| (GAsyncReadyCallback) device_delete_link_ready, |
| task); |
| } |
| |
| /*****************************************************************************/ |
| |
| typedef struct { |
| MbimDevice *device; |
| MMPort *data; |
| } ResetContext; |
| |
| static void |
| reset_context_free (ResetContext *ctx) |
| { |
| g_clear_object (&ctx->device); |
| g_clear_object (&ctx->data); |
| g_slice_free (ResetContext, ctx); |
| } |
| |
| gboolean |
| mm_port_mbim_reset_finish (MMPortMbim *self, |
| GAsyncResult *res, |
| GError **error) |
| { |
| return g_task_propagate_boolean (G_TASK (res), error); |
| } |
| |
| static void |
| delete_all_links_ready (MbimDevice *device, |
| GAsyncResult *res, |
| GTask *task) |
| { |
| MMPortMbim *self; |
| GError *error = NULL; |
| |
| self = g_task_get_source_object (task); |
| |
| /* link deletion not fatal */ |
| if (!mbim_device_delete_all_links_finish (device, res, &error)) { |
| mm_obj_dbg (self, "couldn't delete all links: %s", error->message); |
| g_clear_error (&error); |
| } |
| |
| g_task_return_boolean (task, TRUE); |
| g_object_unref (task); |
| } |
| |
| static void |
| reset_device_new_ready (GObject *source, |
| GAsyncResult *res, |
| GTask *task) |
| { |
| MMPortMbim *self; |
| ResetContext *ctx; |
| GError *error = NULL; |
| |
| self = g_task_get_source_object (task); |
| ctx = g_task_get_task_data (task); |
| |
| ctx->device = mbim_device_new_finish (res, &error); |
| if (!ctx->device) { |
| g_task_return_error (task, error); |
| g_object_unref (task); |
| return; |
| } |
| |
| /* first, delete all links found, if any */ |
| mm_obj_dbg (self, "deleting all links in data interface '%s'", |
| mm_port_get_device (ctx->data)); |
| mbim_device_delete_all_links (ctx->device, |
| mm_port_get_device (ctx->data), |
| NULL, |
| (GAsyncReadyCallback)delete_all_links_ready, |
| task); |
| } |
| |
| void |
| mm_port_mbim_reset (MMPortMbim *self, |
| MMPort *data, |
| GAsyncReadyCallback callback, |
| gpointer user_data) |
| { |
| GTask *task; |
| ResetContext *ctx; |
| g_autoptr(GFile) file = NULL; |
| g_autofree gchar *fullpath = NULL; |
| |
| task = g_task_new (self, NULL, callback, user_data); |
| |
| if (self->priv->mbim_device) { |
| g_task_return_new_error (task, MM_CORE_ERROR, MM_CORE_ERROR_WRONG_STATE, "Port is already open"); |
| g_object_unref (task); |
| return; |
| } |
| |
| ctx = g_slice_new0 (ResetContext); |
| ctx->data = g_object_ref (data); |
| g_task_set_task_data (task, ctx, (GDestroyNotify) reset_context_free); |
| |
| fullpath = g_strdup_printf ("/dev/%s", mm_port_get_device (MM_PORT (self))); |
| file = g_file_new_for_path (fullpath); |
| |
| mbim_device_new (file, NULL, |
| (GAsyncReadyCallback) reset_device_new_ready, |
| task); |
| } |
| |
| /*****************************************************************************/ |
| |
| static void |
| reset_monitoring (MMPortMbim *self, |
| MbimDevice *mbim_device) |
| { |
| if (self->priv->notification_monitoring_id && mbim_device) { |
| g_signal_handler_disconnect (mbim_device, self->priv->notification_monitoring_id); |
| self->priv->notification_monitoring_id = 0; |
| } |
| if (self->priv->timeout_monitoring_id && mbim_device) { |
| g_signal_handler_disconnect (mbim_device, self->priv->timeout_monitoring_id); |
| self->priv->timeout_monitoring_id = 0; |
| } |
| if (self->priv->removed_monitoring_id && mbim_device) { |
| g_signal_handler_disconnect (mbim_device, self->priv->removed_monitoring_id); |
| self->priv->removed_monitoring_id = 0; |
| } |
| } |
| |
| static void |
| consecutive_timeouts_updated_cb (MMPortMbim *self, |
| GParamSpec *pspec, |
| MbimDevice *mbim_device) |
| { |
| g_signal_emit_by_name (self, MM_PORT_SIGNAL_TIMED_OUT, mbim_device_get_consecutive_timeouts (mbim_device)); |
| } |
| |
| static void |
| device_removed_cb (MMPortMbim *self) |
| { |
| g_signal_emit_by_name (self, MM_PORT_SIGNAL_REMOVED); |
| } |
| |
| static void |
| notification_cb (MMPortMbim *self, |
| MbimMessage *notification) |
| { |
| g_signal_emit (self, signals[SIGNAL_NOTIFICATION], 0, notification); |
| } |
| |
| static void |
| setup_monitoring (MMPortMbim *self, |
| MbimDevice *mbim_device) |
| { |
| g_assert (mbim_device); |
| |
| reset_monitoring (self, mbim_device); |
| |
| g_assert (!self->priv->notification_monitoring_id); |
| self->priv->notification_monitoring_id = g_signal_connect_swapped (mbim_device, |
| MBIM_DEVICE_SIGNAL_INDICATE_STATUS, |
| G_CALLBACK (notification_cb), |
| self); |
| |
| g_assert (!self->priv->timeout_monitoring_id); |
| self->priv->timeout_monitoring_id = g_signal_connect_swapped (mbim_device, |
| "notify::" MBIM_DEVICE_CONSECUTIVE_TIMEOUTS, |
| G_CALLBACK (consecutive_timeouts_updated_cb), |
| self); |
| |
| g_assert (!self->priv->removed_monitoring_id); |
| self->priv->removed_monitoring_id = g_signal_connect_swapped (mbim_device, |
| MBIM_DEVICE_SIGNAL_REMOVED, |
| G_CALLBACK (device_removed_cb), |
| self); |
| } |
| |
| /*****************************************************************************/ |
| |
| gboolean |
| mm_port_mbim_supports_service (MMPortMbim *self, |
| MbimService service) |
| { |
| return mm_port_mbim_supports_command (self, service, 0); |
| } |
| |
| gboolean |
| mm_port_mbim_supports_command (MMPortMbim *self, |
| MbimService service, |
| guint cid) |
| { |
| guint i; |
| |
| for (i = 0; i < self->priv->device_services_count; i++) { |
| if (mbim_uuid_to_service (&self->priv->device_services[i]->device_service_id) == service) { |
| guint j; |
| |
| /* If not asking for a specific CID, we're asking for overall service |
| * support */ |
| if (cid == 0) |
| return TRUE; |
| |
| for (j = 0; j < self->priv->device_services[i]->cids_count; j++) { |
| if (self->priv->device_services[i]->cids[j] == cid) |
| return TRUE; |
| } |
| /* service found but not matching cid */ |
| return FALSE; |
| } |
| } |
| /* service not found */ |
| return FALSE; |
| } |
| |
| /*****************************************************************************/ |
| |
| gboolean |
| mm_port_mbim_open_finish (MMPortMbim *self, |
| GAsyncResult *res, |
| GError **error) |
| { |
| return g_task_propagate_boolean (G_TASK (res), error); |
| } |
| |
| static void |
| open_complete (GTask *task, GError *error) |
| { |
| MMPortMbim *self; |
| |
| self = g_task_get_source_object (task); |
| self->priv->in_progress = FALSE; |
| if (error) { |
| g_clear_object (&self->priv->mbim_device); |
| g_task_return_error (task, error); |
| } else |
| g_task_return_boolean (task, TRUE); |
| g_object_unref (task); |
| } |
| |
| #if defined WITH_QMI && QMI_MBIM_QMUX_SUPPORTED |
| |
| static void |
| qmi_device_open_ready (QmiDevice *dev, |
| GAsyncResult *res, |
| GTask *task) |
| { |
| g_autoptr(GError) error = NULL; |
| MMPortMbim *self; |
| |
| self = g_task_get_source_object (task); |
| |
| if (!qmi_device_open_finish (dev, res, &error)) { |
| mm_obj_dbg (self, "error: couldn't open QmiDevice: %s", error->message); |
| g_clear_object (&self->priv->qmi_device); |
| /* Ignore error and complete */ |
| mm_obj_msg (self, "MBIM device is not QMI capable"); |
| } else { |
| mm_obj_msg (self, "MBIM device is QMI capable"); |
| } |
| |
| open_complete (task, NULL); |
| } |
| |
| static void |
| qmi_device_new_ready (GObject *unused, |
| GAsyncResult *res, |
| GTask *task) |
| { |
| g_autoptr(GError) error = NULL; |
| MMPortMbim *self; |
| |
| self = g_task_get_source_object (task); |
| |
| self->priv->qmi_device = qmi_device_new_finish (res, &error); |
| if (!self->priv->qmi_device) { |
| mm_obj_dbg (self, "error: couldn't create QmiDevice: %s", error->message); |
| /* Ignore error and complete */ |
| mm_obj_msg (self, "MBIM device is not QMI capable"); |
| open_complete (task, NULL); |
| return; |
| } |
| |
| /* Try to open using QMI over MBIM */ |
| mm_obj_dbg (self, "trying to open QMI over MBIM device..."); |
| qmi_device_open (self->priv->qmi_device, |
| (QMI_DEVICE_OPEN_FLAGS_PROXY | |
| QMI_DEVICE_OPEN_FLAGS_MBIM | |
| QMI_DEVICE_OPEN_FLAGS_VERSION_INFO | |
| QMI_DEVICE_OPEN_FLAGS_EXPECT_INDICATIONS), |
| 15, |
| g_task_get_cancellable (task), |
| (GAsyncReadyCallback)qmi_device_open_ready, |
| task); |
| } |
| |
| static void |
| check_qmi_support (GTask *task) |
| { |
| MMPortMbim *self; |
| GFile *file; |
| |
| self = g_task_get_source_object (task); |
| file = G_FILE (g_task_get_task_data (task)); |
| |
| if (!file) { |
| mm_obj_msg (self, "Skipping QMI support check in MBIM device"); |
| open_complete (task, NULL); |
| return; |
| } |
| |
| if (!mm_port_mbim_supports_service (self, MBIM_SERVICE_QMI)) { |
| mm_obj_msg (self, "MBIM device is not QMI capable"); |
| open_complete (task, NULL); |
| return; |
| } |
| |
| /* Attempt to create and open the QMI device */ |
| mm_obj_dbg (self, "checking if QMI over MBIM is supported..."); |
| qmi_device_new (file, |
| g_task_get_cancellable (task), |
| (GAsyncReadyCallback) qmi_device_new_ready, |
| task); |
| } |
| |
| #endif |
| |
| static void |
| mbim_query_device_services_ready (MbimDevice *device, |
| GAsyncResult *res, |
| GTask *task) |
| { |
| g_autoptr(MbimMessage) response = NULL; |
| g_autoptr(GError) error = NULL; |
| MMPortMbim *self; |
| |
| self = g_task_get_source_object (task); |
| |
| /* reset any before reloading */ |
| self->priv->device_services_count = 0; |
| g_clear_pointer (&self->priv->device_services, (GDestroyNotify)mbim_device_service_element_array_free); |
| |
| response = mbim_device_command_finish (device, res, &error); |
| if (!response || |
| !mbim_message_response_get_result (response, MBIM_MESSAGE_TYPE_COMMAND_DONE, &error) || |
| !mbim_message_device_services_response_parse ( |
| response, |
| &self->priv->device_services_count, |
| NULL, /* max_dss_sessions */ |
| &self->priv->device_services, |
| &error)) { |
| mm_obj_warn (self, "couldn't query device services: %s", error->message); |
| } |
| |
| #if defined WITH_QMI && QMI_MBIM_QMUX_SUPPORTED |
| check_qmi_support (task); |
| #else |
| open_complete (task, NULL); |
| #endif |
| } |
| |
| static void |
| mbim_query_device_services (GTask *task) |
| { |
| g_autoptr(MbimMessage) message = NULL; |
| MMPortMbim *self; |
| |
| self = g_task_get_source_object (task); |
| |
| message = mbim_message_device_services_query_new (NULL); |
| mbim_device_command (self->priv->mbim_device, |
| message, |
| 20, |
| NULL, |
| (GAsyncReadyCallback)mbim_query_device_services_ready, |
| task); |
| } |
| |
| static void |
| mbim_device_open_ready (MbimDevice *mbim_device, |
| GAsyncResult *res, |
| GTask *task) |
| { |
| GError *error = NULL; |
| MMPortMbim *self; |
| |
| self = g_task_get_source_object (task); |
| |
| if (!mbim_device_open_full_finish (mbim_device, res, &error)) { |
| open_complete (task, error); |
| return; |
| } |
| |
| mm_obj_dbg (self, "MBIM device is now open"); |
| setup_monitoring (self, mbim_device); |
| |
| mbim_query_device_services (task); |
| } |
| |
| static void |
| mbim_device_new_ready (GObject *unused, |
| GAsyncResult *res, |
| GTask *task) |
| { |
| GError *error = NULL; |
| MMPortMbim *self; |
| |
| self = g_task_get_source_object (task); |
| self->priv->mbim_device = mbim_device_new_finish (res, &error); |
| if (!self->priv->mbim_device) { |
| open_complete (task, error); |
| return; |
| } |
| |
| /* Now open the MBIM device */ |
| mbim_device_open_full (self->priv->mbim_device, |
| MBIM_DEVICE_OPEN_FLAGS_PROXY | MBIM_DEVICE_OPEN_FLAGS_MS_MBIMEX_V3, |
| 45, |
| g_task_get_cancellable (task), |
| (GAsyncReadyCallback)mbim_device_open_ready, |
| task); |
| } |
| |
| void |
| mm_port_mbim_open (MMPortMbim *self, |
| #if defined WITH_QMI && QMI_MBIM_QMUX_SUPPORTED |
| gboolean try_qmi_over_mbim, |
| #endif |
| GCancellable *cancellable, |
| GAsyncReadyCallback callback, |
| gpointer user_data) |
| { |
| g_autoptr(GFile) file = NULL; |
| g_autofree gchar *fullpath = NULL; |
| GTask *task; |
| |
| g_return_if_fail (MM_IS_PORT_MBIM (self)); |
| |
| task = g_task_new (self, cancellable, callback, user_data); |
| |
| if (self->priv->in_progress) { |
| g_task_return_new_error (task, |
| MM_CORE_ERROR, |
| MM_CORE_ERROR_IN_PROGRESS, |
| "MBIM device open/close operation in progress"); |
| g_object_unref (task); |
| return; |
| } |
| |
| if (self->priv->mbim_device) { |
| g_task_return_boolean (task, TRUE); |
| g_object_unref (task); |
| return; |
| } |
| |
| fullpath = g_strdup_printf ("/dev/%s", mm_port_get_device (MM_PORT (self))); |
| file = g_file_new_for_path (fullpath); |
| |
| #if defined WITH_QMI && QMI_MBIM_QMUX_SUPPORTED |
| /* If we want to try QMI over MBIM, store the GFile as task data */ |
| if (try_qmi_over_mbim) |
| g_task_set_task_data (task, g_object_ref (file), g_object_unref); |
| #endif |
| |
| self->priv->in_progress = TRUE; |
| mbim_device_new (file, |
| cancellable, |
| (GAsyncReadyCallback)mbim_device_new_ready, |
| task); |
| } |
| |
| /*****************************************************************************/ |
| |
| gboolean |
| mm_port_mbim_is_open (MMPortMbim *self) |
| { |
| g_return_val_if_fail (MM_IS_PORT_MBIM (self), FALSE); |
| |
| return !!self->priv->mbim_device; |
| } |
| |
| /*****************************************************************************/ |
| |
| typedef struct { |
| MbimDevice *mbim_device; |
| #if defined WITH_QMI && QMI_MBIM_QMUX_SUPPORTED |
| QmiDevice *qmi_device; |
| #endif |
| } PortMbimCloseContext; |
| |
| static void |
| port_mbim_close_context_free (PortMbimCloseContext *ctx) |
| { |
| g_clear_object (&ctx->mbim_device); |
| #if defined WITH_QMI && QMI_MBIM_QMUX_SUPPORTED |
| g_clear_object (&ctx->qmi_device); |
| #endif |
| g_slice_free (PortMbimCloseContext, ctx); |
| } |
| |
| gboolean |
| mm_port_mbim_close_finish (MMPortMbim *self, |
| GAsyncResult *res, |
| GError **error) |
| { |
| return g_task_propagate_boolean (G_TASK (res), error); |
| } |
| |
| static void |
| mbim_device_close_ready (MbimDevice *mbim_device, |
| GAsyncResult *res, |
| GTask *task) |
| { |
| GError *error = NULL; |
| MMPortMbim *self; |
| |
| self = g_task_get_source_object (task); |
| |
| g_assert (!self->priv->mbim_device); |
| self->priv->in_progress = FALSE; |
| reset_monitoring (self, mbim_device); |
| |
| if (!mbim_device_close_finish (mbim_device, res, &error)) |
| g_task_return_error (task, error); |
| else |
| g_task_return_boolean (task, TRUE); |
| g_object_unref (task); |
| } |
| |
| static void |
| port_mbim_device_close (GTask *task) |
| { |
| PortMbimCloseContext *ctx; |
| |
| ctx = g_task_get_task_data (task); |
| g_assert (ctx->mbim_device); |
| mbim_device_close (ctx->mbim_device, |
| 5, |
| NULL, |
| (GAsyncReadyCallback)mbim_device_close_ready, |
| task); |
| } |
| |
| #if defined WITH_QMI && QMI_MBIM_QMUX_SUPPORTED |
| |
| static void |
| qmi_device_close_ready (QmiDevice *qmi_device, |
| GAsyncResult *res, |
| GTask *task) |
| { |
| GError *error = NULL; |
| MMPortMbim *self; |
| |
| self = g_task_get_source_object (task); |
| |
| if (!qmi_device_close_finish (qmi_device, res, &error)) { |
| mm_obj_warn (self, "Couldn't properly close QMI device: %s", error->message); |
| g_error_free (error); |
| } |
| |
| port_mbim_device_close (task); |
| } |
| |
| #endif |
| |
| void |
| mm_port_mbim_close (MMPortMbim *self, |
| GAsyncReadyCallback callback, |
| gpointer user_data) |
| { |
| PortMbimCloseContext *ctx; |
| GTask *task; |
| |
| g_return_if_fail (MM_IS_PORT_MBIM (self)); |
| |
| task = g_task_new (self, NULL, callback, user_data); |
| |
| if (self->priv->in_progress) { |
| g_task_return_new_error (task, |
| MM_CORE_ERROR, |
| MM_CORE_ERROR_IN_PROGRESS, |
| "MBIM device open/close operation in progress"); |
| g_object_unref (task); |
| return; |
| } |
| |
| if (!self->priv->mbim_device) { |
| g_task_return_boolean (task, TRUE); |
| g_object_unref (task); |
| return; |
| } |
| |
| self->priv->in_progress = TRUE; |
| |
| /* Store device(s) to close in the context */ |
| ctx = g_slice_new0 (PortMbimCloseContext); |
| ctx->mbim_device = g_steal_pointer (&self->priv->mbim_device); |
| g_task_set_task_data (task, ctx, (GDestroyNotify)port_mbim_close_context_free); |
| |
| #if defined WITH_QMI && QMI_MBIM_QMUX_SUPPORTED |
| if (self->priv->qmi_device) { |
| GList *l; |
| |
| /* Release all allocated clients */ |
| for (l = self->priv->qmi_clients; l; l = g_list_next (l)) { |
| QmiClient *qmi_client = QMI_CLIENT (l->data); |
| |
| mm_obj_dbg (self, "Releasing client for service '%s'...", |
| qmi_service_get_string (qmi_client_get_service (qmi_client))); |
| qmi_device_release_client (self->priv->qmi_device, |
| qmi_client, |
| QMI_DEVICE_RELEASE_CLIENT_FLAGS_RELEASE_CID, |
| 3, NULL, NULL, NULL); |
| } |
| g_list_free_full (self->priv->qmi_clients, g_object_unref); |
| self->priv->qmi_clients = NULL; |
| |
| ctx->qmi_device = g_steal_pointer (&self->priv->qmi_device); |
| qmi_device_close_async (ctx->qmi_device, |
| 5, |
| NULL, |
| (GAsyncReadyCallback)qmi_device_close_ready, |
| task); |
| return; |
| } |
| #endif |
| |
| port_mbim_device_close (task); |
| } |
| |
| /*****************************************************************************/ |
| |
| MbimDevice * |
| mm_port_mbim_peek_device (MMPortMbim *self) |
| { |
| g_return_val_if_fail (MM_IS_PORT_MBIM (self), NULL); |
| |
| return self->priv->mbim_device; |
| } |
| |
| /*****************************************************************************/ |
| |
| MMPortMbim * |
| mm_port_mbim_new (const gchar *name, |
| MMPortSubsys subsys) |
| { |
| return MM_PORT_MBIM (g_object_new (MM_TYPE_PORT_MBIM, |
| MM_PORT_DEVICE, name, |
| MM_PORT_SUBSYS, subsys, |
| MM_PORT_GROUP, MM_PORT_GROUP_USED, |
| MM_PORT_TYPE, MM_PORT_TYPE_MBIM, |
| NULL)); |
| } |
| |
| static void |
| mm_port_mbim_init (MMPortMbim *self) |
| { |
| self->priv = G_TYPE_INSTANCE_GET_PRIVATE (self, MM_TYPE_PORT_MBIM, MMPortMbimPrivate); |
| } |
| |
| static void |
| dispose (GObject *object) |
| { |
| MMPortMbim *self = MM_PORT_MBIM (object); |
| |
| self->priv->device_services_count = 0; |
| g_clear_pointer (&self->priv->device_services, (GDestroyNotify)mbim_device_service_element_array_free); |
| |
| #if defined WITH_QMI && QMI_MBIM_QMUX_SUPPORTED |
| g_list_free_full (self->priv->qmi_clients, g_object_unref); |
| self->priv->qmi_clients = NULL; |
| g_clear_object (&self->priv->qmi_device); |
| #endif |
| |
| /* Clear device object */ |
| reset_monitoring (self, self->priv->mbim_device); |
| g_clear_object (&self->priv->mbim_device); |
| |
| G_OBJECT_CLASS (mm_port_mbim_parent_class)->dispose (object); |
| } |
| |
| static void |
| mm_port_mbim_class_init (MMPortMbimClass *klass) |
| { |
| GObjectClass *object_class = G_OBJECT_CLASS (klass); |
| |
| g_type_class_add_private (object_class, sizeof (MMPortMbimPrivate)); |
| |
| /* Virtual methods */ |
| object_class->dispose = dispose; |
| |
| signals[SIGNAL_NOTIFICATION] = |
| g_signal_new (MM_PORT_MBIM_SIGNAL_NOTIFICATION, |
| G_OBJECT_CLASS_TYPE (G_OBJECT_CLASS (klass)), |
| G_SIGNAL_RUN_LAST, |
| G_STRUCT_OFFSET (MMPortMbimClass, notification), |
| NULL, NULL, |
| g_cclosure_marshal_generic, |
| G_TYPE_NONE, 1, MBIM_TYPE_MESSAGE); |
| } |