1
0
Fork 0
mirror of https://gitlab.gnome.org/GNOME/calls.git synced 2025-01-07 12:25:31 +00:00

media: manager: support multiple codecs SDP wise

static capabilities are now supporting multiple codecs and generate correct SDP
messages. This only relates to SDP offers. When we are answering an offer we
must take the offer from the other party into account when creating the answer.

Hence why we differentiate between "static" and "dynamic" (not implemented yet)
capability strings.
This commit is contained in:
Evangelos Ribeiro Tzaras 2021-03-31 09:46:44 +02:00 committed by Guido Gunther
parent 9fd43eaca8
commit 7d113d4180
3 changed files with 64 additions and 26 deletions

View file

@ -33,6 +33,8 @@
typedef struct _CallsSipMediaManager typedef struct _CallsSipMediaManager
{ {
GObject parent; GObject parent;
GList *supported_codecs;
} CallsSipMediaManager; } CallsSipMediaManager;
G_DEFINE_TYPE (CallsSipMediaManager, calls_sip_media_manager, G_TYPE_OBJECT); G_DEFINE_TYPE (CallsSipMediaManager, calls_sip_media_manager, G_TYPE_OBJECT);
@ -43,6 +45,8 @@ calls_sip_media_manager_finalize (GObject *object)
{ {
gst_deinit (); gst_deinit ();
g_list_free (CALLS_SIP_MEDIA_MANAGER (object)->supported_codecs);
G_OBJECT_CLASS (calls_sip_media_manager_parent_class)->finalize (object); G_OBJECT_CLASS (calls_sip_media_manager_parent_class)->finalize (object);
} }
@ -60,6 +64,8 @@ static void
calls_sip_media_manager_init (CallsSipMediaManager *self) calls_sip_media_manager_init (CallsSipMediaManager *self)
{ {
gst_init (NULL, NULL); gst_init (NULL, NULL);
self->supported_codecs = media_codecs_get_candidates ();
} }
@ -93,34 +99,45 @@ calls_sip_media_manager_static_capabilities (CallsSipMediaManager *self,
gboolean use_srtp) gboolean use_srtp)
{ {
char *payload_type = use_srtp ? "SAVP" : "AVP"; char *payload_type = use_srtp ? "SAVP" : "AVP";
g_autofree char *media_line = NULL; g_autoptr (GString) media_line = NULL;
g_autofree char *attribute_line = NULL; g_autoptr (GString) attribute_lines = NULL;
MediaCodecInfo *codec; GList *node;
g_return_val_if_fail (CALLS_IS_SIP_MEDIA_MANAGER (self), NULL); g_return_val_if_fail (CALLS_IS_SIP_MEDIA_MANAGER (self), NULL);
codec = get_best_codec (self); media_line = g_string_new (NULL);
/* TODO support multiplice codecs: f.e. audio 31337 RTP/AVP 9 8 0 96 */ attribute_lines = g_string_new (NULL);
media_line = g_strdup_printf ("audio %d RTP/%s %s",
port, payload_type, codec->payload_id);
attribute_line = g_strdup_printf ("rtpmap:%s %s/%s",
codec->payload_id, codec->name, codec->clock_rate);
/* TODO add attribute describing RTCP stream */ if (self->supported_codecs == NULL) {
g_warning ("No supported codecs found. Can't build meaningful SDP message");
g_string_append_printf (media_line, "m=audio 0 RTP/AVP 0");
goto done;
}
/* 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);
for (node = self->supported_codecs; node != NULL; node = node->next) {
MediaCodecInfo *codec = node->data;
g_string_append_printf (media_line, " %s", codec->payload_id);
g_string_append_printf (attribute_lines,
"a=rtpmap:%s %s/%s%s",
codec->payload_id,
codec->name,
codec->clock_rate,
"\r\n");
}
g_string_append_printf (attribute_lines, "a=rtcp:%d\r\n", port + 1);
done:
return g_strdup_printf ("v=0\r\n" return g_strdup_printf ("v=0\r\n"
"m=%s\r\n" "%s\r\n"
"a=%s\r\n", "%s\r\n",
media_line, media_line->str,
attribute_line); attribute_lines->str);
}
/* TODO lookup plugins in GStreamer */
gboolean
calls_sip_media_manager_supports_media (CallsSipMediaManager *self,
const char *media_type)
{
return TRUE;
} }

View file

@ -26,16 +26,21 @@
#include <glib.h> #include <glib.h>
/* TODO check available codecs during runtime */ /* Use the following codecs in order of preference */
static MediaCodecInfo gst_codecs[] = { static MediaCodecInfo gst_codecs[] = {
{"8", "PCMA", "8000", 1, "rtppcmapay", "rtppcmadepay", "alawenc", "alawdec"},
{"0", "PCMU", "8000", 1, "rtppcmupay", "rtppcmudepay", "mulawenc", "mulawdec"}, {"0", "PCMU", "8000", 1, "rtppcmupay", "rtppcmudepay", "mulawenc", "mulawdec"},
{"3", "GSM", "8000", 1, "rtpgsmpay", "rtpgsmdepay", "gsmenc", "gsmdec"}, {"3", "GSM", "8000", 1, "rtpgsmpay", "rtpgsmdepay", "gsmenc", "gsmdec"},
{"4", "G723", "8000", 1, "rtpg723pay", "rtpg723depay", "avenc_g723_1", "avdec_g723_1"}, // does not seem to work
{"8", "PCMA", "8000", 1, "rtppcmapay", "rtppcmadepay", "alawenc", "alawdec"},
{"9", "G722", "8000", 1, "rtpg722pay", "rtpg722depay", "avenc_g722", "avdec_g722"}, {"9", "G722", "8000", 1, "rtpg722pay", "rtpg722depay", "avenc_g722", "avdec_g722"},
{"4", "G723", "8000", 1, "rtpg723pay", "rtpg723depay", "avenc_g723_1", "avdec_g723_1"}, // does not seem to work
}; };
static gboolean
media_codec_available_in_gst (MediaCodecInfo *codec) {
/* TODO probe available plugins in GStreamer */
return TRUE;
}
MediaCodecInfo * MediaCodecInfo *
media_codec_by_name (const char *name) media_codec_by_name (const char *name)
@ -59,3 +64,18 @@ media_codec_get_gst_capabilities (MediaCodecInfo *codec)
codec->name, codec->name,
codec->payload_id); codec->payload_id);
} }
GList *
media_codecs_get_candidates ()
{
GList *candidates = NULL;
for (guint i = 0; i < G_N_ELEMENTS (gst_codecs); i++) {
if (media_codec_available_in_gst (&gst_codecs[i])) {
g_debug ("Adding %s to the codec candidates", gst_codecs[i].name);
candidates = g_list_append (candidates, &gst_codecs[i]);
}
}
return candidates;
}

View file

@ -45,3 +45,4 @@ typedef struct {
MediaCodecInfo* media_codec_by_name (const char *name); MediaCodecInfo* media_codec_by_name (const char *name);
gchar* media_codec_get_gst_capabilities (MediaCodecInfo *codec); gchar* media_codec_get_gst_capabilities (MediaCodecInfo *codec);
GList* media_codecs_get_candidates ();