diff options
author | Aleksander Morgado <aleksander@aleksander.es> | 2019-07-02 13:43:00 +0200 |
---|---|---|
committer | Aleksander Morgado <aleksander@aleksander.es> | 2019-07-11 23:21:00 +0200 |
commit | 24c0c7aa6b027911a30c9b1cd9784ad221abac5d (patch) | |
tree | b0144da5276f22a7d6a46591b331a1118caf69fc | |
parent | b7fa4e3661075ad8c9585e2b17c6738f63e83e60 (diff) |
huawei: audio channel setup at modem-level
The audio channel setup is shared between all ongoing calls, so it is
a modem-level feature, not specific to a single call.
-rw-r--r-- | plugins/Makefile.am | 2 | ||||
-rw-r--r-- | plugins/huawei/mm-broadband-modem-huawei.c | 107 | ||||
-rw-r--r-- | plugins/huawei/mm-call-huawei.c | 261 | ||||
-rw-r--r-- | plugins/huawei/mm-call-huawei.h | 55 |
4 files changed, 93 insertions, 332 deletions
diff --git a/plugins/Makefile.am b/plugins/Makefile.am index 57cd6bd6..9f8a6612 100644 --- a/plugins/Makefile.am +++ b/plugins/Makefile.am @@ -355,8 +355,6 @@ libmm_plugin_huawei_la_SOURCES = \ huawei/mm-plugin-huawei.h \ huawei/mm-sim-huawei.c \ huawei/mm-sim-huawei.h \ - huawei/mm-call-huawei.c \ - huawei/mm-call-huawei.h \ huawei/mm-broadband-modem-huawei.c \ huawei/mm-broadband-modem-huawei.h \ huawei/mm-broadband-bearer-huawei.c \ diff --git a/plugins/huawei/mm-broadband-modem-huawei.c b/plugins/huawei/mm-broadband-modem-huawei.c index 3d4d72f5..41544391 100644 --- a/plugins/huawei/mm-broadband-modem-huawei.c +++ b/plugins/huawei/mm-broadband-modem-huawei.c @@ -14,8 +14,8 @@ * Copyright (C) 2009 - 2012 Red Hat, Inc. * Copyright (C) 2011 - 2012 Google Inc. * Copyright (C) 2012 Huawei Technologies Co., Ltd - * Copyright (C) 2012 - 2013 Aleksander Morgado <aleksander@gnu.org> * Copyright (C) 2015 Marco Bascetta <marco.bascetta@sadel.it> + * Copyright (C) 2012 - 2019 Aleksander Morgado <aleksander@aleksander.es> */ #include <config.h> @@ -49,7 +49,6 @@ #include "mm-broadband-bearer.h" #include "mm-bearer-list.h" #include "mm-sim-huawei.h" -#include "mm-call-huawei.h" static void iface_modem_init (MMIfaceModem *iface); static void iface_modem_3gpp_init (MMIfaceModem3gpp *iface); @@ -2940,6 +2939,89 @@ modem_voice_check_support (MMIfaceModemVoice *self, } /*****************************************************************************/ +/* In-call audio channel setup/cleanup */ + +static gboolean +modem_voice_setup_in_call_audio_channel_finish (MMIfaceModemVoice *_self, + GAsyncResult *res, + MMPort **audio_port, + MMCallAudioFormat **audio_format, + GError **error) +{ + MMBroadbandModemHuawei *self = MM_BROADBAND_MODEM_HUAWEI (_self); + + if (!g_task_propagate_boolean (G_TASK (res), error)) + return FALSE; + + if (self->priv->cvoice_support == FEATURE_SUPPORTED) { + /* Setup audio format */ + if (audio_format) { + gchar *resolution_str; + + resolution_str = g_strdup_printf ("s%ule", self->priv->audio_bits); + *audio_format = mm_call_audio_format_new (); + mm_call_audio_format_set_encoding (*audio_format, "pcm"); + mm_call_audio_format_set_resolution (*audio_format, resolution_str); + mm_call_audio_format_set_rate (*audio_format, self->priv->audio_hz); + g_free (resolution_str); + } + + /* The QCDM port, if present, switches from QCDM to voice while + * a voice call is active. */ + if (audio_port) + *audio_port = MM_PORT (mm_base_modem_get_port_qcdm (MM_BASE_MODEM (self))); + } else { + if (audio_format) + *audio_format = NULL; + if (audio_port) + *audio_port = NULL; + } + + return TRUE; +} + +static void +ddsetex_ready (MMBaseModem *self, + GAsyncResult *res, + GTask *task) +{ + GError *error = NULL; + + if (!mm_base_modem_at_command_finish (self, res, &error)) + g_task_return_error (task, error); + else + g_task_return_boolean (task, TRUE); + g_object_unref (task); +} + +static void +modem_voice_setup_in_call_audio_channel (MMIfaceModemVoice *_self, + GAsyncReadyCallback callback, + gpointer user_data) +{ + MMBroadbandModemHuawei *self = MM_BROADBAND_MODEM_HUAWEI (_self); + GTask *task; + + task = g_task_new (self, NULL, callback, user_data); + + /* If there is no CVOICE support, no custom audio setup required + * (i.e. audio path is externally managed) */ + if (self->priv->cvoice_support != FEATURE_SUPPORTED) { + g_task_return_boolean (task, TRUE); + g_object_unref (task); + return; + } + + /* Enable audio streaming on the audio port */ + mm_base_modem_at_command (MM_BASE_MODEM (self), + "^DDSETEX=2", + 5, + FALSE, + (GAsyncReadyCallback)ddsetex_ready, + task); +} + +/*****************************************************************************/ /* Common setup/cleanup voice unsolicited events */ typedef enum { @@ -3260,21 +3342,16 @@ modem_voice_disable_unsolicited_events (MMIfaceModemVoice *self, /* Create call (Voice interface) */ static MMBaseCall * -create_call (MMIfaceModemVoice *_self, +create_call (MMIfaceModemVoice *self, MMCallDirection direction, const gchar *number) { - MMBroadbandModemHuawei *self = MM_BROADBAND_MODEM_HUAWEI (_self); - - /* If CVOICE is supported we must have audio settings */ - g_assert (self->priv->cvoice_support == FEATURE_NOT_SUPPORTED || - (self->priv->cvoice_support == FEATURE_SUPPORTED && self->priv->audio_hz && self->priv->audio_bits)); - - return mm_call_huawei_new (MM_BASE_MODEM (self), - direction, - number, - self->priv->audio_hz, - self->priv->audio_bits); + return mm_base_call_new (MM_BASE_MODEM (self), + direction, + number, + TRUE, /* skip_incoming_timeout */ + TRUE, /* supports_dialing_to_ringing */ + TRUE); /* supports_ringing_to_active) */ } /*****************************************************************************/ @@ -4528,6 +4605,8 @@ iface_modem_voice_init (MMIfaceModemVoice *iface) iface->enable_unsolicited_events_finish = modem_voice_enable_unsolicited_events_finish; iface->disable_unsolicited_events = modem_voice_disable_unsolicited_events; iface->disable_unsolicited_events_finish = modem_voice_disable_unsolicited_events_finish; + iface->setup_in_call_audio_channel = modem_voice_setup_in_call_audio_channel; + iface->setup_in_call_audio_channel_finish = modem_voice_setup_in_call_audio_channel_finish; iface->create_call = create_call; } diff --git a/plugins/huawei/mm-call-huawei.c b/plugins/huawei/mm-call-huawei.c deleted file mode 100644 index c02c247a..00000000 --- a/plugins/huawei/mm-call-huawei.c +++ /dev/null @@ -1,261 +0,0 @@ -/* -*- 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) 2015 Riccardo Vangelisti <riccardo.vangelisti@sadel.it> - */ - -#include <config.h> -#include <stdio.h> -#include <stdlib.h> -#include <unistd.h> -#include <string.h> -#include <ctype.h> - -#include <ModemManager.h> -#define _LIBMM_INSIDE_MM -#include <libmm-glib.h> - -#include "mm-log.h" -#include "mm-base-modem-at.h" -#include "mm-broadband-modem-huawei.h" -#include "mm-call-huawei.h" - -G_DEFINE_TYPE (MMCallHuawei, mm_call_huawei, MM_TYPE_BASE_CALL) - -enum { - PROP_0, - PROP_AUDIO_HZ, - PROP_AUDIO_BITS, - PROP_LAST -}; - -static GParamSpec *properties[PROP_LAST]; - -struct _MMCallHuaweiPrivate { - guint audio_hz; - guint audio_bits; -}; - -/*****************************************************************************/ -/* Audio channel setup */ - -typedef struct { - MMBaseModem *modem; - MMPort *audio_port; - MMCallAudioFormat *audio_format; -} SetupAudioChannelContext; - -static void -setup_audio_channel_context_free (SetupAudioChannelContext *ctx) -{ - g_clear_object (&ctx->audio_port); - g_clear_object (&ctx->audio_format); - g_clear_object (&ctx->modem); - g_slice_free (SetupAudioChannelContext, ctx); -} - -static gboolean -setup_audio_channel_finish (MMBaseCall *self, - GAsyncResult *res, - MMPort **audio_port, - MMCallAudioFormat **audio_format, - GError **error) -{ - SetupAudioChannelContext *ctx; - - if (!g_task_propagate_boolean (G_TASK (res), error)) - return FALSE; - - ctx = g_task_get_task_data (G_TASK (res)); - - if (audio_port && ctx->audio_port) - *audio_port = g_object_ref (ctx->audio_port); - if (audio_format && ctx->audio_format) - *audio_format = g_object_ref (ctx->audio_format); - - return TRUE; -} - -static void -ddsetex_ready (MMBaseModem *modem, - GAsyncResult *res, - GTask *task) -{ - MMCallHuawei *self; - SetupAudioChannelContext *ctx; - GError *error = NULL; - const gchar *response = NULL; - gchar *resolution_str; - - response = mm_base_modem_at_command_finish (modem, res, &error); - if (!response) { - mm_dbg ("Error enabling audio streaming: '%s'", error->message); - g_task_return_error (task, error); - g_object_unref (task); - return; - } - - self = g_task_get_source_object (task); - ctx = g_task_get_task_data (task); - - /* Setup audio format */ - g_assert (self->priv->audio_hz && self->priv->audio_bits); - resolution_str = g_strdup_printf ("s%ule", self->priv->audio_bits); - ctx->audio_format = mm_call_audio_format_new (); - mm_call_audio_format_set_encoding (ctx->audio_format, "pcm"); - mm_call_audio_format_set_resolution (ctx->audio_format, resolution_str); - mm_call_audio_format_set_rate (ctx->audio_format, self->priv->audio_hz); - - /* The QCDM port, if present, switches from QCDM to voice while - * a voice call is active. */ - ctx->audio_port = MM_PORT (mm_base_modem_get_port_qcdm (modem)); - - g_task_return_boolean (task, TRUE); - g_object_unref (task); -} - -static void -setup_audio_channel (MMBaseCall *_self, - GAsyncReadyCallback callback, - gpointer user_data) -{ - SetupAudioChannelContext *ctx; - MMCallHuawei *self; - GTask *task; - MMBaseModem *modem = NULL; - - self = MM_CALL_HUAWEI (_self); - - task = g_task_new (self, NULL, callback, user_data); - - /* If there is no CVOICE support, no custom audio setup required - * (i.e. audio path is externally managed) */ - if (!self->priv->audio_hz && !self->priv->audio_bits) { - g_task_return_boolean (task, TRUE); - g_object_unref (task); - return; - } - - ctx = g_slice_new0 (SetupAudioChannelContext); - g_object_get (self, - MM_BASE_CALL_MODEM, &ctx->modem, - NULL); - g_task_set_task_data (task, ctx, (GDestroyNotify) setup_audio_channel_context_free); - - /* Enable audio streaming on the audio port */ - mm_base_modem_at_command (modem, - "AT^DDSETEX=2", - 5, - FALSE, - (GAsyncReadyCallback)ddsetex_ready, - task); -} - -/*****************************************************************************/ - -MMBaseCall * -mm_call_huawei_new (MMBaseModem *modem, - MMCallDirection direction, - const gchar *number, - guint audio_hz, - guint audio_bits) -{ - return MM_BASE_CALL (g_object_new (MM_TYPE_CALL_HUAWEI, - MM_BASE_CALL_MODEM, modem, - "direction", direction, - "number", number, - MM_CALL_HUAWEI_AUDIO_HZ, audio_hz, - MM_CALL_HUAWEI_AUDIO_BITS, audio_bits, - MM_BASE_CALL_SUPPORTS_DIALING_TO_RINGING, TRUE, - MM_BASE_CALL_SUPPORTS_RINGING_TO_ACTIVE, TRUE, - NULL)); -} - -static void -mm_call_huawei_init (MMCallHuawei *self) -{ - /* Initialize private data */ - self->priv = G_TYPE_INSTANCE_GET_PRIVATE (self, MM_TYPE_CALL_HUAWEI, MMCallHuaweiPrivate); -} - -static void -set_property (GObject *object, - guint prop_id, - const GValue *value, - GParamSpec *pspec) -{ - MMCallHuawei *self = MM_CALL_HUAWEI (object); - - switch (prop_id) { - case PROP_AUDIO_HZ: - self->priv->audio_hz = g_value_get_uint (value); - break; - case PROP_AUDIO_BITS: - self->priv->audio_bits = g_value_get_uint (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) -{ - MMCallHuawei *self = MM_CALL_HUAWEI (object); - - switch (prop_id) { - case PROP_AUDIO_HZ: - g_value_set_uint (value, self->priv->audio_hz); - break; - case PROP_AUDIO_BITS: - g_value_set_uint (value, self->priv->audio_bits); - break; - default: - G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); - break; - } -} - -static void -mm_call_huawei_class_init (MMCallHuaweiClass *klass) -{ - GObjectClass *object_class = G_OBJECT_CLASS (klass); - MMBaseCallClass *base_call_class = MM_BASE_CALL_CLASS (klass); - - g_type_class_add_private (object_class, sizeof (MMCallHuaweiPrivate)); - - object_class->get_property = get_property; - object_class->set_property = set_property; - - base_call_class->setup_audio_channel = setup_audio_channel; - base_call_class->setup_audio_channel_finish = setup_audio_channel_finish; - - properties[PROP_AUDIO_HZ] = - g_param_spec_uint (MM_CALL_HUAWEI_AUDIO_HZ, - "Audio Hz", - "Voice call audio hz if call audio is routed via the host", - 0, 24000, 0, - G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY); - g_object_class_install_property (object_class, PROP_AUDIO_HZ, properties[PROP_AUDIO_HZ]); - - properties[PROP_AUDIO_BITS] = - g_param_spec_uint (MM_CALL_HUAWEI_AUDIO_BITS, - "Audio Bits", - "Voice call audio bits if call audio is routed via the host", - 0, 24, 0, - G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY); - g_object_class_install_property (object_class, PROP_AUDIO_BITS, properties[PROP_AUDIO_BITS]); -} diff --git a/plugins/huawei/mm-call-huawei.h b/plugins/huawei/mm-call-huawei.h deleted file mode 100644 index 1b70e81f..00000000 --- a/plugins/huawei/mm-call-huawei.h +++ /dev/null @@ -1,55 +0,0 @@ -/* -*- 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) 2015 Riccardo Vangelisti <riccardo.vangelisti@sadel.it> - */ - -#ifndef MM_CALL_HUAWEI_H -#define MM_CALL_HUAWEI_H - -#include <glib.h> -#include <glib-object.h> - -#include "mm-base-call.h" - -#define MM_TYPE_CALL_HUAWEI (mm_call_huawei_get_type ()) -#define MM_CALL_HUAWEI(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), MM_TYPE_CALL_HUAWEI, MMCallHuawei)) -#define MM_CALL_HUAWEI_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), MM_TYPE_CALL_HUAWEI, MMCallHuaweiClass)) -#define MM_IS_CALL_HUAWEI(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), MM_TYPE_CALL_HUAWEI)) -#define MM_IS_CALL_HUAWEI_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), MM_TYPE_CALL_HUAWEI)) -#define MM_CALL_HUAWEI_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), MM_TYPE_CALL_HUAWEI, MMCallHuaweiClass)) - -#define MM_CALL_HUAWEI_AUDIO_HZ "call-huawei-audio-hz" -#define MM_CALL_HUAWEI_AUDIO_BITS "call-huawei-audio-bits" - -typedef struct _MMCallHuawei MMCallHuawei; -typedef struct _MMCallHuaweiClass MMCallHuaweiClass; -typedef struct _MMCallHuaweiPrivate MMCallHuaweiPrivate; - -struct _MMCallHuawei { - MMBaseCall parent; - MMCallHuaweiPrivate *priv; -}; - -struct _MMCallHuaweiClass { - MMBaseCallClass parent; -}; - -GType mm_call_huawei_get_type (void); - -MMBaseCall *mm_call_huawei_new (MMBaseModem *modem, - MMCallDirection direction, - const gchar *number, - guint audio_hz, - guint audio_bits); - -#endif /* MM_CALL_HUAWEI_H */ |