1
0
mirror of https://github.com/mpv-player/mpv synced 2025-01-14 02:51:26 +00:00

af_lavfi, vf_lavfi: work around recent libavfilter EOF bug

Looks quite like a bug. If you have a filter chain with only the
dynaudnorm filter, and send call av_buffersrc_add_frame(s, NULL), then
subsequent av_buffersink_get_frame() calls will return EAGAIN instead of
EOF.

This was apparently caused by a recent change in FFmpeg.

Some other circumstances (which I didn't fully analyze and which is due
to the playloop's absurd temporary-EOF behavior on seeks) then led the
decoder loop to send data again, but since libavfilter was stuck in the
EOF state now, it could never recover. It kept sending new input (due to
missing output), until the demuxer refused to return more audio packets.
Each time a filter error was printed.

Fortunately, it's pretty easy to workaround. We just mark the p->eof
flag as we send an EOF frame to libavfilter. The p->eof flag is used
only to recover from temporary EOF: it resets the filter if new data is
available again. We don't care much about av_buffersink_get_frame()
returning a broken EAGAIN state in this situation and essentially ignore
it, meaning if we get EAGAIN after sending EOF, we assume effectively
that EOF was fully reached.
This commit is contained in:
wm4 2017-01-02 18:12:51 +01:00
parent 8e41f314f1
commit 43386a7c92
2 changed files with 12 additions and 0 deletions

View File

@ -266,6 +266,12 @@ static int filter_frame(struct af_instance *af, struct mp_audio *data)
if (!p->graph) if (!p->graph)
goto error; goto error;
if (!data) {
if (p->eof)
return 0;
p->eof = true;
}
if (data) { if (data) {
frame = mp_audio_to_avframe_and_unref(data); frame = mp_audio_to_avframe_and_unref(data);
data = NULL; data = NULL;

View File

@ -298,6 +298,12 @@ static int filter_ext(struct vf_instance *vf, struct mp_image *mpi)
if (!p->graph) if (!p->graph)
return -1; return -1;
if (!mpi) {
if (p->eof)
return 0;
p->eof = true;
}
AVFrame *frame = mp_to_av(vf, mpi); AVFrame *frame = mp_to_av(vf, mpi);
int r = av_buffersrc_add_frame(p->in, frame) < 0 ? -1 : 0; int r = av_buffersrc_add_frame(p->in, frame) < 0 ? -1 : 0;
av_frame_free(&frame); av_frame_free(&frame);