2021-02-17 22:48:33 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2021 Purism SPC
|
|
|
|
*
|
|
|
|
* This file is part of Calls.
|
|
|
|
*
|
|
|
|
* Calls 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 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* Calls 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.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with Calls. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* Author: Evangelos Ribeiro Tzaras <evangelos.tzaras@puri.sm>
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2021-04-09 15:41:14 +00:00
|
|
|
#define G_LOG_DOMAIN "CallsSipMediaManager"
|
2021-02-17 22:48:33 +00:00
|
|
|
|
2021-11-23 14:04:18 +00:00
|
|
|
#include "calls-settings.h"
|
2021-02-17 22:48:33 +00:00
|
|
|
#include "calls-sip-media-manager.h"
|
2021-04-10 22:25:13 +00:00
|
|
|
#include "gst-rfc3551.h"
|
2021-02-17 22:48:33 +00:00
|
|
|
|
|
|
|
#include <gst/gst.h>
|
|
|
|
|
2021-04-10 22:25:13 +00:00
|
|
|
/**
|
|
|
|
* SECTION:sip-media-manager
|
|
|
|
* @short_description: The media manager singleton
|
|
|
|
* @Title: CallsSipMediaManager
|
|
|
|
*
|
|
|
|
* #CallsSipMediaManager is mainly responsible for generating appropriate
|
|
|
|
* SDP messages for the set of supported codecs. In the future it
|
|
|
|
* shall also manage the #CallsSipMediaPipeline objects that are in use.
|
|
|
|
*/
|
|
|
|
|
2021-09-24 07:33:38 +00:00
|
|
|
enum {
|
|
|
|
PROP_0,
|
|
|
|
PROP_SESSION_IP,
|
|
|
|
PROP_LAST_PROP
|
|
|
|
};
|
|
|
|
static GParamSpec *props[PROP_LAST_PROP];
|
|
|
|
|
2021-02-17 22:48:33 +00:00
|
|
|
typedef struct _CallsSipMediaManager
|
|
|
|
{
|
|
|
|
GObject parent;
|
2021-03-31 07:46:44 +00:00
|
|
|
|
2021-11-23 14:04:18 +00:00
|
|
|
char *session_ip;
|
|
|
|
CallsSettings *settings;
|
|
|
|
GList *preferred_codecs;
|
2021-02-17 22:48:33 +00:00
|
|
|
} CallsSipMediaManager;
|
|
|
|
|
|
|
|
G_DEFINE_TYPE (CallsSipMediaManager, calls_sip_media_manager, G_TYPE_OBJECT);
|
|
|
|
|
|
|
|
|
2021-11-23 14:04:18 +00:00
|
|
|
static void
|
|
|
|
on_notify_preferred_audio_codecs (CallsSipMediaManager *self)
|
|
|
|
{
|
|
|
|
GList *supported_codecs;
|
|
|
|
g_auto (GStrv) settings_codec_preference = NULL;
|
|
|
|
|
|
|
|
g_assert (CALLS_IS_SIP_MEDIA_MANAGER (self));
|
|
|
|
|
|
|
|
g_clear_list (&self->preferred_codecs, NULL);
|
|
|
|
supported_codecs = media_codecs_get_candidates ();
|
|
|
|
|
|
|
|
if (!supported_codecs) {
|
|
|
|
g_warning ("There aren't any supported codecs installed on your system");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
settings_codec_preference = calls_settings_get_preferred_audio_codecs (self->settings);
|
|
|
|
|
|
|
|
if (!settings_codec_preference) {
|
|
|
|
g_debug ("No audio codec preference set. Using all supported codecs");
|
|
|
|
self->preferred_codecs = supported_codecs;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (guint i = 0; settings_codec_preference[i] != NULL; i++) {
|
|
|
|
MediaCodecInfo *codec = media_codec_by_name (settings_codec_preference[i]);
|
|
|
|
|
|
|
|
if (!codec) {
|
|
|
|
g_debug ("Did not find audio codec %s", settings_codec_preference[i]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (media_codec_available_in_gst (codec))
|
|
|
|
self->preferred_codecs = g_list_append (self->preferred_codecs, codec);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!self->preferred_codecs) {
|
|
|
|
g_warning ("Cannot satisfy audio codec preference, "
|
|
|
|
"falling back to all supported codecs");
|
|
|
|
self->preferred_codecs = supported_codecs;
|
|
|
|
} else {
|
|
|
|
g_list_free (supported_codecs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-24 07:33:38 +00:00
|
|
|
static void
|
|
|
|
calls_sip_media_manager_set_property (GObject *object,
|
|
|
|
guint property_id,
|
|
|
|
const GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
CallsSipMediaManager *self = CALLS_SIP_MEDIA_MANAGER (object);
|
|
|
|
|
|
|
|
switch (property_id) {
|
|
|
|
case PROP_SESSION_IP:
|
|
|
|
calls_sip_media_manager_set_session_ip (self, g_value_get_string (value));
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-02-17 22:48:33 +00:00
|
|
|
static void
|
|
|
|
calls_sip_media_manager_finalize (GObject *object)
|
|
|
|
{
|
2021-09-24 07:33:38 +00:00
|
|
|
CallsSipMediaManager *self = CALLS_SIP_MEDIA_MANAGER (object);
|
2021-02-17 22:48:33 +00:00
|
|
|
gst_deinit ();
|
|
|
|
|
2021-11-23 14:04:18 +00:00
|
|
|
g_list_free (self->preferred_codecs);
|
2021-09-24 07:33:38 +00:00
|
|
|
g_free (self->session_ip);
|
2021-11-23 14:04:18 +00:00
|
|
|
g_object_unref (self->settings);
|
2021-03-31 07:46:44 +00:00
|
|
|
|
2021-02-17 22:48:33 +00:00
|
|
|
G_OBJECT_CLASS (calls_sip_media_manager_parent_class)->finalize (object);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
calls_sip_media_manager_class_init (CallsSipMediaManagerClass *klass)
|
|
|
|
{
|
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
|
2021-09-24 07:33:38 +00:00
|
|
|
object_class->set_property = calls_sip_media_manager_set_property;
|
2021-02-17 22:48:33 +00:00
|
|
|
object_class->finalize = calls_sip_media_manager_finalize;
|
2021-09-24 07:33:38 +00:00
|
|
|
|
|
|
|
props[PROP_SESSION_IP] =
|
|
|
|
g_param_spec_string ("session-ip",
|
|
|
|
"Session IP",
|
|
|
|
"The public IP used as the session line in SDP",
|
|
|
|
NULL,
|
|
|
|
G_PARAM_WRITABLE | G_PARAM_STATIC_STRINGS);
|
|
|
|
|
|
|
|
g_object_class_install_properties (object_class, PROP_LAST_PROP, props);
|
2021-02-17 22:48:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
calls_sip_media_manager_init (CallsSipMediaManager *self)
|
|
|
|
{
|
|
|
|
gst_init (NULL, NULL);
|
2021-03-31 07:46:44 +00:00
|
|
|
|
2021-11-23 14:04:18 +00:00
|
|
|
self->settings = calls_settings_new ();
|
|
|
|
g_signal_connect_swapped (self->settings,
|
|
|
|
"notify::preferred-audio-codecs",
|
|
|
|
G_CALLBACK (on_notify_preferred_audio_codecs),
|
|
|
|
self);
|
|
|
|
on_notify_preferred_audio_codecs (self);
|
2021-02-17 22:48:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Public functions */
|
|
|
|
|
|
|
|
CallsSipMediaManager *
|
2021-06-02 18:18:50 +00:00
|
|
|
calls_sip_media_manager_default (void)
|
2021-02-17 22:48:33 +00:00
|
|
|
{
|
|
|
|
static CallsSipMediaManager *instance = NULL;
|
|
|
|
|
|
|
|
if (instance == NULL) {
|
|
|
|
g_debug ("Creating CallsSipMediaManager");
|
|
|
|
instance = g_object_new (CALLS_TYPE_SIP_MEDIA_MANAGER, NULL);
|
|
|
|
g_object_add_weak_pointer (G_OBJECT (instance), (gpointer *) &instance);
|
|
|
|
}
|
|
|
|
return instance;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-04-08 14:43:13 +00:00
|
|
|
/* calls_sip_media_manager_get_capabilities:
|
2021-02-17 22:48:33 +00:00
|
|
|
*
|
2021-04-08 14:43:13 +00:00
|
|
|
* @self: A #CallsSipMediaManager
|
2021-02-17 22:48:33 +00:00
|
|
|
* @port: Should eventually come from the ICE stack
|
|
|
|
* @use_srtp: Whether to use srtp (not really handled)
|
2021-04-08 14:43:13 +00:00
|
|
|
* @supported_codecs: A #GList of #MediaCodecInfo
|
2021-02-17 22:48:33 +00:00
|
|
|
*
|
2021-04-26 08:29:24 +00:00
|
|
|
* Returns: (transfer full): string describing capabilities
|
2021-02-17 22:48:33 +00:00
|
|
|
* to be used in the session description (SDP)
|
|
|
|
*/
|
|
|
|
char *
|
2021-04-08 14:43:13 +00:00
|
|
|
calls_sip_media_manager_get_capabilities (CallsSipMediaManager *self,
|
|
|
|
guint port,
|
|
|
|
gboolean use_srtp,
|
|
|
|
GList *supported_codecs)
|
2021-02-17 22:48:33 +00:00
|
|
|
{
|
|
|
|
char *payload_type = use_srtp ? "SAVP" : "AVP";
|
2021-03-31 07:46:44 +00:00
|
|
|
g_autoptr (GString) media_line = NULL;
|
|
|
|
g_autoptr (GString) attribute_lines = NULL;
|
|
|
|
GList *node;
|
2021-02-17 22:48:33 +00:00
|
|
|
|
|
|
|
g_return_val_if_fail (CALLS_IS_SIP_MEDIA_MANAGER (self), NULL);
|
|
|
|
|
2021-03-31 07:46:44 +00:00
|
|
|
media_line = g_string_new (NULL);
|
|
|
|
attribute_lines = g_string_new (NULL);
|
2021-02-17 22:48:33 +00:00
|
|
|
|
2021-04-08 14:43:13 +00:00
|
|
|
if (supported_codecs == NULL) {
|
2021-03-31 07:46:44 +00:00
|
|
|
g_warning ("No supported codecs found. Can't build meaningful SDP message");
|
2021-04-09 15:41:32 +00:00
|
|
|
g_string_append_printf (media_line, "m=audio 0 RTP/AVP");
|
2021-03-31 07:46:44 +00:00
|
|
|
goto done;
|
|
|
|
}
|
2021-02-17 22:48:33 +00:00
|
|
|
|
2021-03-31 07:46:44 +00:00
|
|
|
/* media lines look f.e like "audio 31337 RTP/AVP 9 8 0" */
|
|
|
|
g_string_append_printf (media_line,
|
|
|
|
"m=audio %d RTP/%s", port, payload_type);
|
2021-02-17 22:48:33 +00:00
|
|
|
|
2021-04-08 14:43:13 +00:00
|
|
|
for (node = supported_codecs; node != NULL; node = node->next) {
|
2021-03-31 07:46:44 +00:00
|
|
|
MediaCodecInfo *codec = node->data;
|
|
|
|
|
2021-04-07 21:53:34 +00:00
|
|
|
g_string_append_printf (media_line, " %u", codec->payload_id);
|
2021-03-31 07:46:44 +00:00
|
|
|
g_string_append_printf (attribute_lines,
|
2021-04-07 21:53:34 +00:00
|
|
|
"a=rtpmap:%u %s/%u%s",
|
2021-03-31 07:46:44 +00:00
|
|
|
codec->payload_id,
|
|
|
|
codec->name,
|
|
|
|
codec->clock_rate,
|
|
|
|
"\r\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
g_string_append_printf (attribute_lines, "a=rtcp:%d\r\n", port + 1);
|
|
|
|
|
|
|
|
done:
|
2021-09-24 07:33:38 +00:00
|
|
|
if (self->session_ip && *self->session_ip)
|
|
|
|
return g_strdup_printf ("v=0\r\n"
|
|
|
|
"s=%s\r\n"
|
|
|
|
"%s\r\n"
|
|
|
|
"%s\r\n",
|
|
|
|
self->session_ip,
|
|
|
|
media_line->str,
|
|
|
|
attribute_lines->str);
|
|
|
|
else
|
|
|
|
return g_strdup_printf ("v=0\r\n"
|
|
|
|
"%s\r\n"
|
|
|
|
"%s\r\n",
|
|
|
|
media_line->str,
|
|
|
|
attribute_lines->str);
|
|
|
|
|
2021-02-17 22:48:33 +00:00
|
|
|
}
|
|
|
|
|
2021-03-30 16:15:24 +00:00
|
|
|
|
2021-04-08 14:43:13 +00:00
|
|
|
/* calls_sip_media_manager_static_capabilities:
|
|
|
|
*
|
|
|
|
* @self: A #CallsSipMediaManager
|
|
|
|
* @port: Should eventually come from the ICE stack
|
|
|
|
* @use_srtp: Whether to use srtp (not really handled)
|
|
|
|
*
|
2021-04-26 08:29:24 +00:00
|
|
|
* Returns: (transfer full): string describing capabilities
|
2021-04-08 14:43:13 +00:00
|
|
|
* to be used in the session description (SDP)
|
|
|
|
*/
|
|
|
|
char *
|
|
|
|
calls_sip_media_manager_static_capabilities (CallsSipMediaManager *self,
|
|
|
|
guint port,
|
|
|
|
gboolean use_srtp)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (CALLS_IS_SIP_MEDIA_MANAGER (self), NULL);
|
|
|
|
|
|
|
|
return calls_sip_media_manager_get_capabilities (self,
|
|
|
|
port,
|
|
|
|
use_srtp,
|
2021-11-23 14:04:18 +00:00
|
|
|
self->preferred_codecs);
|
2021-04-08 14:43:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-30 16:15:24 +00:00
|
|
|
MediaCodecInfo*
|
|
|
|
get_best_codec (CallsSipMediaManager *self)
|
|
|
|
{
|
|
|
|
return media_codec_by_name ("PCMA");
|
|
|
|
}
|
2021-04-08 14:43:13 +00:00
|
|
|
|
|
|
|
|
|
|
|
/* calls_sip_media_manager_codec_candiates:
|
|
|
|
*
|
|
|
|
* @self: A #CallsSipMediaManager
|
|
|
|
*
|
2021-04-26 08:29:24 +00:00
|
|
|
* Returns: (transfer none): A #GList of supported
|
2021-04-08 14:43:13 +00:00
|
|
|
* #MediaCodecInfo
|
|
|
|
*/
|
|
|
|
GList *
|
|
|
|
calls_sip_media_manager_codec_candidates (CallsSipMediaManager *self)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (CALLS_IS_SIP_MEDIA_MANAGER (self), NULL);
|
|
|
|
|
2021-11-23 14:04:18 +00:00
|
|
|
return self->preferred_codecs;
|
2021-04-08 14:43:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* calls_sip_media_manager_get_codecs_from_sdp
|
|
|
|
*
|
|
|
|
* @self: A #CallsSipMediaManager
|
|
|
|
* @sdp: A #sdp_media_t media description
|
|
|
|
*
|
2021-09-17 13:48:49 +00:00
|
|
|
* Returns: (transfer container): A #GList of codecs found in the
|
2021-04-08 14:43:13 +00:00
|
|
|
* SDP message
|
|
|
|
*/
|
|
|
|
GList *
|
|
|
|
calls_sip_media_manager_get_codecs_from_sdp (CallsSipMediaManager *self,
|
|
|
|
sdp_media_t *sdp_media)
|
|
|
|
{
|
|
|
|
GList *codecs = NULL;
|
|
|
|
sdp_rtpmap_t *rtpmap = NULL;
|
|
|
|
|
|
|
|
g_return_val_if_fail (CALLS_IS_SIP_MEDIA_MANAGER (self), NULL);
|
|
|
|
g_return_val_if_fail (sdp_media, NULL);
|
|
|
|
|
|
|
|
if (sdp_media->m_type != sdp_media_audio) {
|
|
|
|
g_warning ("Only the 'audio' media type is supported");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (rtpmap = sdp_media->m_rtpmaps; rtpmap != NULL; rtpmap = rtpmap->rm_next) {
|
|
|
|
MediaCodecInfo *codec = media_codec_by_payload_id (rtpmap->rm_pt);
|
|
|
|
if (codec)
|
|
|
|
codecs = g_list_append (codecs, codec);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sdp_media->m_next != NULL)
|
|
|
|
g_warning ("Currently only a single media session is supported");
|
|
|
|
|
|
|
|
if (codecs == NULL)
|
|
|
|
g_warning ("Did not find any common codecs");
|
|
|
|
|
|
|
|
return codecs;
|
|
|
|
}
|
2021-09-24 07:33:38 +00:00
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
calls_sip_media_manager_set_session_ip (CallsSipMediaManager *self,
|
|
|
|
const char *session_ip)
|
|
|
|
{
|
|
|
|
g_return_if_fail (CALLS_IS_SIP_MEDIA_MANAGER (self));
|
|
|
|
|
|
|
|
g_clear_pointer (&self->session_ip, g_free);
|
|
|
|
if (session_ip && *session_ip) {
|
|
|
|
g_debug ("Setting session IP to %s", session_ip);
|
|
|
|
self->session_ip = g_strdup (session_ip);
|
|
|
|
}
|
|
|
|
}
|