rtspsrc: Add a small configurable teardown delay

This causes rtspsrc to send a teardown and wait on
PAUSED->READY transition, with a configurable delay.
Otherwise, typically teardown never gets sent in
playbin / uridecodebin where the transition back to NULL
happens too quickly.

The timeout is set to 100ms default.

https://bugzilla.gnome.org/show_bug.cgi?id=751994
This commit is contained in:
Jan Schmidt 2018-07-27 00:41:57 +10:00
parent 9a80cdbb40
commit f067b50dd6
2 changed files with 55 additions and 1 deletions

View File

@ -264,6 +264,7 @@ gst_rtsp_backchannel_get_type (void)
#define DEFAULT_MAX_TS_OFFSET G_GINT64_CONSTANT(3000000000) #define DEFAULT_MAX_TS_OFFSET G_GINT64_CONSTANT(3000000000)
#define DEFAULT_VERSION GST_RTSP_VERSION_1_0 #define DEFAULT_VERSION GST_RTSP_VERSION_1_0
#define DEFAULT_BACKCHANNEL GST_RTSP_BACKCHANNEL_NONE #define DEFAULT_BACKCHANNEL GST_RTSP_BACKCHANNEL_NONE
#define DEFAULT_TEARDOWN_TIMEOUT (100 * GST_MSECOND)
enum enum
{ {
@ -308,6 +309,7 @@ enum
PROP_MAX_TS_OFFSET, PROP_MAX_TS_OFFSET,
PROP_DEFAULT_VERSION, PROP_DEFAULT_VERSION,
PROP_BACKCHANNEL, PROP_BACKCHANNEL,
PROP_TEARDOWN_TIMEOUT,
}; };
#define GST_TYPE_RTSP_NAT_METHOD (gst_rtsp_nat_method_get_type()) #define GST_TYPE_RTSP_NAT_METHOD (gst_rtsp_nat_method_get_type())
@ -875,6 +877,21 @@ gst_rtspsrc_class_init (GstRTSPSrcClass * klass)
GST_TYPE_RTSP_BACKCHANNEL, BACKCHANNEL_NONE, GST_TYPE_RTSP_BACKCHANNEL, BACKCHANNEL_NONE,
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
/**
* GstRtspSrc:teardown-timeout
*
* When transitioning PAUSED-READY, allow up to timeout (in nanoseconds)
* delay in order to send teardown (0 = disabled)
*
* Since: 1.14
*/
g_object_class_install_property (gobject_class, PROP_TEARDOWN_TIMEOUT,
g_param_spec_uint64 ("teardown-timeout", "Teardown Timeout",
"When transitioning PAUSED-READY, allow up to timeout (in nanoseconds) "
"delay in order to send teardown (0 = disabled)",
0, G_MAXUINT64, DEFAULT_TEARDOWN_TIMEOUT,
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
/** /**
* GstRTSPSrc::handle-request: * GstRTSPSrc::handle-request:
* @rtspsrc: a #GstRTSPSrc * @rtspsrc: a #GstRTSPSrc
@ -1087,6 +1104,7 @@ gst_rtspsrc_init (GstRTSPSrc * src)
src->max_ts_offset_is_set = FALSE; src->max_ts_offset_is_set = FALSE;
src->default_version = DEFAULT_VERSION; src->default_version = DEFAULT_VERSION;
src->version = GST_RTSP_VERSION_INVALID; src->version = GST_RTSP_VERSION_INVALID;
src->teardown_timeout = DEFAULT_TEARDOWN_TIMEOUT;
/* get a list of all extensions */ /* get a list of all extensions */
src->extensions = gst_rtsp_ext_list_get (); src->extensions = gst_rtsp_ext_list_get ();
@ -1106,6 +1124,7 @@ gst_rtspsrc_init (GstRTSPSrc * src)
g_mutex_init (&src->conninfo.send_lock); g_mutex_init (&src->conninfo.send_lock);
g_mutex_init (&src->conninfo.recv_lock); g_mutex_init (&src->conninfo.recv_lock);
g_cond_init (&src->cmd_cond);
GST_OBJECT_FLAG_SET (src, GST_ELEMENT_FLAG_SOURCE); GST_OBJECT_FLAG_SET (src, GST_ELEMENT_FLAG_SOURCE);
gst_bin_set_suppressed_flags (GST_BIN (src), gst_bin_set_suppressed_flags (GST_BIN (src),
@ -1150,6 +1169,7 @@ gst_rtspsrc_finalize (GObject * object)
g_mutex_clear (&rtspsrc->conninfo.send_lock); g_mutex_clear (&rtspsrc->conninfo.send_lock);
g_mutex_clear (&rtspsrc->conninfo.recv_lock); g_mutex_clear (&rtspsrc->conninfo.recv_lock);
g_cond_clear (&rtspsrc->cmd_cond);
G_OBJECT_CLASS (parent_class)->finalize (object); G_OBJECT_CLASS (parent_class)->finalize (object);
} }
@ -1399,6 +1419,9 @@ gst_rtspsrc_set_property (GObject * object, guint prop_id, const GValue * value,
case PROP_BACKCHANNEL: case PROP_BACKCHANNEL:
rtspsrc->backchannel = g_value_get_enum (value); rtspsrc->backchannel = g_value_get_enum (value);
break; break;
case PROP_TEARDOWN_TIMEOUT:
rtspsrc->teardown_timeout = g_value_get_uint64 (value);
break;
default: default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break; break;
@ -1560,6 +1583,9 @@ gst_rtspsrc_get_property (GObject * object, guint prop_id, GValue * value,
case PROP_BACKCHANNEL: case PROP_BACKCHANNEL:
g_value_set_enum (value, rtspsrc->backchannel); g_value_set_enum (value, rtspsrc->backchannel);
break; break;
case PROP_TEARDOWN_TIMEOUT:
g_value_set_uint64 (value, rtspsrc->teardown_timeout);
break;
default: default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break; break;
@ -5599,6 +5625,28 @@ gst_rtspsrc_loop_send_cmd (GstRTSPSrc * src, gint cmd, gint mask)
return flushed; return flushed;
} }
static gboolean
gst_rtspsrc_loop_send_cmd_and_wait (GstRTSPSrc * src, gint cmd, gint mask,
GstClockTime timeout)
{
gboolean flushed = gst_rtspsrc_loop_send_cmd (src, cmd, mask);
if (timeout > 0) {
gint64 end_time = g_get_monotonic_time () + (timeout / 1000);
GST_OBJECT_LOCK (src);
while (src->pending_cmd == cmd || src->busy_cmd == cmd) {
if (!g_cond_wait_until (&src->cmd_cond, GST_OBJECT_GET_LOCK (src),
end_time)) {
GST_WARNING_OBJECT (src,
"Timed out waiting for TEARDOWN to be processed.");
break; /* timeout passed */
}
}
GST_OBJECT_UNLOCK (src);
}
return flushed;
}
static gboolean static gboolean
gst_rtspsrc_loop (GstRTSPSrc * src) gst_rtspsrc_loop (GstRTSPSrc * src)
{ {
@ -7603,6 +7651,7 @@ gst_rtspsrc_close (GstRTSPSrc * src, gboolean async, gboolean only_close)
/* do TEARDOWN */ /* do TEARDOWN */
res = res =
gst_rtspsrc_init_request (src, &request, GST_RTSP_TEARDOWN, setup_url); gst_rtspsrc_init_request (src, &request, GST_RTSP_TEARDOWN, setup_url);
GST_LOG_OBJECT (src, "Teardown on %s", setup_url);
if (res < 0) if (res < 0)
goto create_request_failed; goto create_request_failed;
@ -8370,6 +8419,8 @@ gst_rtspsrc_thread (GstRTSPSrc * src)
} }
GST_OBJECT_LOCK (src); GST_OBJECT_LOCK (src);
/* No more cmds, wake any waiters */
g_cond_broadcast (&src->cmd_cond);
/* and go back to sleep */ /* and go back to sleep */
if (src->pending_cmd == CMD_WAIT) { if (src->pending_cmd == CMD_WAIT) {
if (src->task) if (src->task)
@ -8505,7 +8556,8 @@ gst_rtspsrc_change_state (GstElement * element, GstStateChange transition)
ret = GST_STATE_CHANGE_NO_PREROLL; ret = GST_STATE_CHANGE_NO_PREROLL;
break; break;
case GST_STATE_CHANGE_PAUSED_TO_READY: case GST_STATE_CHANGE_PAUSED_TO_READY:
gst_rtspsrc_loop_send_cmd (rtspsrc, CMD_CLOSE, CMD_ALL); gst_rtspsrc_loop_send_cmd_and_wait (rtspsrc, CMD_CLOSE, CMD_ALL,
rtspsrc->teardown_timeout);
ret = GST_STATE_CHANGE_SUCCESS; ret = GST_STATE_CHANGE_SUCCESS;
break; break;
case GST_STATE_CHANGE_READY_TO_NULL: case GST_STATE_CHANGE_READY_TO_NULL:

View File

@ -200,6 +200,7 @@ struct _GstRTSPSrc {
/* UDP mode loop */ /* UDP mode loop */
gint pending_cmd; gint pending_cmd;
gint busy_cmd; gint busy_cmd;
GCond cmd_cond;
gboolean ignore_timeout; gboolean ignore_timeout;
gboolean open_error; gboolean open_error;
@ -256,6 +257,7 @@ struct _GstRTSPSrc {
gint64 max_ts_offset; gint64 max_ts_offset;
gboolean max_ts_offset_is_set; gboolean max_ts_offset_is_set;
gint backchannel; gint backchannel;
GstClockTime teardown_timeout;
/* state */ /* state */
GstRTSPState state; GstRTSPState state;