mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-10 01:15:39 +00:00
04aa977550
Previously, reassigning loop index l in nicestream.c could cause a segfault if l->data was null, as it could reassign l to a null variable, triggering the loop postassignment l->next, which then segfaults due to l now being null. It is instead moved into the loop. _delete_transport already performs the reassignment inline. Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/4213>
338 lines
9.1 KiB
C
338 lines
9.1 KiB
C
/* GStreamer
|
|
* Copyright (C) 2017 Matthew Waters <matthew@centricular.com>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library 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
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with this library; if not, write to the
|
|
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
|
* Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#include "nicestream.h"
|
|
#include "nicetransport.h"
|
|
|
|
#define GST_CAT_DEFAULT gst_webrtc_nice_stream_debug
|
|
GST_DEBUG_CATEGORY_STATIC (GST_CAT_DEFAULT);
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_ICE,
|
|
};
|
|
|
|
struct _GstWebRTCNiceStreamPrivate
|
|
{
|
|
gboolean gathered;
|
|
GList *transports;
|
|
gboolean gathering_started;
|
|
gulong candidate_gathering_done_id;
|
|
GWeakRef ice_weak;
|
|
};
|
|
|
|
#define gst_webrtc_nice_stream_parent_class parent_class
|
|
G_DEFINE_TYPE_WITH_CODE (GstWebRTCNiceStream, gst_webrtc_nice_stream,
|
|
GST_TYPE_WEBRTC_ICE_STREAM, G_ADD_PRIVATE (GstWebRTCNiceStream)
|
|
GST_DEBUG_CATEGORY_INIT (gst_webrtc_nice_stream_debug,
|
|
"webrtcnicestream", 0, "webrtcnicestream"););
|
|
|
|
static void
|
|
gst_webrtc_nice_stream_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstWebRTCNiceStream *stream = GST_WEBRTC_NICE_STREAM (object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_ICE:
|
|
g_weak_ref_set (&stream->priv->ice_weak, g_value_get_object (value));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_webrtc_nice_stream_get_property (GObject * object, guint prop_id,
|
|
GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstWebRTCNiceStream *stream = GST_WEBRTC_NICE_STREAM (object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_ICE:
|
|
g_value_take_object (value, g_weak_ref_get (&stream->priv->ice_weak));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static GWeakRef *
|
|
weak_new (gpointer object)
|
|
{
|
|
GWeakRef *weak = g_new0 (GWeakRef, 1);
|
|
g_weak_ref_init (weak, object);
|
|
return weak;
|
|
}
|
|
|
|
static void
|
|
weak_free (GWeakRef * weak)
|
|
{
|
|
g_weak_ref_clear (weak);
|
|
g_free (weak);
|
|
}
|
|
|
|
static void
|
|
gst_webrtc_nice_stream_finalize (GObject * object)
|
|
{
|
|
GstWebRTCNiceStream *stream = GST_WEBRTC_NICE_STREAM (object);
|
|
GstWebRTCNice *ice = g_weak_ref_get (&stream->priv->ice_weak);
|
|
|
|
if (ice) {
|
|
NiceAgent *agent;
|
|
g_object_get (ice, "agent", &agent, NULL);
|
|
|
|
if (stream->priv->candidate_gathering_done_id != 0) {
|
|
g_signal_handler_disconnect (agent,
|
|
stream->priv->candidate_gathering_done_id);
|
|
}
|
|
|
|
g_object_unref (agent);
|
|
gst_object_unref (ice);
|
|
}
|
|
|
|
g_list_foreach (stream->priv->transports, (GFunc) weak_free, NULL);
|
|
g_list_free (stream->priv->transports);
|
|
stream->priv->transports = NULL;
|
|
|
|
g_weak_ref_clear (&stream->priv->ice_weak);
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
static GList *
|
|
_delete_transport (GList ** transports, GList * link)
|
|
{
|
|
GList *next = link->next;
|
|
weak_free (link->data);
|
|
*transports = g_list_delete_link (*transports, link);
|
|
return next;
|
|
}
|
|
|
|
static void
|
|
_on_candidate_gathering_done (NiceAgent * agent, guint stream_id,
|
|
GWeakRef * ice_weak)
|
|
{
|
|
GstWebRTCNiceStream *ice = g_weak_ref_get (ice_weak);
|
|
GList *l;
|
|
|
|
if (!ice)
|
|
return;
|
|
|
|
if (stream_id != GST_WEBRTC_ICE_STREAM (ice)->stream_id)
|
|
goto cleanup;
|
|
|
|
GST_DEBUG_OBJECT (ice, "%u gathering done", stream_id);
|
|
|
|
ice->priv->gathered = TRUE;
|
|
|
|
for (l = ice->priv->transports; l;) {
|
|
GstWebRTCICETransport *trans = g_weak_ref_get (l->data);
|
|
|
|
if (trans) {
|
|
gst_webrtc_ice_transport_gathering_state_change (trans,
|
|
GST_WEBRTC_ICE_GATHERING_STATE_COMPLETE);
|
|
g_object_unref (trans);
|
|
l = l->next;
|
|
} else {
|
|
l = _delete_transport (&ice->priv->transports, l);
|
|
}
|
|
}
|
|
|
|
cleanup:
|
|
gst_object_unref (ice);
|
|
}
|
|
|
|
static GstWebRTCICETransport *
|
|
gst_webrtc_nice_stream_find_transport (GstWebRTCICEStream * stream,
|
|
GstWebRTCICEComponent component)
|
|
{
|
|
GstWebRTCICEComponent trans_comp;
|
|
GstWebRTCICETransport *ret;
|
|
GList *l;
|
|
GstWebRTCNiceStream *nice_stream = GST_WEBRTC_NICE_STREAM (stream);
|
|
|
|
for (l = nice_stream->priv->transports; l;) {
|
|
GstWebRTCICETransport *trans = g_weak_ref_get (l->data);
|
|
if (trans) {
|
|
g_object_get (trans, "component", &trans_comp, NULL);
|
|
|
|
if (component == trans_comp)
|
|
return trans;
|
|
else
|
|
gst_object_unref (trans);
|
|
l = l->next;
|
|
} else {
|
|
l = _delete_transport (&nice_stream->priv->transports, l);
|
|
}
|
|
}
|
|
|
|
ret =
|
|
GST_WEBRTC_ICE_TRANSPORT (gst_webrtc_nice_transport_new (nice_stream,
|
|
component));
|
|
nice_stream->priv->transports =
|
|
g_list_prepend (nice_stream->priv->transports, weak_new (ret));
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void
|
|
gst_webrtc_nice_stream_constructed (GObject * object)
|
|
{
|
|
GstWebRTCNiceStream *stream;
|
|
NiceAgent *agent;
|
|
GstWebRTCNice *ice;
|
|
|
|
G_OBJECT_CLASS (parent_class)->constructed (object);
|
|
|
|
stream = GST_WEBRTC_NICE_STREAM (object);
|
|
ice = g_weak_ref_get (&stream->priv->ice_weak);
|
|
|
|
|
|
g_assert (ice != NULL);
|
|
g_object_get (ice, "agent", &agent, NULL);
|
|
stream->priv->candidate_gathering_done_id = g_signal_connect_data (agent,
|
|
"candidate-gathering-done", G_CALLBACK (_on_candidate_gathering_done),
|
|
weak_new (stream), (GClosureNotify) weak_free, (GConnectFlags) 0);
|
|
|
|
g_object_unref (agent);
|
|
gst_object_unref (ice);
|
|
}
|
|
|
|
static gboolean
|
|
gst_webrtc_nice_stream_gather_candidates (GstWebRTCICEStream * stream)
|
|
{
|
|
NiceAgent *agent;
|
|
GList *l;
|
|
GstWebRTCICE *ice;
|
|
gboolean ret = TRUE;
|
|
GstWebRTCNiceStream *nice_stream = GST_WEBRTC_NICE_STREAM (stream);
|
|
|
|
GST_DEBUG_OBJECT (nice_stream, "start gathering candidates");
|
|
|
|
if (nice_stream->priv->gathered)
|
|
return TRUE;
|
|
|
|
for (l = nice_stream->priv->transports; l;) {
|
|
GstWebRTCICETransport *trans = g_weak_ref_get (l->data);
|
|
|
|
if (trans) {
|
|
gst_webrtc_ice_transport_gathering_state_change (trans,
|
|
GST_WEBRTC_ICE_GATHERING_STATE_GATHERING);
|
|
g_object_unref (trans);
|
|
l = l->next;
|
|
} else {
|
|
l = _delete_transport (&nice_stream->priv->transports, l);
|
|
}
|
|
}
|
|
|
|
ice = GST_WEBRTC_ICE (g_weak_ref_get (&nice_stream->priv->ice_weak));
|
|
g_assert (ice != NULL);
|
|
|
|
g_object_get (ice, "agent", &agent, NULL);
|
|
|
|
if (!nice_stream->priv->gathering_started) {
|
|
if (ice->min_rtp_port != 0 || ice->max_rtp_port != 65535) {
|
|
if (ice->min_rtp_port > ice->max_rtp_port) {
|
|
GST_ERROR_OBJECT (ice,
|
|
"invalid port range: min-rtp-port %d must be <= max-rtp-port %d",
|
|
ice->min_rtp_port, ice->max_rtp_port);
|
|
ret = FALSE;
|
|
goto cleanup;
|
|
}
|
|
|
|
nice_agent_set_port_range (agent, stream->stream_id,
|
|
NICE_COMPONENT_TYPE_RTP, ice->min_rtp_port, ice->max_rtp_port);
|
|
}
|
|
/* mark as gathering started to prevent changing ports again */
|
|
nice_stream->priv->gathering_started = TRUE;
|
|
}
|
|
|
|
if (!nice_agent_gather_candidates (agent, stream->stream_id)) {
|
|
ret = FALSE;
|
|
goto cleanup;
|
|
}
|
|
|
|
for (l = nice_stream->priv->transports; l;) {
|
|
GstWebRTCNiceTransport *trans = g_weak_ref_get (l->data);
|
|
|
|
if (trans) {
|
|
gst_webrtc_nice_transport_update_buffer_size (trans);
|
|
g_object_unref (trans);
|
|
l = l->next;
|
|
} else {
|
|
l = _delete_transport (&nice_stream->priv->transports, l);
|
|
}
|
|
}
|
|
|
|
cleanup:
|
|
if (agent)
|
|
g_object_unref (agent);
|
|
if (ice)
|
|
gst_object_unref (ice);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void
|
|
gst_webrtc_nice_stream_class_init (GstWebRTCNiceStreamClass * klass)
|
|
{
|
|
GObjectClass *gobject_class = (GObjectClass *) klass;
|
|
GstWebRTCICEStreamClass *gst_webrtc_ice_stream_class =
|
|
GST_WEBRTC_ICE_STREAM_CLASS (klass);
|
|
|
|
gst_webrtc_ice_stream_class->find_transport =
|
|
gst_webrtc_nice_stream_find_transport;
|
|
gst_webrtc_ice_stream_class->gather_candidates =
|
|
gst_webrtc_nice_stream_gather_candidates;
|
|
|
|
gobject_class->constructed = gst_webrtc_nice_stream_constructed;
|
|
gobject_class->get_property = gst_webrtc_nice_stream_get_property;
|
|
gobject_class->set_property = gst_webrtc_nice_stream_set_property;
|
|
gobject_class->finalize = gst_webrtc_nice_stream_finalize;
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_ICE,
|
|
g_param_spec_object ("ice",
|
|
"ICE", "ICE agent associated with this stream",
|
|
GST_TYPE_WEBRTC_ICE,
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
|
|
}
|
|
|
|
static void
|
|
gst_webrtc_nice_stream_init (GstWebRTCNiceStream * stream)
|
|
{
|
|
stream->priv = gst_webrtc_nice_stream_get_instance_private (stream);
|
|
|
|
g_weak_ref_init (&stream->priv->ice_weak, NULL);
|
|
}
|
|
|
|
GstWebRTCNiceStream *
|
|
gst_webrtc_nice_stream_new (GstWebRTCICE * ice, guint stream_id)
|
|
{
|
|
return g_object_new (GST_TYPE_WEBRTC_NICE_STREAM, "ice", ice,
|
|
"stream-id", stream_id, NULL);
|
|
}
|