/* -*- 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 Aleksander Morgado */ #include #include #include #include #include "mm-mbim-port.h" #include "mm-log.h" G_DEFINE_TYPE (MMMbimPort, mm_mbim_port, MM_TYPE_PORT) struct _MMMbimPortPrivate { gboolean in_progress; MbimDevice *mbim_device; }; /*****************************************************************************/ typedef struct { MMMbimPort *self; GSimpleAsyncResult *result; GCancellable *cancellable; } PortContext; static void port_context_complete_and_free (PortContext *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 (PortContext, ctx); } static PortContext * port_context_new (MMMbimPort *self, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data) { PortContext *ctx; ctx = g_slice_new0 (PortContext); ctx->self = g_object_ref (self); ctx->result = g_simple_async_result_new (G_OBJECT (self), callback, user_data, port_context_new); ctx->cancellable = cancellable ? g_object_ref (cancellable) : NULL; return ctx; } /*****************************************************************************/ gboolean mm_mbim_port_open_finish (MMMbimPort *self, GAsyncResult *res, GError **error) { return !g_simple_async_result_propagate_error (G_SIMPLE_ASYNC_RESULT (res), error); } static void mbim_device_open_ready (MbimDevice *mbim_device, GAsyncResult *res, PortContext *ctx) { GError *error = NULL; /* Reset the progress flag */ ctx->self->priv->in_progress = FALSE; if (!mbim_device_open_finish (mbim_device, res, &error)) { g_clear_object (&ctx->self->priv->mbim_device); g_simple_async_result_take_error (ctx->result, error); } else g_simple_async_result_set_op_res_gboolean (ctx->result, TRUE); port_context_complete_and_free (ctx); } static void mbim_device_new_ready (GObject *unused, GAsyncResult *res, PortContext *ctx) { GError *error = NULL; ctx->self->priv->mbim_device = mbim_device_new_finish (res, &error); if (!ctx->self->priv->mbim_device) { g_simple_async_result_take_error (ctx->result, error); port_context_complete_and_free (ctx); return; } /* Now open the MBIM device */ mbim_device_open (ctx->self->priv->mbim_device, 10, ctx->cancellable, (GAsyncReadyCallback)mbim_device_open_ready, ctx); } void mm_mbim_port_open (MMMbimPort *self, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data) { GFile *file; gchar *fullpath; PortContext *ctx; g_return_if_fail (MM_IS_MBIM_PORT (self)); ctx = port_context_new (self, cancellable, callback, user_data); if (self->priv->in_progress) { g_simple_async_result_set_error (ctx->result, MM_CORE_ERROR, MM_CORE_ERROR_IN_PROGRESS, "MBIM device open/close operation in progress"); port_context_complete_and_free (ctx); return; } if (self->priv->mbim_device) { g_simple_async_result_set_op_res_gboolean (ctx->result, TRUE); port_context_complete_and_free (ctx); return; } fullpath = g_strdup_printf ("/dev/%s", mm_port_get_device (MM_PORT (self))); file = g_file_new_for_path (fullpath); self->priv->in_progress = TRUE; mbim_device_new (file, ctx->cancellable, (GAsyncReadyCallback)mbim_device_new_ready, ctx); g_free (fullpath); g_object_unref (file); } /*****************************************************************************/ gboolean mm_mbim_port_is_open (MMMbimPort *self) { g_return_val_if_fail (MM_IS_MBIM_PORT (self), FALSE); return !!self->priv->mbim_device; } /*****************************************************************************/ gboolean mm_mbim_port_close_finish (MMMbimPort *self, GAsyncResult *res, GError **error) { return !g_simple_async_result_propagate_error (G_SIMPLE_ASYNC_RESULT (res), error); } static void mbim_device_close_ready (MbimDevice *device, GAsyncResult *res, PortContext *ctx) { GError *error = NULL; if (!mbim_device_close_finish (device, res, &error)) g_simple_async_result_take_error (ctx->result, error); else g_simple_async_result_set_op_res_gboolean (ctx->result, TRUE); ctx->self->priv->in_progress = FALSE; g_clear_object (&ctx->self->priv->mbim_device); port_context_complete_and_free (ctx); } void mm_mbim_port_close (MMMbimPort *self, GAsyncReadyCallback callback, gpointer user_data) { PortContext *ctx; g_return_if_fail (MM_IS_MBIM_PORT (self)); ctx = port_context_new (self, NULL, callback, user_data); if (self->priv->in_progress) { g_simple_async_result_set_error (ctx->result, MM_CORE_ERROR, MM_CORE_ERROR_IN_PROGRESS, "MBIM device open/close operation in progress"); port_context_complete_and_free (ctx); return; } if (!self->priv->mbim_device) { g_simple_async_result_set_op_res_gboolean (ctx->result, TRUE); port_context_complete_and_free (ctx); return; } self->priv->in_progress = TRUE; mbim_device_close (self->priv->mbim_device, 5, NULL, (GAsyncReadyCallback)mbim_device_close_ready, ctx); g_clear_object (&self->priv->mbim_device); } /*****************************************************************************/ MbimDevice * mm_mbim_port_peek_device (MMMbimPort *self) { g_return_val_if_fail (MM_IS_MBIM_PORT (self), NULL); return self->priv->mbim_device; } /*****************************************************************************/ MMMbimPort * mm_mbim_port_new (const gchar *name) { return MM_MBIM_PORT (g_object_new (MM_TYPE_MBIM_PORT, MM_PORT_DEVICE, name, MM_PORT_SUBSYS, MM_PORT_SUBSYS_USB, MM_PORT_TYPE, MM_PORT_TYPE_MBIM, NULL)); } static void mm_mbim_port_init (MMMbimPort *self) { self->priv = G_TYPE_INSTANCE_GET_PRIVATE (self, MM_TYPE_MBIM_PORT, MMMbimPortPrivate); } static void dispose (GObject *object) { MMMbimPort *self = MM_MBIM_PORT (object); /* Clear device object */ g_clear_object (&self->priv->mbim_device); G_OBJECT_CLASS (mm_mbim_port_parent_class)->dispose (object); } static void mm_mbim_port_class_init (MMMbimPortClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); g_type_class_add_private (object_class, sizeof (MMMbimPortPrivate)); /* Virtual methods */ object_class->dispose = dispose; }