videorate: fix assertion when pushing last and only buffer without duration

Fixing this pipeline:
  gst-launch-1.0 filesrc location=sample.png ! pngdec ! videorate ! fakesink

- videorate receives a single buffer with pts = 0, duration = invalid;
- then it receives eos triggering this buffer to be pushed downstream;
- the pushing code was assuming that a duration was set, which is
  impossible as we received a single buffer and no output framerate was
  set either. So the best we can do is to push the buffer without
  duration.

Fix #1177

Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/2320>
This commit is contained in:
Guillaume Desmottes 2022-04-26 10:58:08 +02:00 committed by Tim-Philipp Müller
parent b8462b0624
commit 69b205613f

View file

@ -649,7 +649,7 @@ gst_video_rate_init (GstVideoRate * videorate)
/* @outbuf: (transfer full) needs to be writable */ /* @outbuf: (transfer full) needs to be writable */
static GstFlowReturn static GstFlowReturn
gst_video_rate_push_buffer (GstVideoRate * videorate, GstBuffer * outbuf, gst_video_rate_push_buffer (GstVideoRate * videorate, GstBuffer * outbuf,
gboolean duplicate, GstClockTime next_intime) gboolean duplicate, GstClockTime next_intime, gboolean invalid_duration)
{ {
GstFlowReturn res; GstFlowReturn res;
GstClockTime push_ts; GstClockTime push_ts;
@ -707,7 +707,7 @@ gst_video_rate_push_buffer (GstVideoRate * videorate, GstBuffer * outbuf,
videorate->to_rate_denominator * GST_SECOND, videorate->to_rate_denominator * GST_SECOND,
videorate->to_rate_numerator); videorate->to_rate_numerator);
GST_BUFFER_DURATION (outbuf) = videorate->next_ts - push_ts; GST_BUFFER_DURATION (outbuf) = videorate->next_ts - push_ts;
} else { } else if (!invalid_duration) {
/* There must always be a valid duration on prevbuf if rate > 0, /* There must always be a valid duration on prevbuf if rate > 0,
* it is ensured in the transform_ip function */ * it is ensured in the transform_ip function */
g_assert (GST_BUFFER_PTS_IS_VALID (outbuf)); g_assert (GST_BUFFER_PTS_IS_VALID (outbuf));
@ -737,7 +737,7 @@ gst_video_rate_push_buffer (GstVideoRate * videorate, GstBuffer * outbuf,
/* flush the oldest buffer */ /* flush the oldest buffer */
static GstFlowReturn static GstFlowReturn
gst_video_rate_flush_prev (GstVideoRate * videorate, gboolean duplicate, gst_video_rate_flush_prev (GstVideoRate * videorate, gboolean duplicate,
GstClockTime next_intime) GstClockTime next_intime, gboolean invalid_duration)
{ {
GstBuffer *outbuf; GstBuffer *outbuf;
@ -748,7 +748,8 @@ gst_video_rate_flush_prev (GstVideoRate * videorate, gboolean duplicate,
/* make sure we can write to the metadata */ /* make sure we can write to the metadata */
outbuf = gst_buffer_make_writable (outbuf); outbuf = gst_buffer_make_writable (outbuf);
return gst_video_rate_push_buffer (videorate, outbuf, duplicate, next_intime); return gst_video_rate_push_buffer (videorate, outbuf, duplicate, next_intime,
invalid_duration);
/* WARNINGS */ /* WARNINGS */
eos_before_buffers: eos_before_buffers:
@ -824,7 +825,7 @@ gst_video_rate_sink_event (GstBaseTransform * trans, GstEvent * event)
|| count < 1)) { || count < 1)) {
res = res =
gst_video_rate_flush_prev (videorate, count > 0, gst_video_rate_flush_prev (videorate, count > 0,
GST_CLOCK_TIME_NONE); GST_CLOCK_TIME_NONE, FALSE);
count++; count++;
} }
if (count > 1) { if (count > 1) {
@ -886,7 +887,7 @@ gst_video_rate_sink_event (GstBaseTransform * trans, GstEvent * event)
videorate->segment.start) videorate->segment.start)
)) { )) {
res = gst_video_rate_flush_prev (videorate, count > 0, res = gst_video_rate_flush_prev (videorate, count > 0,
GST_CLOCK_TIME_NONE); GST_CLOCK_TIME_NONE, FALSE);
count++; count++;
} }
} else if (!videorate->drop_only && videorate->prevbuf) { } else if (!videorate->drop_only && videorate->prevbuf) {
@ -904,12 +905,15 @@ gst_video_rate_sink_event (GstBaseTransform * trans, GstEvent * event)
|| count < 1)) { || count < 1)) {
res = res =
gst_video_rate_flush_prev (videorate, count > 0, gst_video_rate_flush_prev (videorate, count > 0,
GST_CLOCK_TIME_NONE); GST_CLOCK_TIME_NONE, FALSE);
count++; count++;
} }
} else { } else {
/* allow the duration to be invalid as there is no way to infer it if we
* received a single buffer and not output framerate was set. */
res = res =
gst_video_rate_flush_prev (videorate, FALSE, GST_CLOCK_TIME_NONE); gst_video_rate_flush_prev (videorate, FALSE, GST_CLOCK_TIME_NONE,
TRUE);
count = 1; count = 1;
} }
} }
@ -1397,12 +1401,14 @@ gst_video_rate_do_max_duplicate (GstVideoRate * videorate, GstBuffer * buffer,
* previous buffer */ * previous buffer */
if (videorate->segment.rate < 0.0) { if (videorate->segment.rate < 0.0) {
while (videorate->next_ts > prevtime) { while (videorate->next_ts > prevtime) {
gst_video_rate_flush_prev (videorate, *count > 0, GST_CLOCK_TIME_NONE); gst_video_rate_flush_prev (videorate, *count > 0, GST_CLOCK_TIME_NONE,
FALSE);
*count += 1; *count += 1;
} }
} else { } else {
while (videorate->next_ts <= prevtime) { while (videorate->next_ts <= prevtime) {
gst_video_rate_flush_prev (videorate, *count > 0, GST_CLOCK_TIME_NONE); gst_video_rate_flush_prev (videorate, *count > 0, GST_CLOCK_TIME_NONE,
FALSE);
*count += 1; *count += 1;
} }
} }
@ -1582,7 +1588,7 @@ gst_video_rate_transform_ip (GstBaseTransform * trans, GstBuffer * buffer)
* GstBaseTransform can get its reference back. */ * GstBaseTransform can get its reference back. */
if ((r = gst_video_rate_push_buffer (videorate, if ((r = gst_video_rate_push_buffer (videorate,
gst_buffer_ref (buffer), FALSE, gst_buffer_ref (buffer), FALSE,
GST_CLOCK_TIME_NONE)) != GST_FLOW_OK) { GST_CLOCK_TIME_NONE, FALSE)) != GST_FLOW_OK) {
res = r; res = r;
goto done; goto done;
} }
@ -1710,7 +1716,7 @@ gst_video_rate_transform_ip (GstBaseTransform * trans, GstBuffer * buffer)
/* on error the _flush function posted a warning already */ /* on error the _flush function posted a warning already */
if ((r = gst_video_rate_flush_prev (videorate, if ((r = gst_video_rate_flush_prev (videorate,
count > 1, intime)) != GST_FLOW_OK) { count > 1, intime, FALSE)) != GST_FLOW_OK) {
res = r; res = r;
goto done; goto done;
} }