[FFmpeg-devel] [PATCH 3/5] ffmpeg: flush and drain video filters.
Nicolas George
nicolas.george at normalesup.org
Thu Mar 8 14:49:16 CET 2012
When EOF is reached on the input stream, ffmpeg must
ensure that the graph input returns EOF instead of EAGAIN
and request frames once more until exhaustion.
Signed-off-by: Nicolas George <nicolas.george at normalesup.org>
---
ffmpeg.c | 38 ++++++++++++++++++++++++++++----------
1 files changed, 28 insertions(+), 10 deletions(-)
diff --git a/ffmpeg.c b/ffmpeg.c
index 609d112..739fd29 100644
--- a/ffmpeg.c
+++ b/ffmpeg.c
@@ -2082,7 +2082,7 @@ static int transcode_audio(InputStream *ist, AVPacket *pkt, int *got_output)
return ret;
}
-static int transcode_video(InputStream *ist, AVPacket *pkt, int *got_output, int64_t *pkt_pts)
+static int transcode_video(InputStream *ist, AVPacket *pkt, int *got_output, int64_t *pkt_pts, int flush)
{
AVFrame *decoded_frame;
void *buffer_to_free = NULL;
@@ -2105,10 +2105,14 @@ static int transcode_video(InputStream *ist, AVPacket *pkt, int *got_output, int
return ret;
if (!*got_output) {
- /* no picture yet */
+#if CONFIG_AVFILTER
+ if (!flush)
+#endif
return ret;
}
+ if (*got_output) {
+ /* TODO reindent */
best_effort_timestamp= av_opt_ptr(avcodec_get_frame_class(), decoded_frame, "best_effort_timestamp");
if(*best_effort_timestamp != AV_NOPTS_VALUE)
ist->next_pts = ist->pts = decoded_frame->pts = *best_effort_timestamp;
@@ -2116,12 +2120,15 @@ static int transcode_video(InputStream *ist, AVPacket *pkt, int *got_output, int
pkt->size = 0;
pre_process_video_frame(ist, (AVPicture *)decoded_frame, &buffer_to_free);
+ }
#if CONFIG_AVFILTER
frame_sample_aspect= av_opt_ptr(avcodec_get_frame_class(), decoded_frame, "sample_aspect_ratio");
for(i=0;i<nb_output_streams;i++) {
OutputStream *ost = ost = &output_streams[i];
if(check_output_constraints(ist, ost) && ost->encoding_needed){
+ if (*got_output) {
+ /* TODO reindent */
if (!frame_sample_aspect->num)
*frame_sample_aspect = ist->st->sample_aspect_ratio;
decoded_frame->pts = ist->pts;
@@ -2144,6 +2151,9 @@ static int transcode_video(InputStream *ist, AVPacket *pkt, int *got_output, int
av_log(NULL, AV_LOG_FATAL, "Failed to inject frame into filter network\n");
exit_program(1);
}
+ } else {
+ av_vsrc_buffer_set_return_value(ost->input_video_filter, AVERROR_EOF);
+ }
}
}
#endif
@@ -2157,13 +2167,21 @@ static int transcode_video(InputStream *ist, AVPacket *pkt, int *got_output, int
continue;
#if CONFIG_AVFILTER
- while (av_buffersink_poll_frame(ost->output_video_filter)) {
+ while (1) {
AVRational ist_pts_tb = ost->output_video_filter->inputs[0]->time_base;
AVFrame *filtered_frame;
- if (av_buffersink_get_buffer_ref(ost->output_video_filter, &ost->picref, 0) < 0){
- av_log(NULL, AV_LOG_WARNING, "AV Filter told us it has a frame available but failed to output one\n");
- goto cont;
+ ret = av_buffersink_get_buffer_ref(ost->output_video_filter, &ost->picref, 0);
+ if (ret < 0){
+ if (ret != AVERROR(EAGAIN) && ret != AVERROR_EOF) {
+ char errbuf[128];
+ av_strerror(ret, errbuf, sizeof(errbuf));
+ av_log(NULL, AV_LOG_WARNING,
+ "Error filtering video frame: %s\n", errbuf);
+ } else {
+ ret = 0;
+ }
+ break;
}
if (!ist->filtered_frame && !(ist->filtered_frame = avcodec_alloc_frame())) {
ret = AVERROR(ENOMEM);
@@ -2176,7 +2194,6 @@ static int transcode_video(InputStream *ist, AVPacket *pkt, int *got_output, int
if (ost->picref->video && !ost->frame_aspect_ratio)
ost->st->codec->sample_aspect_ratio = ost->picref->video->sample_aspect_ratio;
do_video_out(output_files[ost->file_index].ctx, ost, ist, filtered_frame);
- cont:
avfilter_unref_buffer(ost->picref);
}
#else
@@ -2219,7 +2236,7 @@ static int output_packet(InputStream *ist,
OutputStream *ost_table, int nb_ostreams,
const AVPacket *pkt)
{
- int ret = 0, i;
+ int ret = 0, i, flush;
int got_output;
int64_t pkt_pts = AV_NOPTS_VALUE;
@@ -2230,7 +2247,8 @@ static int output_packet(InputStream *ist,
if (ist->next_pts == AV_NOPTS_VALUE)
ist->next_pts = ist->pts;
- if (pkt == NULL) {
+ flush = pkt == NULL;
+ if (flush) {
/* EOF handling */
av_init_packet(&avpkt);
avpkt.data = NULL;
@@ -2267,7 +2285,7 @@ static int output_packet(InputStream *ist,
ret = transcode_audio (ist, &avpkt, &got_output);
break;
case AVMEDIA_TYPE_VIDEO:
- ret = transcode_video (ist, &avpkt, &got_output, &pkt_pts);
+ ret = transcode_video (ist, &avpkt, &got_output, &pkt_pts, flush);
if (avpkt.duration) {
duration = av_rescale_q(avpkt.duration, ist->st->time_base, AV_TIME_BASE_Q);
} else if(ist->st->codec->time_base.num != 0 && ist->st->codec->time_base.den != 0) {
--
1.7.9.1
More information about the ffmpeg-devel
mailing list