/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * libqmi-glib -- GLib/GIO based library to control QMI devices * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301 USA. * * Copyright (C) 2012 Lanedo GmbH * Copyright (C) 2012 - 2014 Aleksander Morgado */ #include #include #include #include #include #include #include #include #include #include "qmi-device.h" #include "qmi-message.h" #include "qmi-ctl.h" #include "qmi-dms.h" #include "qmi-wds.h" #include "qmi-nas.h" #include "qmi-wms.h" #include "qmi-pds.h" #include "qmi-pbm.h" #include "qmi-uim.h" #include "qmi-oma.h" #include "qmi-wda.h" #include "qmi-voice.h" #include "qmi-utils.h" #include "qmi-error-types.h" #include "qmi-enum-types.h" #include "qmi-proxy.h" /** * SECTION:qmi-device * @title: QmiDevice * @short_description: Generic QMI device handling routines * * #QmiDevice is a generic type in charge of controlling the access of multiple * #QmiClient objects to the managed QMI port. * * A #QmiDevice can only handle one single QMI port. */ static void async_initable_iface_init (GAsyncInitableIface *iface); G_DEFINE_TYPE_EXTENDED (QmiDevice, qmi_device, G_TYPE_OBJECT, 0, G_IMPLEMENT_INTERFACE (G_TYPE_ASYNC_INITABLE, async_initable_iface_init)) #define MAX_SPAWN_RETRIES 10 enum { PROP_0, PROP_FILE, PROP_NO_FILE_CHECK, PROP_PROXY_PATH, PROP_WWAN_IFACE, PROP_LAST }; enum { SIGNAL_INDICATION, SIGNAL_LAST }; static GParamSpec *properties[PROP_LAST]; static guint signals [SIGNAL_LAST] = { 0 }; struct _QmiDevicePrivate { /* File */ GFile *file; gchar *path; gchar *path_display; gboolean no_file_check; gchar *proxy_path; /* WWAN interface */ gboolean no_wwan_check; gchar *wwan_iface; /* Implicit CTL client */ QmiClientCtl *client_ctl; guint sync_indication_id; /* Supported services */ GArray *supported_services; /* I/O stream, set when the file is open */ GInputStream *istream; GOutputStream *ostream; GSource *input_source; GByteArray *buffer; /* Support for qmi-proxy */ GSocketClient *socket_client; GSocketConnection *socket_connection; /* HT to keep track of ongoing transactions */ GHashTable *transactions; /* HT of clients that want to get indications */ GHashTable *registered_clients; }; #define BUFFER_SIZE 2048 /*****************************************************************************/ /* Message transactions (private) */ typedef struct { QmiDevice *self; gpointer key; } TransactionWaitContext; typedef struct { QmiMessage *message; GSimpleAsyncResult *result; GSource *timeout_source; GCancellable *cancellable; gulong cancellable_id; TransactionWaitContext *wait_ctx; } Transaction; static Transaction * transaction_new (QmiDevice *self, QmiMessage *message, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data) { Transaction *tr; tr = g_slice_new0 (Transaction); tr->message = qmi_message_ref (message); tr->result = g_simple_async_result_new (G_OBJECT (self), callback, user_data, transaction_new); if (cancellable) tr->cancellable = g_object_ref (cancellable); return tr; } static void transaction_complete_and_free (Transaction *tr, QmiMessage *reply, const GError *error) { g_assert (reply != NULL || error != NULL); if (tr->timeout_source) g_source_destroy (tr->timeout_source); if (tr->cancellable) { if (tr->cancellable_id) g_cancellable_disconnect (tr->cancellable, tr->cancellable_id); g_object_unref (tr->cancellable); } if (tr->wait_ctx) g_slice_free (TransactionWaitContext, tr->wait_ctx); if (reply) g_simple_async_result_set_op_res_gpointer (tr->result, qmi_message_ref (reply), (GDestroyNotify)qmi_message_unref); else g_simple_async_result_set_from_error (tr->result, error); g_simple_async_result_complete_in_idle (tr->result); g_object_unref (tr->result); qmi_message_unref (tr->message); g_slice_free (Transaction, tr); } static inline gpointer build_transaction_key (QmiMessage *message) { gpointer key; guint8 service; guint8 client_id; guint16 transaction_id; service = (guint8)qmi_message_get_service (message); client_id = qmi_message_get_client_id (message); transaction_id = qmi_message_get_transaction_id (message); /* We're putting a 32 bit value into a gpointer */ key = GUINT_TO_POINTER ((((service << 8) | client_id) << 16) | transaction_id); return key; } static Transaction * device_release_transaction (QmiDevice *self, gpointer key) { Transaction *tr = NULL; if (self->priv->transactions) { tr = g_hash_table_lookup (self->priv->transactions, key); if (tr) /* If found, remove it from the HT */ g_hash_table_remove (self->priv->transactions, key); } return tr; } static gboolean transaction_timed_out (TransactionWaitContext *ctx) { Transaction *tr; GError *error = NULL; tr = device_release_transaction (ctx->self, ctx->key); tr->timeout_source = NULL; /* Complete transaction with a timeout error */ error = g_error_new (QMI_CORE_ERROR, QMI_CORE_ERROR_TIMEOUT, "Transaction timed out"); transaction_complete_and_free (tr, NULL, error); g_error_free (error); return FALSE; } static void transaction_cancelled (GCancellable *cancellable, TransactionWaitContext *ctx) { Transaction *tr; GError *error = NULL; tr = device_release_transaction (ctx->self, ctx->key); tr->cancellable_id = 0; /* Complete transaction with an abort error */ error = g_error_new (QMI_PROTOCOL_ERROR, QMI_PROTOCOL_ERROR_ABORTED, "Transaction aborted"); transaction_complete_and_free (tr, NULL, error); g_error_free (error); } static gboolean device_store_transaction (QmiDevice *self, Transaction *tr, guint timeout, GError **error) { gpointer key; if (G_UNLIKELY (!self->priv->transactions)) self->priv->transactions = g_hash_table_new (g_direct_hash, g_direct_equal); key = build_transaction_key (tr->message); /* Setup the timeout and cancellation */ tr->wait_ctx = g_slice_new (TransactionWaitContext); tr->wait_ctx->self = self; tr->wait_ctx->key = key; /* valid as long as the transaction is in the HT */ tr->timeout_source = g_timeout_source_new_seconds (timeout); g_source_set_callback (tr->timeout_source, (GSourceFunc)transaction_timed_out, tr->wait_ctx, NULL); g_source_attach (tr->timeout_source, g_main_context_get_thread_default ()); g_source_unref (tr->timeout_source); if (tr->cancellable) { tr->cancellable_id = g_cancellable_connect (tr->cancellable, (GCallback)transaction_cancelled, tr->wait_ctx, NULL); if (!tr->cancellable_id) { g_set_error (error, QMI_PROTOCOL_ERROR, QMI_PROTOCOL_ERROR_ABORTED, "Request is already cancelled"); return FALSE; } } /* Keep in the HT */ g_hash_table_insert (self->priv->transactions, key, tr); return TRUE; } static Transaction * device_match_transaction (QmiDevice *self, QmiMessage *message) { /* msg can be either the original message or the response */ return device_release_transaction (self, build_transaction_key (message)); } /*****************************************************************************/ /* Version info request */ /** * qmi_device_get_service_version_info_finish: * @self: a #QmiDevice. * @res: a #GAsyncResult. * @error: Return location for error or %NULL. * * Finishes an operation started with qmi_device_get_service_version_info(). * * Returns: a #GArray of #QmiDeviceServiceVersionInfo elements, or #NULL if @error is set. The returned value should be freed with g_array_unref(). */ GArray * qmi_device_get_service_version_info_finish (QmiDevice *self, GAsyncResult *res, GError **error) { if (g_simple_async_result_propagate_error (G_SIMPLE_ASYNC_RESULT (res), error)) return NULL; return g_array_ref (g_simple_async_result_get_op_res_gpointer (G_SIMPLE_ASYNC_RESULT (res))); } static void version_info_ready (QmiClientCtl *client_ctl, GAsyncResult *res, GSimpleAsyncResult *simple) { GArray *service_list = NULL; GArray *out; QmiMessageCtlGetVersionInfoOutput *output; GError *error = NULL; guint i; /* Check result of the async operation */ output = qmi_client_ctl_get_version_info_finish (client_ctl, res, &error); if (!output) { g_simple_async_result_take_error (simple, error); g_simple_async_result_complete (simple); g_object_unref (simple); return; } /* Check result of the QMI operation */ if (!qmi_message_ctl_get_version_info_output_get_result (output, &error)) { qmi_message_ctl_get_version_info_output_unref (output); g_simple_async_result_take_error (simple, error); g_simple_async_result_complete (simple); g_object_unref (simple); return; } /* QMI operation succeeded, we can now get the outputs */ qmi_message_ctl_get_version_info_output_get_service_list (output, &service_list, NULL); out = g_array_sized_new (FALSE, FALSE, sizeof (QmiDeviceServiceVersionInfo), service_list->len); for (i = 0; i < service_list->len; i++) { QmiMessageCtlGetVersionInfoOutputServiceListService *info; QmiDeviceServiceVersionInfo outinfo; info = &g_array_index (service_list, QmiMessageCtlGetVersionInfoOutputServiceListService, i); outinfo.service = info->service; outinfo.major_version = info->major_version; outinfo.minor_version = info->minor_version; g_array_append_val (out, outinfo); } qmi_message_ctl_get_version_info_output_unref (output); g_simple_async_result_set_op_res_gpointer (simple, out, (GDestroyNotify)g_array_unref); g_simple_async_result_complete (simple); g_object_unref (simple); } /** * qmi_device_get_service_version_info: * @self: a #QmiDevice. * @timeout: maximum time to wait for the method to complete, in seconds. * @cancellable: a #GCancellable or %NULL. * @callback: a #GAsyncReadyCallback to call when the request is satisfied. * @user_data: user data to pass to @callback. * * Asynchronously requests the service version information of the device. * * When the operation is finished, @callback will be invoked in the thread-default main loop of the thread you are calling this method from. * * You can then call qmi_device_get_service_version_info_finish() to get the result of the operation. */ void qmi_device_get_service_version_info (QmiDevice *self, guint timeout, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data) { qmi_client_ctl_get_version_info ( self->priv->client_ctl, NULL, timeout, cancellable, (GAsyncReadyCallback)version_info_ready, g_simple_async_result_new (G_OBJECT (self), callback, user_data, qmi_device_get_service_version_info)); } /*****************************************************************************/ /* Version info checks (private) */ static const QmiMessageCtlGetVersionInfoOutputServiceListService * find_service_version_info (QmiDevice *self, QmiService service) { guint i; if (!self->priv->supported_services) return NULL; for (i = 0; i < self->priv->supported_services->len; i++) { const QmiMessageCtlGetVersionInfoOutputServiceListService *info; info = &g_array_index (self->priv->supported_services, QmiMessageCtlGetVersionInfoOutputServiceListService, i); if (service == info->service) return info; } return NULL; } static gboolean check_service_supported (QmiDevice *self, QmiService service) { /* If we didn't check supported services, just assume it is supported */ if (!self->priv->supported_services) { g_debug ("[%s] Assuming service '%s' is supported...", self->priv->path_display, qmi_service_get_string (service)); return TRUE; } return !!find_service_version_info (self, service); } static gboolean check_message_supported (QmiDevice *self, QmiMessage *message, GError **error) { const QmiMessageCtlGetVersionInfoOutputServiceListService *info; guint major = 0; guint minor = 0; /* If we didn't check supported services, just assume it is supported */ if (!self->priv->supported_services) return TRUE; /* For CTL, we assume all are supported */ if (qmi_message_get_service (message) == QMI_SERVICE_CTL) return TRUE; /* If we cannot get in which version this message was introduced, we'll just * assume it's supported */ if (!qmi_message_get_version_introduced (message, &major, &minor)) return TRUE; /* Get version info. It MUST exist because we allowed creating a client * of this service type */ info = find_service_version_info (self, qmi_message_get_service (message)); g_assert (info != NULL); g_assert (info->service == qmi_message_get_service (message)); /* If the version of the message is greater than the version of the service, * report unsupported */ if (major > info->major_version || (major == info->major_version && minor > info->minor_version)) { g_set_error (error, QMI_CORE_ERROR, QMI_CORE_ERROR_UNSUPPORTED, "QMI service '%s' version '%u.%u' required, got version '%u.%u'", qmi_service_get_string (qmi_message_get_service (message)), major, minor, info->major_version, info->minor_version); return FALSE; } /* Supported! */ return TRUE; } /*****************************************************************************/ /** * qmi_device_get_file: * @self: a #QmiDevice. * * Get the #GFile associated with this #QmiDevice. * * Returns: a #GFile that must be freed with g_object_unref(). */ GFile * qmi_device_get_file (QmiDevice *self) { GFile *file = NULL; g_return_val_if_fail (QMI_IS_DEVICE (self), NULL); g_object_get (G_OBJECT (self), QMI_DEVICE_FILE, &file, NULL); return file; } /** * qmi_device_peek_file: * @self: a #QmiDevice. * * Get the #GFile associated with this #QmiDevice, without increasing the reference count * on the returned object. * * Returns: a #GFile. Do not free the returned object, it is owned by @self. */ GFile * qmi_device_peek_file (QmiDevice *self) { g_return_val_if_fail (QMI_IS_DEVICE (self), NULL); return self->priv->file; } /** * qmi_device_get_path: * @self: a #QmiDevice. * * Get the system path of the underlying QMI device. * * Returns: the system path of the device. */ const gchar * qmi_device_get_path (QmiDevice *self) { g_return_val_if_fail (QMI_IS_DEVICE (self), NULL); return self->priv->path; } /** * qmi_device_get_path_display: * @self: a #QmiDevice. * * Get the system path of the underlying QMI device in UTF-8. * * Returns: UTF-8 encoded system path of the device. */ const gchar * qmi_device_get_path_display (QmiDevice *self) { g_return_val_if_fail (QMI_IS_DEVICE (self), NULL); return self->priv->path_display; } /** * qmi_device_is_open: * @self: a #QmiDevice. * * Checks whether the #QmiDevice is open for I/O. * * Returns: %TRUE if @self is open, %FALSE otherwise. */ gboolean qmi_device_is_open (QmiDevice *self) { g_return_val_if_fail (QMI_IS_DEVICE (self), FALSE); return !!(self->priv->istream && self->priv->ostream); } /*****************************************************************************/ /* WWAN iface name * Always reload from scratch, to handle possible net interface renames */ static void reload_wwan_iface_name (QmiDevice *self) { const gchar *cdc_wdm_device_name; static const gchar *driver_names[] = { "usbmisc", "usb" }; guint i; /* Early cleanup */ g_free (self->priv->wwan_iface); self->priv->wwan_iface = NULL; cdc_wdm_device_name = strrchr (self->priv->path, '/'); if (!cdc_wdm_device_name) { g_warning ("[%s] invalid path for cdc-wdm control port", self->priv->path_display); return; } cdc_wdm_device_name++; for (i = 0; i < G_N_ELEMENTS (driver_names) && !self->priv->wwan_iface; i++) { gchar *sysfs_path; GFile *sysfs_file; GFileEnumerator *enumerator; GError *error = NULL; sysfs_path = g_strdup_printf ("/sys/class/%s/%s/device/net/", driver_names[i], cdc_wdm_device_name); sysfs_file = g_file_new_for_path (sysfs_path); enumerator = g_file_enumerate_children (sysfs_file, G_FILE_ATTRIBUTE_STANDARD_NAME, G_FILE_QUERY_INFO_NONE, NULL, &error); if (!enumerator) { g_debug ("[%s] cannot enumerate files at path '%s': %s", self->priv->path_display, sysfs_path, error->message); g_error_free (error); } else { GFileInfo *file_info; /* Ignore errors when enumerating */ while ((file_info = g_file_enumerator_next_file (enumerator, NULL, NULL)) != NULL) { const gchar *name; name = g_file_info_get_name (file_info); if (name) { /* We only expect ONE file in the sysfs directory corresponding * to this control port, if more found for any reason, warn about it */ if (self->priv->wwan_iface) g_warning ("[%s] invalid additional wwan iface found: %s", self->priv->path_display, name); else self->priv->wwan_iface = g_strdup (name); } g_object_unref (file_info); } g_object_unref (enumerator); } g_free (sysfs_path); g_object_unref (sysfs_file); } if (!self->priv->wwan_iface) g_warning ("[%s] wwan iface not found", self->priv->path_display); } /** * qmi_device_get_wwan_iface: * @self: a #QmiDevice. * * Get the WWAN interface name associated with this /dev/cdc-wdm control port. * This value will be loaded the first time it's asked for it. * * Returns: UTF-8 encoded network interface name, or %NULL if not available. */ const gchar * qmi_device_get_wwan_iface (QmiDevice *self) { g_return_val_if_fail (QMI_IS_DEVICE (self), NULL); reload_wwan_iface_name (self); return self->priv->wwan_iface; } /*****************************************************************************/ /* Register/Unregister clients that want to receive indications */ static gpointer build_registered_client_key (guint8 cid, QmiService service) { return GUINT_TO_POINTER (((guint8)service << 8) | cid); } static gboolean register_client (QmiDevice *self, QmiClient *client, GError **error) { gpointer key; key = build_registered_client_key (qmi_client_get_cid (client), qmi_client_get_service (client)); /* Only add the new client if not already registered one with the same CID * for the same service */ if (g_hash_table_lookup (self->priv->registered_clients, key)) { g_set_error (error, QMI_CORE_ERROR, QMI_CORE_ERROR_FAILED, "A client with CID '%u' and service '%s' is already registered", qmi_client_get_cid (client), qmi_service_get_string (qmi_client_get_service (client))); return FALSE; } g_hash_table_insert (self->priv->registered_clients, key, g_object_ref (client)); return TRUE; } static void unregister_client (QmiDevice *self, QmiClient *client) { g_hash_table_remove (self->priv->registered_clients, build_registered_client_key (qmi_client_get_cid (client), qmi_client_get_service (client))); } /*****************************************************************************/ /* Allocate new client */ typedef struct { QmiDevice *self; GSimpleAsyncResult *result; QmiService service; GType client_type; guint8 cid; } AllocateClientContext; static void allocate_client_context_complete_and_free (AllocateClientContext *ctx) { g_simple_async_result_complete_in_idle (ctx->result); g_object_unref (ctx->result); g_object_unref (ctx->self); g_slice_free (AllocateClientContext, ctx); } /** * qmi_device_allocate_client_finish: * @self: a #QmiDevice. * @res: a #GAsyncResult. * @error: Return location for error or %NULL. * * Finishes an operation started with qmi_device_allocate_client(). * * Returns: a newly allocated #QmiClient, or #NULL if @error is set. */ QmiClient * qmi_device_allocate_client_finish (QmiDevice *self, GAsyncResult *res, GError **error) { if (g_simple_async_result_propagate_error (G_SIMPLE_ASYNC_RESULT (res), error)) return NULL; return QMI_CLIENT (g_object_ref (g_simple_async_result_get_op_res_gpointer (G_SIMPLE_ASYNC_RESULT (res)))); } static void build_client_object (AllocateClientContext *ctx) { gchar *version_string = NULL; QmiClient *client; GError *error = NULL; const QmiMessageCtlGetVersionInfoOutputServiceListService *version_info; /* We now have a proper CID for the client, we should be able to create it * right away */ client = g_object_new (ctx->client_type, QMI_CLIENT_DEVICE, ctx->self, QMI_CLIENT_SERVICE, ctx->service, QMI_CLIENT_CID, ctx->cid, NULL); /* Add version info to the client if it was retrieved */ version_info = find_service_version_info (ctx->self, ctx->service); if (version_info) g_object_set (client, QMI_CLIENT_VERSION_MAJOR, version_info->major_version, QMI_CLIENT_VERSION_MINOR, version_info->minor_version, NULL); /* Register the client to get indications */ if (!register_client (ctx->self, client, &error)) { g_prefix_error (&error, "Cannot register new client with CID '%u' and service '%s'", ctx->cid, qmi_service_get_string (ctx->service)); g_simple_async_result_take_error (ctx->result, error); allocate_client_context_complete_and_free (ctx); g_object_unref (client); return; } /* Build version string for the logging */ if (ctx->self->priv->supported_services) { const QmiMessageCtlGetVersionInfoOutputServiceListService *info; info = find_service_version_info (ctx->self, ctx->service); if (info) version_string = g_strdup_printf ("%u.%u", info->major_version, info->minor_version); } g_debug ("[%s] Registered '%s' (version %s) client with ID '%u'", ctx->self->priv->path_display, qmi_service_get_string (ctx->service), version_string ? version_string : "unknown", ctx->cid); g_free (version_string); /* Client created and registered, complete successfully */ g_simple_async_result_set_op_res_gpointer (ctx->result, client, (GDestroyNotify)g_object_unref); allocate_client_context_complete_and_free (ctx); } static void allocate_cid_ready (QmiClientCtl *client_ctl, GAsyncResult *res, AllocateClientContext *ctx) { QmiMessageCtlAllocateCidOutput *output; QmiService service; guint8 cid; GError *error = NULL; /* Check result of the async operation */ output = qmi_client_ctl_allocate_cid_finish (client_ctl, res, &error); if (!output) { g_prefix_error (&error, "CID allocation failed in the CTL client: "); g_simple_async_result_take_error (ctx->result, error); allocate_client_context_complete_and_free (ctx); return; } /* Check result of the QMI operation */ if (!qmi_message_ctl_allocate_cid_output_get_result (output, &error)) { g_simple_async_result_take_error (ctx->result, error); allocate_client_context_complete_and_free (ctx); qmi_message_ctl_allocate_cid_output_unref (output); return; } /* Allocation info is mandatory when result is success */ g_assert (qmi_message_ctl_allocate_cid_output_get_allocation_info (output, &service, &cid, NULL)); if (service != ctx->service) { g_simple_async_result_set_error ( ctx->result, QMI_CORE_ERROR, QMI_CORE_ERROR_FAILED, "CID allocation failed in the CTL client: " "Service mismatch (requested '%s', got '%s')", qmi_service_get_string (ctx->service), qmi_service_get_string (service)); allocate_client_context_complete_and_free (ctx); qmi_message_ctl_allocate_cid_output_unref (output); return; } ctx->cid = cid; build_client_object (ctx); qmi_message_ctl_allocate_cid_output_unref (output); } /** * qmi_device_allocate_client: * @self: a #QmiDevice. * @service: a valid #QmiService. * @cid: a valid client ID, or #QMI_CID_NONE. * @timeout: maximum time to wait. * @cancellable: optional #GCancellable object, #NULL to ignore. * @callback: a #GAsyncReadyCallback to call when the operation is finished. * @user_data: the data to pass to callback function. * * Asynchronously allocates a new #QmiClient in @self. * * If #QMI_CID_NONE is given in @cid, a new client ID will be allocated; * otherwise a client with the given @cid will be generated. * * When the operation is finished @callback will be called. You can then call * qmi_device_allocate_client_finish() to get the result of the operation. * * Note: Clients for the #QMI_SERVICE_CTL cannot be created with this method; * instead get/peek the implicit one from @self. */ void qmi_device_allocate_client (QmiDevice *self, QmiService service, guint8 cid, guint timeout, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data) { AllocateClientContext *ctx; g_return_if_fail (QMI_IS_DEVICE (self)); g_return_if_fail (service != QMI_SERVICE_UNKNOWN); ctx = g_slice_new0 (AllocateClientContext); ctx->self = g_object_ref (self); ctx->result = g_simple_async_result_new (G_OBJECT (self), callback, user_data, qmi_device_allocate_client); ctx->service = service; /* Check if the requested service is supported by the device */ if (!check_service_supported (self, service)) { g_simple_async_result_set_error (ctx->result, QMI_CORE_ERROR, QMI_CORE_ERROR_UNSUPPORTED, "Service '%s' not supported by the device", qmi_service_get_string (service)); allocate_client_context_complete_and_free (ctx); return; } switch (service) { case QMI_SERVICE_CTL: g_simple_async_result_set_error (ctx->result, QMI_CORE_ERROR, QMI_CORE_ERROR_INVALID_ARGS, "Cannot create additional clients for the CTL service"); allocate_client_context_complete_and_free (ctx); return; case QMI_SERVICE_DMS: ctx->client_type = QMI_TYPE_CLIENT_DMS; break; case QMI_SERVICE_WDS: ctx->client_type = QMI_TYPE_CLIENT_WDS; break; case QMI_SERVICE_NAS: ctx->client_type = QMI_TYPE_CLIENT_NAS; break; case QMI_SERVICE_WMS: ctx->client_type = QMI_TYPE_CLIENT_WMS; break; case QMI_SERVICE_PDS: ctx->client_type = QMI_TYPE_CLIENT_PDS; break; case QMI_SERVICE_PBM: ctx->client_type = QMI_TYPE_CLIENT_PBM; break; case QMI_SERVICE_UIM: ctx->client_type = QMI_TYPE_CLIENT_UIM; break; case QMI_SERVICE_OMA: ctx->client_type = QMI_TYPE_CLIENT_OMA; break; case QMI_SERVICE_WDA: ctx->client_type = QMI_TYPE_CLIENT_WDA; break; case QMI_SERVICE_VOICE: ctx->client_type = QMI_TYPE_CLIENT_VOICE; break; default: g_simple_async_result_set_error (ctx->result, QMI_CORE_ERROR, QMI_CORE_ERROR_INVALID_ARGS, "Clients for service '%s' not yet supported", qmi_service_get_string (service)); allocate_client_context_complete_and_free (ctx); return; } /* Allocate a new CID for the client to be created */ if (cid == QMI_CID_NONE) { QmiMessageCtlAllocateCidInput *input; input = qmi_message_ctl_allocate_cid_input_new (); qmi_message_ctl_allocate_cid_input_set_service (input, ctx->service, NULL); g_debug ("[%s] Allocating new client ID...", ctx->self->priv->path_display); qmi_client_ctl_allocate_cid (self->priv->client_ctl, input, timeout, cancellable, (GAsyncReadyCallback)allocate_cid_ready, ctx); qmi_message_ctl_allocate_cid_input_unref (input); return; } /* Reuse the given CID */ g_debug ("[%s] Reusing client CID '%u'...", ctx->self->priv->path_display, cid); ctx->cid = cid; build_client_object (ctx); } /*****************************************************************************/ /* Release client */ typedef struct { QmiClient *client; GSimpleAsyncResult *result; } ReleaseClientContext; static void release_client_context_complete_and_free (ReleaseClientContext *ctx) { g_simple_async_result_complete_in_idle (ctx->result); g_object_unref (ctx->result); g_object_unref (ctx->client); g_slice_free (ReleaseClientContext, ctx); } /** * qmi_device_release_client_finish: * @self: a #QmiDevice. * @res: a #GAsyncResult. * @error: Return location for error or %NULL. * * Finishes an operation started with qmi_device_release_client(). * * Note that even if the release operation returns an error, the client should * anyway be considered released, and shouldn't be used afterwards. * * Returns: %TRUE if successful, or #NULL if @error is set. */ gboolean qmi_device_release_client_finish (QmiDevice *self, GAsyncResult *res, GError **error) { return !g_simple_async_result_propagate_error (G_SIMPLE_ASYNC_RESULT (res), error); } static void client_ctl_release_cid_ready (QmiClientCtl *client_ctl, GAsyncResult *res, ReleaseClientContext *ctx) { GError *error = NULL; QmiMessageCtlReleaseCidOutput *output; /* Note: even if we return an error, the client is to be considered * released! (so shouldn't be used) */ /* Check result of the async operation */ output = qmi_client_ctl_release_cid_finish (client_ctl, res, &error); if (!output) { g_simple_async_result_take_error (ctx->result, error); release_client_context_complete_and_free (ctx); return; } /* Check result of the QMI operation */ if (!qmi_message_ctl_release_cid_output_get_result (output, &error)) { g_simple_async_result_take_error (ctx->result, error); release_client_context_complete_and_free (ctx); qmi_message_ctl_release_cid_output_unref (output); return; } g_simple_async_result_set_op_res_gboolean (ctx->result, TRUE); release_client_context_complete_and_free (ctx); qmi_message_ctl_release_cid_output_unref (output); } /** * qmi_device_release_client: * @self: a #QmiDevice. * @client: the #QmiClient to release. * @flags: mask of #QmiDeviceReleaseClientFlags specifying how the client should be released. * @timeout: maximum time to wait. * @cancellable: optional #GCancellable object, #NULL to ignore. * @callback: a #GAsyncReadyCallback to call when the operation is finished. * @user_data: the data to pass to callback function. * * Asynchronously releases the #QmiClient from the #QmiDevice. * * Once the #QmiClient has been released, it cannot be used any more to * perform operations. * * * When the operation is finished @callback will be called. You can then call * qmi_device_release_client_finish() to get the result of the operation. */ void qmi_device_release_client (QmiDevice *self, QmiClient *client, QmiDeviceReleaseClientFlags flags, guint timeout, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data) { ReleaseClientContext *ctx; QmiService service; guint8 cid; gchar *flags_str; g_return_if_fail (QMI_IS_DEVICE (self)); g_return_if_fail (QMI_IS_CLIENT (client)); cid = qmi_client_get_cid (client); service = (guint8)qmi_client_get_service (client); /* The CTL client should not have been created out of the QmiDevice */ g_return_if_fail (service != QMI_SERVICE_CTL); flags_str = qmi_device_release_client_flags_build_string_from_mask (flags); g_debug ("[%s] Releasing '%s' client with flags '%s'...", self->priv->path_display, qmi_service_get_string (service), flags_str); g_free (flags_str); /* NOTE! The operation must not take a reference to self, or we won't be * able to use it implicitly from our dispose() */ ctx = g_slice_new0 (ReleaseClientContext); ctx->client = g_object_ref (client); ctx->result = g_simple_async_result_new (G_OBJECT (self), callback, user_data, qmi_device_release_client); /* Do not try to release an already released client */ if (cid == QMI_CID_NONE) { g_simple_async_result_set_error (ctx->result, QMI_CORE_ERROR, QMI_CORE_ERROR_INVALID_ARGS, "Client is already released"); release_client_context_complete_and_free (ctx); return; } /* Unregister from device */ unregister_client (self, client); g_debug ("[%s] Unregistered '%s' client with ID '%u'", self->priv->path_display, qmi_service_get_string (service), cid); /* Reset the contents of the client object, making it unusable */ g_object_set (client, QMI_CLIENT_CID, QMI_CID_NONE, QMI_CLIENT_SERVICE, QMI_SERVICE_UNKNOWN, QMI_CLIENT_DEVICE, NULL, NULL); if (flags & QMI_DEVICE_RELEASE_CLIENT_FLAGS_RELEASE_CID) { QmiMessageCtlReleaseCidInput *input; /* And now, really try to release the CID */ input = qmi_message_ctl_release_cid_input_new (); qmi_message_ctl_release_cid_input_set_release_info (input, service,cid, NULL); /* And now, really try to release the CID */ qmi_client_ctl_release_cid (self->priv->client_ctl, input, timeout, cancellable, (GAsyncReadyCallback)client_ctl_release_cid_ready, ctx); qmi_message_ctl_release_cid_input_unref (input); return; } /* No need to release the CID, so just done */ g_simple_async_result_set_op_res_gboolean (ctx->result, TRUE); release_client_context_complete_and_free (ctx); return; } /*****************************************************************************/ /* Set instance ID */ /** * qmi_device_set_instance_id_finish: * @self: a #QmiDevice. * @res: a #GAsyncResult. * @link_id: a placeholder for the output #guint16, or #NULL if not required. * @error: Return location for error or %NULL. * * Finishes an operation started with qmi_device_set_instance_id(). * * Returns: %TRUE if successful, %FALSE if @error is set. */ gboolean qmi_device_set_instance_id_finish (QmiDevice *self, GAsyncResult *res, guint16 *link_id, GError **error) { if (g_simple_async_result_propagate_error (G_SIMPLE_ASYNC_RESULT (res), error)) return FALSE; if (link_id) *link_id = ((guint16) GPOINTER_TO_UINT (g_simple_async_result_get_op_res_gpointer (G_SIMPLE_ASYNC_RESULT (res)))); return TRUE; } static void set_instance_id_ready (QmiClientCtl *client_ctl, GAsyncResult *res, GSimpleAsyncResult *simple) { QmiMessageCtlSetInstanceIdOutput *output; GError *error = NULL; /* Check result of the async operation */ output = qmi_client_ctl_set_instance_id_finish (client_ctl, res, &error); if (!output) g_simple_async_result_take_error (simple, error); else { /* Check result of the QMI operation */ if (!qmi_message_ctl_set_instance_id_output_get_result (output, &error)) g_simple_async_result_take_error (simple, error); else { guint16 link_id; qmi_message_ctl_set_instance_id_output_get_link_id (output, &link_id, NULL); g_simple_async_result_set_op_res_gpointer (simple, GUINT_TO_POINTER ((guint)link_id), NULL); } qmi_message_ctl_set_instance_id_output_unref (output); } g_simple_async_result_complete (simple); } /** * qmi_device_set_instance_id: * @self: a #QmiDevice. * @instance_id: the instance ID. * @timeout: maximum time to wait. * @cancellable: optional #GCancellable object, #NULL to ignore. * @callback: a #GAsyncReadyCallback to call when the operation is finished. * @user_data: the data to pass to callback function. * * Sets the instance ID of the #QmiDevice. * * When the operation is finished @callback will be called. You can then call * qmi_device_set_instance_id_finish() to get the result of the operation. */ void qmi_device_set_instance_id (QmiDevice *self, guint8 instance_id, guint timeout, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data) { GSimpleAsyncResult *result; QmiMessageCtlSetInstanceIdInput *input; result = g_simple_async_result_new (G_OBJECT (self), callback, user_data, qmi_device_set_instance_id); input = qmi_message_ctl_set_instance_id_input_new (); qmi_message_ctl_set_instance_id_input_set_id ( input, instance_id, NULL); qmi_client_ctl_set_instance_id (self->priv->client_ctl, input, timeout, cancellable, (GAsyncReadyCallback)set_instance_id_ready, result); qmi_message_ctl_set_instance_id_input_unref (input); } /*****************************************************************************/ /* Input channel processing */ typedef struct { QmiClient *client; QmiMessage *message; } IdleIndicationContext; static gboolean process_indication_idle (IdleIndicationContext *ctx) { g_assert (ctx->client != NULL); g_assert (ctx->message != NULL); qmi_client_process_indication (ctx->client, ctx->message); g_object_unref (ctx->client); qmi_message_unref (ctx->message); g_slice_free (IdleIndicationContext, ctx); return FALSE; } static void report_indication (QmiClient *client, QmiMessage *message) { IdleIndicationContext *ctx; GSource *source; /* Setup an idle to Pass the indication down to the client */ ctx = g_slice_new (IdleIndicationContext); ctx->client = g_object_ref (client); ctx->message = qmi_message_ref (message); source = g_idle_source_new (); g_source_set_callback (source, (GSourceFunc)process_indication_idle, ctx, NULL); g_source_attach (source, g_main_context_get_thread_default ()); g_source_unref (source); } static void process_message (QmiDevice *self, QmiMessage *message) { if (qmi_utils_get_traces_enabled ()) { gchar *printable; printable = __qmi_utils_str_hex (((GByteArray *)message)->data, ((GByteArray *)message)->len, ':'); g_debug ("[%s] Received message...\n" ">>>>>> RAW:\n" ">>>>>> length = %u\n" ">>>>>> data = %s\n", self->priv->path_display, ((GByteArray *)message)->len, printable); g_free (printable); printable = qmi_message_get_printable (message, ">>>>>> "); g_debug ("[%s] Received message (translated)...\n%s", self->priv->path_display, printable); g_free (printable); } if (qmi_message_is_indication (message)) { /* Generic emission of the indication */ g_signal_emit (self, signals[SIGNAL_INDICATION], 0, message); if (qmi_message_get_client_id (message) == QMI_CID_BROADCAST) { GHashTableIter iter; gpointer key; QmiClient *client; g_hash_table_iter_init (&iter, self->priv->registered_clients); while (g_hash_table_iter_next (&iter, &key, (gpointer *)&client)) { /* For broadcast messages, report them just if the service matches */ if (qmi_message_get_service (message) == qmi_client_get_service (client)) report_indication (client, message); } } else { QmiClient *client; client = g_hash_table_lookup (self->priv->registered_clients, build_registered_client_key (qmi_message_get_client_id (message), qmi_message_get_service (message))); if (client) report_indication (client, message); } return; } if (qmi_message_is_response (message)) { Transaction *tr; tr = device_match_transaction (self, message); if (!tr) g_debug ("[%s] No transaction matched in received message", self->priv->path_display); else /* Report the reply message */ transaction_complete_and_free (tr, message, NULL); return; } g_debug ("[%s] Message received but it is neither an indication nor a response. Skipping it.", self->priv->path_display); } static void parse_response (QmiDevice *self) { do { GError *error = NULL; QmiMessage *message; /* Every message received must start with the QMUX marker. * If it doesn't, we broke framing :-/ * If we broke framing, an error should be reported and the device * should get closed */ if (self->priv->buffer->len > 0 && self->priv->buffer->data[0] != QMI_MESSAGE_QMUX_MARKER) { /* TODO: Report fatal error */ g_warning ("[%s] QMI framing error detected", self->priv->path_display); return; } message = qmi_message_new_from_raw (self->priv->buffer, &error); if (!message) { if (!error) /* More data we need */ return; /* Warn about the issue */ g_warning ("[%s] Invalid QMI message received: '%s'", self->priv->path_display, error->message); g_error_free (error); } else { /* Play with the received message */ process_message (self, message); qmi_message_unref (message); } } while (self->priv->buffer->len > 0); } static gboolean input_ready_cb (GInputStream *istream, QmiDevice *self) { guint8 buffer[BUFFER_SIZE]; GError *error = NULL; gssize r; r = g_pollable_input_stream_read_nonblocking (G_POLLABLE_INPUT_STREAM (istream), buffer, BUFFER_SIZE, NULL, &error); if (r < 0) { g_warning ("Error reading from istream: %s", error ? error->message : "unknown"); if (error) g_error_free (error); /* Close the device */ qmi_device_close (self, NULL); return FALSE; } if (r == 0) { /* HUP! */ g_warning ("Cannot read from istream: connection broken"); return FALSE; } /* else, r > 0 */ if (!G_UNLIKELY (self->priv->buffer)) self->priv->buffer = g_byte_array_sized_new (r); g_byte_array_append (self->priv->buffer, buffer, r); /* Try to parse input messages */ parse_response (self); return TRUE; } typedef struct { QmiDevice *self; GSimpleAsyncResult *result; guint spawn_retries; } CreateIostreamContext; static void create_iostream_context_complete_and_free (CreateIostreamContext *ctx) { g_simple_async_result_complete_in_idle (ctx->result); g_object_unref (ctx->result); g_object_unref (ctx->self); g_slice_free (CreateIostreamContext, ctx); } static gboolean create_iostream_finish (QmiDevice *self, GAsyncResult *res, GError **error) { return !g_simple_async_result_propagate_error (G_SIMPLE_ASYNC_RESULT (res), error); } static void setup_iostream (CreateIostreamContext *ctx) { /* Check in/out streams */ if (!ctx->self->priv->istream || !ctx->self->priv->ostream) { g_simple_async_result_set_error ( ctx->result, QMI_CORE_ERROR, QMI_CORE_ERROR_FAILED, "Cannot get input/output streams"); g_clear_object (&ctx->self->priv->istream); g_clear_object (&ctx->self->priv->ostream); g_clear_object (&ctx->self->priv->socket_connection); g_clear_object (&ctx->self->priv->socket_client); create_iostream_context_complete_and_free (ctx); return; } /* Setup input events */ ctx->self->priv->input_source = (g_pollable_input_stream_create_source ( G_POLLABLE_INPUT_STREAM ( ctx->self->priv->istream), NULL)); g_source_set_callback (ctx->self->priv->input_source, (GSourceFunc)input_ready_cb, ctx->self, NULL); g_source_attach (ctx->self->priv->input_source, g_main_context_get_thread_default ()); g_simple_async_result_set_op_res_gboolean (ctx->result, TRUE); create_iostream_context_complete_and_free (ctx); } static void create_iostream_with_fd (CreateIostreamContext *ctx) { gint fd; fd = open (ctx->self->priv->path, O_RDWR | O_EXCL | O_NONBLOCK | O_NOCTTY); if (fd < 0) { g_simple_async_result_set_error ( ctx->result, QMI_CORE_ERROR, QMI_CORE_ERROR_FAILED, "Cannot open device file '%s': %s", ctx->self->priv->path_display, strerror (errno)); create_iostream_context_complete_and_free (ctx); return; } ctx->self->priv->istream = g_unix_input_stream_new (fd, TRUE); ctx->self->priv->ostream = g_unix_output_stream_new (fd, TRUE); setup_iostream (ctx); } static void create_iostream_with_socket (CreateIostreamContext *ctx); static gboolean wait_for_proxy_cb (CreateIostreamContext *ctx) { create_iostream_with_socket (ctx); return FALSE; } static void create_iostream_with_socket (CreateIostreamContext *ctx) { GSocketAddress *socket_address; GError *error = NULL; /* Create socket client */ ctx->self->priv->socket_client = g_socket_client_new (); g_socket_client_set_family (ctx->self->priv->socket_client, G_SOCKET_FAMILY_UNIX); g_socket_client_set_socket_type (ctx->self->priv->socket_client, G_SOCKET_TYPE_STREAM); g_socket_client_set_protocol (ctx->self->priv->socket_client, G_SOCKET_PROTOCOL_DEFAULT); /* Setup socket address */ socket_address = (g_unix_socket_address_new_with_type ( ctx->self->priv->proxy_path, -1, G_UNIX_SOCKET_ADDRESS_ABSTRACT)); /* Connect to address */ ctx->self->priv->socket_connection = (g_socket_client_connect ( ctx->self->priv->socket_client, G_SOCKET_CONNECTABLE (socket_address), NULL, &error)); g_object_unref (socket_address); if (!ctx->self->priv->socket_connection) { gchar **argc; GSource *source; g_debug ("cannot connect to proxy: %s", error->message); g_clear_error (&error); g_clear_object (&ctx->self->priv->socket_client); /* Don't retry forever */ ctx->spawn_retries++; if (ctx->spawn_retries > MAX_SPAWN_RETRIES) { g_simple_async_result_set_error (ctx->result, QMI_CORE_ERROR, QMI_CORE_ERROR_FAILED, "Couldn't spawn the qmi-proxy"); create_iostream_context_complete_and_free (ctx); return; } g_debug ("spawning new qmi-proxy (try %u)...", ctx->spawn_retries); argc = g_new0 (gchar *, 2); argc[0] = g_strdup (LIBEXEC_PATH "/qmi-proxy"); if (!g_spawn_async (NULL, /* working directory */ argc, NULL, /* envp */ G_SPAWN_STDOUT_TO_DEV_NULL | G_SPAWN_STDERR_TO_DEV_NULL, NULL, /* child_setup */ NULL, /* child_setup_user_data */ NULL, &error)) { g_debug ("error spawning qmi-proxy: %s", error->message); g_clear_error (&error); } g_strfreev (argc); /* Wait some ms and retry */ source = g_timeout_source_new (100); g_source_set_callback (source, (GSourceFunc)wait_for_proxy_cb, ctx, NULL); g_source_attach (source, g_main_context_get_thread_default ()); g_source_unref (source); return; } ctx->self->priv->istream = g_io_stream_get_input_stream (G_IO_STREAM (ctx->self->priv->socket_connection)); if (ctx->self->priv->istream) g_object_ref (ctx->self->priv->istream); ctx->self->priv->ostream = g_io_stream_get_output_stream (G_IO_STREAM (ctx->self->priv->socket_connection)); if (ctx->self->priv->ostream) g_object_ref (ctx->self->priv->ostream); setup_iostream (ctx); } static void create_iostream (QmiDevice *self, gboolean proxy, GAsyncReadyCallback callback, gpointer user_data) { CreateIostreamContext *ctx; ctx = g_slice_new (CreateIostreamContext); ctx->self = g_object_ref (self); ctx->result = g_simple_async_result_new (G_OBJECT (self), callback, user_data, create_iostream); ctx->spawn_retries = 0; if (self->priv->istream || self->priv->ostream) { g_simple_async_result_set_error (ctx->result, QMI_CORE_ERROR, QMI_CORE_ERROR_WRONG_STATE, "Already open"); create_iostream_context_complete_and_free (ctx); return; } g_assert (self->priv->file); g_assert (self->priv->path); if (proxy) create_iostream_with_socket (ctx); else create_iostream_with_fd (ctx); } /*****************************************************************************/ /* Open device */ typedef enum { DEVICE_OPEN_CONTEXT_STEP_FIRST = 0, DEVICE_OPEN_CONTEXT_STEP_CREATE_IOSTREAM, DEVICE_OPEN_CONTEXT_STEP_FLAGS_PROXY, DEVICE_OPEN_CONTEXT_STEP_FLAGS_VERSION_INFO, DEVICE_OPEN_CONTEXT_STEP_FLAGS_SYNC, DEVICE_OPEN_CONTEXT_STEP_FLAGS_NETPORT, DEVICE_OPEN_CONTEXT_STEP_LAST } DeviceOpenContextStep; typedef struct { QmiDevice *self; GSimpleAsyncResult *result; GCancellable *cancellable; DeviceOpenContextStep step; QmiDeviceOpenFlags flags; guint timeout; guint version_check_retries; } DeviceOpenContext; static void device_open_context_complete_and_free (DeviceOpenContext *ctx) { g_simple_async_result_complete_in_idle (ctx->result); g_object_unref (ctx->result); if (ctx->cancellable) g_object_unref (ctx->cancellable); g_object_unref (ctx->self); g_slice_free (DeviceOpenContext, ctx); } /** * qmi_device_open_finish: * @self: a #QmiDevice. * @res: a #GAsyncResult. * @error: Return location for error or %NULL. * * Finishes an asynchronous open operation started with qmi_device_open(). * * Returns: %TRUE if successful, %FALSE if @error is set. */ gboolean qmi_device_open_finish (QmiDevice *self, GAsyncResult *res, GError **error) { return !g_simple_async_result_propagate_error (G_SIMPLE_ASYNC_RESULT (res), error); } static void device_open_context_step (DeviceOpenContext *ctx); static void ctl_set_data_format_ready (QmiClientCtl *client, GAsyncResult *res, DeviceOpenContext *ctx) { QmiMessageCtlSetDataFormatOutput *output = NULL; GError *error = NULL; output = qmi_client_ctl_set_data_format_finish (client, res, &error); /* Check result of the async operation */ if (!output) { g_simple_async_result_take_error (ctx->result, error); device_open_context_complete_and_free (ctx); return; } /* Check result of the QMI operation */ if (!qmi_message_ctl_set_data_format_output_get_result (output, &error)) { g_simple_async_result_take_error (ctx->result, error); device_open_context_complete_and_free (ctx); qmi_message_ctl_set_data_format_output_unref (output); return; } g_debug ("[%s] Network port data format operation finished", ctx->self->priv->path_display); qmi_message_ctl_set_data_format_output_unref (output); /* Go on */ ctx->step++; device_open_context_step (ctx); } static void sync_ready (QmiClientCtl *client_ctl, GAsyncResult *res, DeviceOpenContext *ctx) { GError *error = NULL; QmiMessageCtlSyncOutput *output; /* Check result of the async operation */ output = qmi_client_ctl_sync_finish (client_ctl, res, &error); if(!output) { g_simple_async_result_take_error (ctx->result, error); device_open_context_complete_and_free (ctx); return; } /* Check result of the QMI operation */ if (!qmi_message_ctl_sync_output_get_result (output, &error)) { g_simple_async_result_take_error (ctx->result, error); device_open_context_complete_and_free (ctx); qmi_message_ctl_sync_output_unref (output); return; } g_debug ("[%s] Sync operation finished", ctx->self->priv->path_display); qmi_message_ctl_sync_output_unref (output); /* Go on */ ctx->step++; device_open_context_step (ctx); } static void open_version_info_ready (QmiClientCtl *client_ctl, GAsyncResult *res, DeviceOpenContext *ctx) { GArray *service_list; QmiMessageCtlGetVersionInfoOutput *output; GError *error = NULL; guint i; /* Check result of the async operation */ output = qmi_client_ctl_get_version_info_finish (client_ctl, res, &error); if (!output) { if (g_error_matches (error, QMI_CORE_ERROR, QMI_CORE_ERROR_TIMEOUT)) { /* Update retries... */ ctx->version_check_retries--; /* If retries left, retry */ if (ctx->version_check_retries > 0) { g_error_free (error); qmi_client_ctl_get_version_info (ctx->self->priv->client_ctl, NULL, 1, ctx->cancellable, (GAsyncReadyCallback)open_version_info_ready, ctx); return; } /* Otherwise, propagate the error */ } g_simple_async_result_take_error (ctx->result, error); device_open_context_complete_and_free (ctx); return; } /* Check result of the QMI operation */ if (!qmi_message_ctl_get_version_info_output_get_result (output, &error)) { g_simple_async_result_take_error (ctx->result, error); device_open_context_complete_and_free (ctx); qmi_message_ctl_get_version_info_output_unref (output); return; } /* QMI operation succeeded, we can now get the outputs */ service_list = NULL; qmi_message_ctl_get_version_info_output_get_service_list (output, &service_list, NULL); ctx->self->priv->supported_services = g_array_ref (service_list); g_debug ("[%s] QMI Device supports %u services:", ctx->self->priv->path_display, ctx->self->priv->supported_services->len); for (i = 0; i < ctx->self->priv->supported_services->len; i++) { QmiMessageCtlGetVersionInfoOutputServiceListService *info; const gchar *service_str; info = &g_array_index (ctx->self->priv->supported_services, QmiMessageCtlGetVersionInfoOutputServiceListService, i); service_str = qmi_service_get_string (info->service); if (service_str) g_debug ("[%s] %s (%u.%u)", ctx->self->priv->path_display, service_str, info->major_version, info->minor_version); else g_debug ("[%s] unknown [0x%02x] (%u.%u)", ctx->self->priv->path_display, info->service, info->major_version, info->minor_version); } qmi_message_ctl_get_version_info_output_unref (output); /* Go on */ ctx->step++; device_open_context_step (ctx); } static void internal_proxy_open_ready (QmiClientCtl *client_ctl, GAsyncResult *res, DeviceOpenContext *ctx) { QmiMessageCtlInternalProxyOpenOutput *output; GError *error = NULL; /* Check result of the async operation */ output = qmi_client_ctl_internal_proxy_open_finish (client_ctl, res, &error); if (!output) { g_simple_async_result_take_error (ctx->result, error); device_open_context_complete_and_free (ctx); return; } /* Check result of the QMI operation */ if (!qmi_message_ctl_internal_proxy_open_output_get_result (output, &error)) { g_simple_async_result_take_error (ctx->result, error); device_open_context_complete_and_free (ctx); qmi_message_ctl_internal_proxy_open_output_unref (output); return; } qmi_message_ctl_internal_proxy_open_output_unref (output); /* Go on */ ctx->step++; device_open_context_step (ctx); } static void create_iostream_ready (QmiDevice *self, GAsyncResult *res, DeviceOpenContext *ctx) { GError *error = NULL; if (!create_iostream_finish (self, res, &error)) { g_simple_async_result_take_error (ctx->result, error); device_open_context_complete_and_free (ctx); return; } /* Go on */ ctx->step++; device_open_context_step (ctx); } #define NETPORT_FLAGS (QMI_DEVICE_OPEN_FLAGS_NET_802_3 | \ QMI_DEVICE_OPEN_FLAGS_NET_RAW_IP | \ QMI_DEVICE_OPEN_FLAGS_NET_QOS_HEADER | \ QMI_DEVICE_OPEN_FLAGS_NET_NO_QOS_HEADER) static void device_open_context_step (DeviceOpenContext *ctx) { switch (ctx->step) { case DEVICE_OPEN_CONTEXT_STEP_FIRST: ctx->step++; /* Fall down */ case DEVICE_OPEN_CONTEXT_STEP_CREATE_IOSTREAM: create_iostream (ctx->self, !!(ctx->flags & QMI_DEVICE_OPEN_FLAGS_PROXY), (GAsyncReadyCallback)create_iostream_ready, ctx); return; case DEVICE_OPEN_CONTEXT_STEP_FLAGS_PROXY: /* Initialize communication with proxy? */ if (ctx->flags & QMI_DEVICE_OPEN_FLAGS_PROXY) { QmiMessageCtlInternalProxyOpenInput *input; input = qmi_message_ctl_internal_proxy_open_input_new (); qmi_message_ctl_internal_proxy_open_input_set_device_path (input, ctx->self->priv->path, NULL); qmi_client_ctl_internal_proxy_open (ctx->self->priv->client_ctl, input, 5, ctx->cancellable, (GAsyncReadyCallback)internal_proxy_open_ready, ctx); qmi_message_ctl_internal_proxy_open_input_unref (input); return; } ctx->step++; /* Fall down */ case DEVICE_OPEN_CONTEXT_STEP_FLAGS_VERSION_INFO: /* Query version info? */ if (ctx->flags & QMI_DEVICE_OPEN_FLAGS_VERSION_INFO) { /* Setup how many times to retry... We'll retry once per second */ ctx->version_check_retries = ctx->timeout > 0 ? ctx->timeout : 1; g_debug ("[%s] Checking version info (%u retries)...", ctx->self->priv->path_display, ctx->version_check_retries); qmi_client_ctl_get_version_info (ctx->self->priv->client_ctl, NULL, 1, ctx->cancellable, (GAsyncReadyCallback)open_version_info_ready, ctx); return; } ctx->step++; /* Fall down */ case DEVICE_OPEN_CONTEXT_STEP_FLAGS_SYNC: /* Sync? */ if (ctx->flags & QMI_DEVICE_OPEN_FLAGS_SYNC) { g_debug ("[%s] Running sync...", ctx->self->priv->path_display); qmi_client_ctl_sync (ctx->self->priv->client_ctl, NULL, ctx->timeout, ctx->cancellable, (GAsyncReadyCallback)sync_ready, ctx); return; } ctx->step++; /* Fall down */ case DEVICE_OPEN_CONTEXT_STEP_FLAGS_NETPORT: /* Network port setup */ if (ctx->flags & NETPORT_FLAGS) { QmiMessageCtlSetDataFormatInput *input; QmiCtlDataFormat qos = QMI_CTL_DATA_FORMAT_QOS_FLOW_HEADER_ABSENT; QmiCtlDataLinkProtocol link_protocol = QMI_CTL_DATA_LINK_PROTOCOL_802_3; g_debug ("[%s] Setting network port data format...", ctx->self->priv->path_display); input = qmi_message_ctl_set_data_format_input_new (); if (ctx->flags & QMI_DEVICE_OPEN_FLAGS_NET_QOS_HEADER) qos = QMI_CTL_DATA_FORMAT_QOS_FLOW_HEADER_PRESENT; qmi_message_ctl_set_data_format_input_set_format (input, qos, NULL); if (ctx->flags & QMI_DEVICE_OPEN_FLAGS_NET_RAW_IP) link_protocol = QMI_CTL_DATA_LINK_PROTOCOL_RAW_IP; qmi_message_ctl_set_data_format_input_set_protocol (input, link_protocol, NULL); qmi_client_ctl_set_data_format (ctx->self->priv->client_ctl, input, 5, NULL, (GAsyncReadyCallback)ctl_set_data_format_ready, ctx); qmi_message_ctl_set_data_format_input_unref (input); return; } ctx->step++; /* Fall down */ case DEVICE_OPEN_CONTEXT_STEP_LAST: /* Nothing else to process, done we are */ g_simple_async_result_set_op_res_gboolean (ctx->result, TRUE); device_open_context_complete_and_free (ctx); return; default: break; } g_assert_not_reached (); } /** * qmi_device_open: * @self: a #QmiDevice. * @flags: mask of #QmiDeviceOpenFlags specifying how the device should be opened. * @timeout: maximum time, in seconds, to wait for the device to be opened. * @cancellable: optional #GCancellable object, #NULL to ignore. * @callback: a #GAsyncReadyCallback to call when the operation is finished. * @user_data: the data to pass to callback function. * * Asynchronously opens a #QmiDevice for I/O. * * When the operation is finished @callback will be called. You can then call * qmi_device_open_finish() to get the result of the operation. */ void qmi_device_open (QmiDevice *self, QmiDeviceOpenFlags flags, guint timeout, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data) { DeviceOpenContext *ctx; gchar *flags_str; /* Raw IP and 802.3 are mutually exclusive */ g_return_if_fail (!((flags & QMI_DEVICE_OPEN_FLAGS_NET_802_3) && (flags & QMI_DEVICE_OPEN_FLAGS_NET_RAW_IP))); /* QoS and no QoS are mutually exclusive */ g_return_if_fail (!((flags & QMI_DEVICE_OPEN_FLAGS_NET_QOS_HEADER) && (flags & QMI_DEVICE_OPEN_FLAGS_NET_NO_QOS_HEADER))); /* At least one of both link protocol and QoS must be specified */ if (flags & (QMI_DEVICE_OPEN_FLAGS_NET_802_3 | QMI_DEVICE_OPEN_FLAGS_NET_RAW_IP)) g_return_if_fail (flags & (QMI_DEVICE_OPEN_FLAGS_NET_QOS_HEADER | QMI_DEVICE_OPEN_FLAGS_NET_NO_QOS_HEADER)); if (flags & (QMI_DEVICE_OPEN_FLAGS_NET_QOS_HEADER | QMI_DEVICE_OPEN_FLAGS_NET_NO_QOS_HEADER)) g_return_if_fail (flags & (QMI_DEVICE_OPEN_FLAGS_NET_802_3 | QMI_DEVICE_OPEN_FLAGS_NET_RAW_IP)); g_return_if_fail (QMI_IS_DEVICE (self)); flags_str = qmi_device_open_flags_build_string_from_mask (flags); g_debug ("[%s] Opening device with flags '%s'...", self->priv->path_display, flags_str); g_free (flags_str); ctx = g_slice_new (DeviceOpenContext); ctx->self = g_object_ref (self); ctx->result = g_simple_async_result_new (G_OBJECT (self), callback, user_data, qmi_device_open); ctx->step = DEVICE_OPEN_CONTEXT_STEP_FIRST; ctx->flags = flags; ctx->timeout = timeout; ctx->cancellable = (cancellable ? g_object_ref (cancellable) : NULL); /* Start processing */ device_open_context_step (ctx); } /*****************************************************************************/ /* Close stream */ static gboolean destroy_iostream (QmiDevice *self, GError **error) { GError *inner_error = NULL; /* Already closed? */ if (!self->priv->istream && !self->priv->ostream) return TRUE; if (self->priv->input_source) { g_source_destroy (self->priv->input_source); g_source_unref (self->priv->input_source); self->priv->input_source = NULL; } if (self->priv->buffer) { g_byte_array_unref (self->priv->buffer); self->priv->buffer = NULL; } /* Failures when closing still make the device to get closed */ g_clear_object (&self->priv->istream); g_clear_object (&self->priv->ostream); g_clear_object (&self->priv->socket_connection); g_clear_object (&self->priv->socket_client); if (inner_error) { g_propagate_error (error, inner_error); return FALSE; } return TRUE; } /** * qmi_device_close: * @self: a #QmiDevice * @error: Return location for error or %NULL. * * Synchronously closes a #QmiDevice, preventing any further I/O. * * Closing a #QmiDevice multiple times will not return an error. * * Returns: %TRUE if successful, %FALSE if @error is set. */ gboolean qmi_device_close (QmiDevice *self, GError **error) { g_return_val_if_fail (QMI_IS_DEVICE (self), FALSE); if (!destroy_iostream (self, error)) { g_prefix_error (error, "Cannot close QMI device: "); return FALSE; } return TRUE; } /*****************************************************************************/ /* Command */ /** * qmi_device_command_finish: * @self: a #QmiDevice. * @res: a #GAsyncResult. * @error: Return location for error or %NULL. * * Finishes an operation started with qmi_device_command(). * * Returns: a #QmiMessage response, or #NULL if @error is set. The returned value should be freed with qmi_message_unref(). */ QmiMessage * qmi_device_command_finish (QmiDevice *self, GAsyncResult *res, GError **error) { if (g_simple_async_result_propagate_error (G_SIMPLE_ASYNC_RESULT (res), error)) return NULL; return qmi_message_ref (g_simple_async_result_get_op_res_gpointer ( G_SIMPLE_ASYNC_RESULT (res))); } /** * qmi_device_command: * @self: a #QmiDevice. * @message: the message to send. * @timeout: maximum time, in seconds, to wait for the response. * @cancellable: a #GCancellable, or %NULL. * @callback: a #GAsyncReadyCallback to call when the operation is finished. * @user_data: the data to pass to callback function. * * Asynchronously sends a #QmiMessage to the device. * * When the operation is finished @callback will be called. You can then call * qmi_device_command_finish() to get the result of the operation. */ void qmi_device_command (QmiDevice *self, QmiMessage *message, guint timeout, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data) { GError *error = NULL; Transaction *tr; gconstpointer raw_message; gsize raw_message_len; g_return_if_fail (QMI_IS_DEVICE (self)); g_return_if_fail (message != NULL); /* Use a proper transaction id for CTL messages if they don't have one */ if (qmi_message_get_service (message) == QMI_SERVICE_CTL && qmi_message_get_transaction_id (message) == 0) { qmi_message_set_transaction_id ( message, qmi_client_get_next_transaction_id ( QMI_CLIENT ( self->priv->client_ctl))); } tr = transaction_new (self, message, cancellable, callback, user_data); /* Device must be open */ if (!self->priv->istream || !self->priv->ostream) { error = g_error_new (QMI_CORE_ERROR, QMI_CORE_ERROR_WRONG_STATE, "Device must be open to send commands"); transaction_complete_and_free (tr, NULL, error); g_error_free (error); return; } /* Non-CTL services should use a proper CID */ if (qmi_message_get_service (message) != QMI_SERVICE_CTL && qmi_message_get_client_id (message) == 0) { error = g_error_new (QMI_CORE_ERROR, QMI_CORE_ERROR_FAILED, "Cannot send message in service '%s' without a CID", qmi_service_get_string (qmi_message_get_service (message))); transaction_complete_and_free (tr, NULL, error); g_error_free (error); return; } /* Check if the message to be sent is supported by the device * (only applicable if we did version info check when opening) */ if (!check_message_supported (self, message, &error)) { g_prefix_error (&error, "Cannot send message: "); transaction_complete_and_free (tr, NULL, error); g_error_free (error); return; } /* Get raw message */ raw_message = qmi_message_get_raw (message, &raw_message_len, &error); if (!raw_message) { g_prefix_error (&error, "Cannot get raw message: "); transaction_complete_and_free (tr, NULL, error); g_error_free (error); return; } /* Setup context to match response */ if (!device_store_transaction (self, tr, timeout, &error)) { g_prefix_error (&error, "Cannot store transaction: "); transaction_complete_and_free (tr, NULL, error); g_error_free (error); return; } if (qmi_utils_get_traces_enabled ()) { gchar *printable; printable = __qmi_utils_str_hex (((GByteArray *)message)->data, ((GByteArray *)message)->len, ':'); g_debug ("[%s] Sent message...\n" "<<<<<< RAW:\n" "<<<<<< length = %u\n" "<<<<<< data = %s\n", self->priv->path_display, ((GByteArray *)message)->len, printable); g_free (printable); printable = qmi_message_get_printable (message, "<<<<<< "); g_debug ("[%s] Sent message (translated)...\n%s", self->priv->path_display, printable); g_free (printable); } if (!g_output_stream_write_all (self->priv->ostream, raw_message, raw_message_len, NULL, /* bytes_written */ NULL, /* cancellable */ &error)) { g_prefix_error (&error, "Cannot write message: "); /* Match transaction so that we remove it from our tracking table */ tr = device_match_transaction (self, message); transaction_complete_and_free (tr, NULL, error); g_error_free (error); } /* Flush explicitly if correctly written */ g_output_stream_flush (self->priv->ostream, NULL, NULL); } /*****************************************************************************/ /* New QMI device */ /** * qmi_device_new_finish: * @res: a #GAsyncResult. * @error: Return location for error or %NULL. * * Finishes an operation started with qmi_device_new(). * * Returns: A newly created #QmiDevice, or #NULL if @error is set. */ QmiDevice * qmi_device_new_finish (GAsyncResult *res, GError **error) { GObject *ret; GObject *source_object; source_object = g_async_result_get_source_object (res); ret = g_async_initable_new_finish (G_ASYNC_INITABLE (source_object), res, error); g_object_unref (source_object); return (ret ? QMI_DEVICE (ret) : NULL); } /** * qmi_device_new: * @file: a #GFile. * @cancellable: optional #GCancellable object, #NULL to ignore. * @callback: a #GAsyncReadyCallback to call when the initialization is finished. * @user_data: the data to pass to callback function. * * Asynchronously creates a #QmiDevice object to manage @file. * When the operation is finished, @callback will be invoked. You can then call * qmi_device_new_finish() to get the result of the operation. */ void qmi_device_new (GFile *file, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data) { g_async_initable_new_async (QMI_TYPE_DEVICE, G_PRIORITY_DEFAULT, cancellable, callback, user_data, QMI_DEVICE_FILE, file, NULL); } /*****************************************************************************/ /* Async init */ typedef struct { QmiDevice *self; GSimpleAsyncResult *result; GCancellable *cancellable; } InitContext; static void init_context_complete_and_free (InitContext *ctx) { g_simple_async_result_complete_in_idle (ctx->result); if (ctx->cancellable) g_object_unref (ctx->cancellable); g_object_unref (ctx->result); g_object_unref (ctx->self); g_slice_free (InitContext, ctx); } static gboolean initable_init_finish (GAsyncInitable *initable, GAsyncResult *result, GError **error) { return !g_simple_async_result_propagate_error (G_SIMPLE_ASYNC_RESULT (result), error); } static void sync_indication_cb (QmiClientCtl *client_ctl, QmiDevice *self) { /* Just log about it */ g_debug ("[%s] Sync indication received", self->priv->path_display); } static void client_ctl_setup (InitContext *ctx) { GError *error = NULL; /* Create the implicit CTL client */ ctx->self->priv->client_ctl = g_object_new (QMI_TYPE_CLIENT_CTL, QMI_CLIENT_DEVICE, ctx->self, QMI_CLIENT_SERVICE, QMI_SERVICE_CTL, QMI_CLIENT_CID, QMI_CID_NONE, NULL); /* Register the CTL client to get indications */ register_client (ctx->self, QMI_CLIENT (ctx->self->priv->client_ctl), &error); g_assert_no_error (error); /* Connect to 'Sync' indications */ ctx->self->priv->sync_indication_id = g_signal_connect (ctx->self->priv->client_ctl, "sync", G_CALLBACK (sync_indication_cb), ctx->self); /* Done we are */ g_simple_async_result_set_op_res_gboolean (ctx->result, TRUE); init_context_complete_and_free (ctx); } static void query_info_async_ready (GFile *file, GAsyncResult *res, InitContext *ctx) { GError *error = NULL; GFileInfo *info; info = g_file_query_info_finish (file, res, &error); if (!info) { g_prefix_error (&error, "Couldn't query file info: "); g_simple_async_result_take_error (ctx->result, error); init_context_complete_and_free (ctx); return; } /* Our QMI device must be of SPECIAL type */ if (g_file_info_get_file_type (info) != G_FILE_TYPE_SPECIAL) { g_simple_async_result_set_error (ctx->result, QMI_CORE_ERROR, QMI_CORE_ERROR_FAILED, "Wrong file type"); init_context_complete_and_free (ctx); return; } g_object_unref (info); /* Go on with client CTL setup */ client_ctl_setup (ctx); } static void initable_init_async (GAsyncInitable *initable, int io_priority, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data) { InitContext *ctx; ctx = g_slice_new0 (InitContext); ctx->self = g_object_ref (initable); if (cancellable) ctx->cancellable = g_object_ref (cancellable); ctx->result = g_simple_async_result_new (G_OBJECT (initable), callback, user_data, initable_init_async); /* We need a proper file to initialize */ if (!ctx->self->priv->file) { g_simple_async_result_set_error (ctx->result, QMI_CORE_ERROR, QMI_CORE_ERROR_INVALID_ARGS, "Cannot initialize QMI device: No file given"); init_context_complete_and_free (ctx); return; } /* If no file check requested, don't do it */ if (ctx->self->priv->no_file_check) { client_ctl_setup (ctx); return; } /* Check the file type. Note that this is just a quick check to avoid * creating QmiDevices pointing to a location already known not to be a QMI * device. */ g_file_query_info_async (ctx->self->priv->file, G_FILE_ATTRIBUTE_STANDARD_TYPE, G_FILE_QUERY_INFO_NONE, G_PRIORITY_DEFAULT, ctx->cancellable, (GAsyncReadyCallback)query_info_async_ready, ctx); } /*****************************************************************************/ static void set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { QmiDevice *self = QMI_DEVICE (object); switch (prop_id) { case PROP_FILE: g_assert (self->priv->file == NULL); self->priv->file = g_value_dup_object (value); if (self->priv->file) { self->priv->path = g_file_get_path (self->priv->file); self->priv->path_display = g_filename_display_name (self->priv->path); } break; case PROP_NO_FILE_CHECK: self->priv->no_file_check = g_value_get_boolean (value); break; case PROP_PROXY_PATH: g_free (self->priv->proxy_path); self->priv->proxy_path = g_value_dup_string (value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { QmiDevice *self = QMI_DEVICE (object); switch (prop_id) { case PROP_FILE: g_value_set_object (value, self->priv->file); break; case PROP_WWAN_IFACE: reload_wwan_iface_name (self); g_value_set_string (value, self->priv->wwan_iface); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void qmi_device_init (QmiDevice *self) { self->priv = G_TYPE_INSTANCE_GET_PRIVATE ((self), QMI_TYPE_DEVICE, QmiDevicePrivate); self->priv->registered_clients = g_hash_table_new_full (g_direct_hash, g_direct_equal, NULL, g_object_unref); self->priv->proxy_path = g_strdup (QMI_PROXY_SOCKET_PATH); } static gboolean foreach_warning (gpointer key, QmiClient *client, QmiDevice *self) { g_warning ("[%s] QMI client for service '%s' with CID '%u' wasn't released", self->priv->path_display, qmi_service_get_string (qmi_client_get_service (client)), qmi_client_get_cid (client)); return TRUE; } static void dispose (GObject *object) { QmiDevice *self = QMI_DEVICE (object); g_clear_object (&self->priv->file); /* unregister our CTL client */ if (self->priv->client_ctl) unregister_client (self, QMI_CLIENT (self->priv->client_ctl)); /* If clients were left unreleased, we'll just warn about it. * There is no point in trying to request CID releases, as the device * itself is being disposed. */ g_hash_table_foreach_remove (self->priv->registered_clients, (GHRFunc)foreach_warning, self); if (self->priv->sync_indication_id && self->priv->client_ctl) { g_signal_handler_disconnect (self->priv->client_ctl, self->priv->sync_indication_id); self->priv->sync_indication_id = 0; } g_clear_object (&self->priv->client_ctl); G_OBJECT_CLASS (qmi_device_parent_class)->dispose (object); } static void finalize (GObject *object) { QmiDevice *self = QMI_DEVICE (object); /* Transactions keep refs to the device, so it's actually * impossible to have any content in the HT */ if (self->priv->transactions) { g_assert (g_hash_table_size (self->priv->transactions) == 0); g_hash_table_unref (self->priv->transactions); } g_hash_table_unref (self->priv->registered_clients); if (self->priv->supported_services) g_array_unref (self->priv->supported_services); g_free (self->priv->path); g_free (self->priv->path_display); g_free (self->priv->proxy_path); g_free (self->priv->wwan_iface); if (self->priv->input_source) { g_source_destroy (self->priv->input_source); g_source_unref (self->priv->input_source); } if (self->priv->buffer) g_byte_array_unref (self->priv->buffer); if (self->priv->istream) g_object_unref (self->priv->istream); if (self->priv->ostream) g_object_unref (self->priv->ostream); if (self->priv->socket_connection) g_object_unref (self->priv->socket_connection); if (self->priv->socket_client) g_object_unref (self->priv->socket_client); G_OBJECT_CLASS (qmi_device_parent_class)->finalize (object); } static void async_initable_iface_init (GAsyncInitableIface *iface) { iface->init_async = initable_init_async; iface->init_finish = initable_init_finish; } static void qmi_device_class_init (QmiDeviceClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); g_type_class_add_private (object_class, sizeof (QmiDevicePrivate)); object_class->get_property = get_property; object_class->set_property = set_property; object_class->finalize = finalize; object_class->dispose = dispose; properties[PROP_FILE] = g_param_spec_object (QMI_DEVICE_FILE, "Device file", "File to the underlying QMI device", G_TYPE_FILE, G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY); g_object_class_install_property (object_class, PROP_FILE, properties[PROP_FILE]); properties[PROP_NO_FILE_CHECK] = g_param_spec_boolean (QMI_DEVICE_NO_FILE_CHECK, "No file check", "Don't check for file existence when creating the Qmi device.", FALSE, G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY); g_object_class_install_property (object_class, PROP_NO_FILE_CHECK, properties[PROP_NO_FILE_CHECK]); properties[PROP_PROXY_PATH] = g_param_spec_string (QMI_DEVICE_PROXY_PATH, "Proxy path", "Path of the abstract socket where the proxy is available.", QMI_PROXY_SOCKET_PATH, G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY); g_object_class_install_property (object_class, PROP_PROXY_PATH, properties[PROP_PROXY_PATH]); properties[PROP_WWAN_IFACE] = g_param_spec_string (QMI_DEVICE_WWAN_IFACE, "WWAN iface", "Name of the WWAN network interface associated with the control port.", NULL, G_PARAM_READABLE); g_object_class_install_property (object_class, PROP_WWAN_IFACE, properties[PROP_WWAN_IFACE]); /** * QmiClientDms::event-report: * @object: A #QmiClientDms. * @output: A #QmiIndicationDmsEventReportOutput. * * The ::event-report signal gets emitted when a 'Event Report' indication is received. */ signals[SIGNAL_INDICATION] = g_signal_new (QMI_DEVICE_SIGNAL_INDICATION, G_OBJECT_CLASS_TYPE (G_OBJECT_CLASS (klass)), G_SIGNAL_RUN_LAST, 0, NULL, NULL, NULL, G_TYPE_NONE, 1, G_TYPE_BYTE_ARRAY); }