mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-09 00:45:56 +00:00
decodebin3: filter error message and store latency message for candidate decoder
If the GST_MESSAGE_SRC of error message belongs to candidate decoders, filter the error message and don't forward it as there might be a following candidate decoder that can be used. If the GST_MESSAGE_SRC of error message belongs to candidate decoders, store the latency message and handle it after decoder is accepted. This is to avoid the selection lock failure if decodebin3 needs to handle latency message for candidate decoders when sending sticky event. Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/4677>
This commit is contained in:
parent
887ae4d9e0
commit
95ac8b0cea
1 changed files with 86 additions and 0 deletions
|
@ -204,6 +204,13 @@ typedef struct _DecodebinInputStream DecodebinInputStream;
|
||||||
typedef struct _DecodebinInput DecodebinInput;
|
typedef struct _DecodebinInput DecodebinInput;
|
||||||
typedef struct _DecodebinOutputStream DecodebinOutputStream;
|
typedef struct _DecodebinOutputStream DecodebinOutputStream;
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
GstElement *element;
|
||||||
|
GstMessage *error; // Last error message seen for that element
|
||||||
|
GstMessage *latency; // Last latency message seen for that element
|
||||||
|
} CandidateDecoder;
|
||||||
|
|
||||||
struct _GstDecodebin3
|
struct _GstDecodebin3
|
||||||
{
|
{
|
||||||
GstBin bin;
|
GstBin bin;
|
||||||
|
@ -264,6 +271,8 @@ struct _GstDecodebin3
|
||||||
|
|
||||||
/* Properties */
|
/* Properties */
|
||||||
GstCaps *caps;
|
GstCaps *caps;
|
||||||
|
|
||||||
|
GList *candidate_decoders;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct _GstDecodebin3Class
|
struct _GstDecodebin3Class
|
||||||
|
@ -919,6 +928,30 @@ send_sticky_events (GstDecodebin3 * dbin, GstPad * pad)
|
||||||
return data.ret;
|
return data.ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static CandidateDecoder *
|
||||||
|
add_candidate_decoder (GstDecodebin3 * dbin, GstElement * element)
|
||||||
|
{
|
||||||
|
GST_OBJECT_LOCK (dbin);
|
||||||
|
CandidateDecoder *candidate;
|
||||||
|
candidate = g_new0 (CandidateDecoder, 1);
|
||||||
|
candidate->element = element;
|
||||||
|
dbin->candidate_decoders =
|
||||||
|
g_list_prepend (dbin->candidate_decoders, candidate);
|
||||||
|
GST_OBJECT_UNLOCK (dbin);
|
||||||
|
return candidate;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
remove_candidate_decoder (GstDecodebin3 * dbin, CandidateDecoder * candidate)
|
||||||
|
{
|
||||||
|
GST_OBJECT_LOCK (dbin);
|
||||||
|
dbin->candidate_decoders =
|
||||||
|
g_list_remove (dbin->candidate_decoders, candidate);
|
||||||
|
if (candidate->error)
|
||||||
|
gst_message_unref (candidate->error);
|
||||||
|
GST_OBJECT_UNLOCK (dbin);
|
||||||
|
}
|
||||||
|
|
||||||
/* Call with INPUT_LOCK taken */
|
/* Call with INPUT_LOCK taken */
|
||||||
static gboolean
|
static gboolean
|
||||||
ensure_input_parsebin (GstDecodebin3 * dbin, DecodebinInput * input)
|
ensure_input_parsebin (GstDecodebin3 * dbin, DecodebinInput * input)
|
||||||
|
@ -2046,9 +2079,34 @@ gst_decodebin3_handle_message (GstBin * bin, GstMessage * message)
|
||||||
{
|
{
|
||||||
GstDecodebin3 *dbin = (GstDecodebin3 *) bin;
|
GstDecodebin3 *dbin = (GstDecodebin3 *) bin;
|
||||||
gboolean posting_collection = FALSE;
|
gboolean posting_collection = FALSE;
|
||||||
|
GList *l;
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (bin, "Got Message %s", GST_MESSAGE_TYPE_NAME (message));
|
GST_DEBUG_OBJECT (bin, "Got Message %s", GST_MESSAGE_TYPE_NAME (message));
|
||||||
|
|
||||||
|
GST_OBJECT_LOCK (dbin);
|
||||||
|
for (l = dbin->candidate_decoders; l; l = l->next) {
|
||||||
|
CandidateDecoder *candidate = l->data;
|
||||||
|
if (GST_OBJECT_CAST (candidate->element) == GST_MESSAGE_SRC (message)) {
|
||||||
|
if (GST_MESSAGE_TYPE (message) == GST_MESSAGE_ERROR) {
|
||||||
|
if (candidate->error)
|
||||||
|
gst_message_unref (candidate->error);
|
||||||
|
candidate->error = message;
|
||||||
|
GST_OBJECT_UNLOCK (dbin);
|
||||||
|
return;
|
||||||
|
} else if (GST_MESSAGE_TYPE (message) == GST_MESSAGE_LATENCY) {
|
||||||
|
if (candidate->latency)
|
||||||
|
gst_message_unref (candidate->latency);
|
||||||
|
GST_DEBUG_OBJECT (bin, "store latency message for %" GST_PTR_FORMAT,
|
||||||
|
candidate->element);
|
||||||
|
candidate->latency = message;
|
||||||
|
GST_OBJECT_UNLOCK (dbin);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
GST_OBJECT_UNLOCK (dbin);
|
||||||
|
|
||||||
switch (GST_MESSAGE_TYPE (message)) {
|
switch (GST_MESSAGE_TYPE (message)) {
|
||||||
case GST_MESSAGE_STREAM_COLLECTION:
|
case GST_MESSAGE_STREAM_COLLECTION:
|
||||||
{
|
{
|
||||||
|
@ -2139,6 +2197,27 @@ drop_message:
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
handle_stored_latency_message (GstDecodebin3 * dbin,
|
||||||
|
DecodebinOutputStream * output, CandidateDecoder * candidate)
|
||||||
|
{
|
||||||
|
GstClockTime min, max;
|
||||||
|
if (candidate->latency && GST_IS_VIDEO_DECODER (candidate->element)) {
|
||||||
|
gst_video_decoder_get_latency (GST_VIDEO_DECODER (candidate->element),
|
||||||
|
&min, &max);
|
||||||
|
GST_DEBUG_OBJECT (dbin,
|
||||||
|
"Got latency update from %" GST_PTR_FORMAT ". min: %"
|
||||||
|
GST_TIME_FORMAT " max: %" GST_TIME_FORMAT, candidate->element,
|
||||||
|
GST_TIME_ARGS (min), GST_TIME_ARGS (max));
|
||||||
|
output->decoder_latency = min;
|
||||||
|
/* Trigger recalculation */
|
||||||
|
gst_decodebin3_update_min_interleave (dbin);
|
||||||
|
|
||||||
|
GST_BIN_CLASS (parent_class)->handle_message (GST_BIN_CAST (dbin),
|
||||||
|
candidate->latency);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static DecodebinOutputStream *
|
static DecodebinOutputStream *
|
||||||
find_free_compatible_output (GstDecodebin3 * dbin, GstStream * stream)
|
find_free_compatible_output (GstDecodebin3 * dbin, GstStream * stream)
|
||||||
{
|
{
|
||||||
|
@ -2948,6 +3027,7 @@ reconfigure_output_stream (DecodebinOutputStream * output,
|
||||||
|
|
||||||
while (next_factory) {
|
while (next_factory) {
|
||||||
gboolean decoder_failed = FALSE;
|
gboolean decoder_failed = FALSE;
|
||||||
|
CandidateDecoder *candidate = NULL;
|
||||||
|
|
||||||
/* If we don't have a decoder yet, instantiate one */
|
/* If we don't have a decoder yet, instantiate one */
|
||||||
output->decoder = gst_element_factory_create ((GstElementFactory *)
|
output->decoder = gst_element_factory_create ((GstElementFactory *)
|
||||||
|
@ -2972,6 +3052,7 @@ reconfigure_output_stream (DecodebinOutputStream * output,
|
||||||
(GstPadProbeCallback) keyframe_waiter_probe, output, NULL);
|
(GstPadProbeCallback) keyframe_waiter_probe, output, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
candidate = add_candidate_decoder (dbin, output->decoder);
|
||||||
if (gst_pad_link_full (slot->src_pad, output->decoder_sink,
|
if (gst_pad_link_full (slot->src_pad, output->decoder_sink,
|
||||||
GST_PAD_LINK_CHECK_NOTHING) != GST_PAD_LINK_OK) {
|
GST_PAD_LINK_CHECK_NOTHING) != GST_PAD_LINK_OK) {
|
||||||
GST_WARNING_OBJECT (dbin, "could not link to %s:%s",
|
GST_WARNING_OBJECT (dbin, "could not link to %s:%s",
|
||||||
|
@ -3016,6 +3097,9 @@ reconfigure_output_stream (DecodebinOutputStream * output,
|
||||||
GST_PAD_STREAM_UNLOCK (output->decoder_sink);
|
GST_PAD_STREAM_UNLOCK (output->decoder_sink);
|
||||||
output->linked = TRUE;
|
output->linked = TRUE;
|
||||||
GST_DEBUG ("created decoder %" GST_PTR_FORMAT, output->decoder);
|
GST_DEBUG ("created decoder %" GST_PTR_FORMAT, output->decoder);
|
||||||
|
|
||||||
|
handle_stored_latency_message (dbin, output, candidate);
|
||||||
|
remove_candidate_decoder (dbin, candidate);
|
||||||
}
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
@ -3024,6 +3108,8 @@ reconfigure_output_stream (DecodebinOutputStream * output,
|
||||||
{
|
{
|
||||||
if (decoder_failed)
|
if (decoder_failed)
|
||||||
remove_decoder_link (output, slot);
|
remove_decoder_link (output, slot);
|
||||||
|
if (candidate)
|
||||||
|
remove_candidate_decoder (dbin, candidate);
|
||||||
|
|
||||||
if (!next_factory->next) {
|
if (!next_factory->next) {
|
||||||
ret = FALSE;
|
ret = FALSE;
|
||||||
|
|
Loading…
Reference in a new issue