Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Update FFmpeg to version 7.1 #950

Closed
wants to merge 80 commits into from
Closed

Conversation

ulmus-scott
Copy link
Contributor

Checklist

Commits

The commits in order are:

FFmpeg reversions
#923

Eliminate usage of internal FFmpeg headers
#931

mpegts-mythtv harmonization
#927

Changes to make MythTV compile with FFmpeg 7.1:

6716364c61 cmake/configure: remove references to unsupported crystalhd
d271e34a86 constify AVIOContext::write_packet buf parameter
1304aee821 constify AVIOContext::write_packet buf parameter (plugins)
84c178159a Use AVFrame::best_effort_timestamp

Update to FFmpeg 7.1:

976ab94b53 update FFmpeg to version 7.1
f8928e842c add ffmpeg_update_instructions.md

Cleanup after update:

807ef793aa remove support for LIBAVFORMAT_VERSION_MAJOR < 61 (plugins)
6c4c2684e2 remove support for LIBAVFORMAT_VERSION_MAJOR < 61
5098f349fb fix TestMusicMetadata::test_mp4()

Silence new FFmpeg deprecation warnings:

383aca2ea1 libmythtv/io/mythavformatwriter.cpp: silence AVCodec deprecation warnings
d43d574a59 replace deprecated av_stream_get_side_data()
93a9bc4d3c replace deprecated av_fft API with av_tx
ae3825083a replace deprecated AVFrame::interlaced_frame, top_field_first, and key_frame
7849f3029b mytharchivehelper.cpp: replace deprecated AVFrame::key_frame
e4e850b5e7 replace deprecated av_fft API with av_tx (mythmusic)

Other cleanup commits made while investigating:

dc03e8d08b remove DecoderBase::m_getRawFrames and m_getRawVideo
8f08f60856 remove unused no-op functions from DecoderBase
ca0e299060 mythaverror: add new functions returning std::string
b22e19e8b9 split libmyth/mythavframe.h from libmyth/mythaverror.h
7fe6c88070 create AvFormatDecoder::do_av_seek()
31af3ce5e2 remove unused GetAVTimeBaseQ()
552e90b564 MythMediaBuffer::WaitForAvail(): simplify return logic

Testing

@kmdewaal , I have reverted your fix regarding blocking with VDPAU referencing https://trac.ffmpeg.org/ticket/9532. Could you test if it is still an issue and update the FFmpeg trac as needed?

I have tested with my various Video samples and US OTA ATSC recordings and everything appeared to work the same. I do not have any hardware acceleration set up, so I could not test with that.

Testing with hardware acceleration or other source types would be helpful.

Quirks in master I may look into more

AvFormatDecoder::GetFrame() log spam every ~100μs: decoding error End of file if paused near end of file. I can sometimes trigger this with Video files when skipping by 1 second near the end while paused.

The seeking code for Videos is very fragile, I suspect multithreading issues.
Sometimes either seek forward or backward does nothing and gives the following error in the log:
AFD: av_seek_frame(ic, -1, 13449820978, 0) -- error
Some further investigation revealed the error code to be -1, which FFmpeg thinks
is EPERM.

For a short (~16s) audio only file I created, skipping backwards usually causes playback to exit instead of skipping.

@warpme
Copy link
Contributor

warpme commented Oct 24, 2024

Scott,
Per our conversation on ML about ffmpeg5->ffmpeg7 regression on n3450 with 2x vaapi deinterlacer: pls find 2 logs for playback of interlaced h264 recording. ffmpeg5 has working 2x DI while ffmpeg7 reports on osd "2x VAAPI Compensated" but visually i see 1x DI on screen
I was trying to catch diff between logs but failed to see meaningful diff :-(
n3450-ffmpeg7-nok.log.zip
n3450-ffmpeg5-ok.log.zip

@ulmus-scott
Copy link
Contributor Author

@warpme I had made a typo in mythvaapiinterop.cpp which is fixed in the first new commit and hopefully fixes the double rate VAAPI deinterlacing.

I did not see anything of interest in the logs either.

If the new commit doesn't work, try testing at 93a9bc4 and, if that still doesn't work, at 84c1781 before the FFmpeg update.

Before merging I want to move some of the final commits earlier and squash some of them together.

@warpme
Copy link
Contributor

warpme commented Oct 26, 2024

Scott,

Current head works perfectly now!
Many thx fixing this.
I will start to use ffmpeg7.1 branch in production on x86 and aarch64 systems to see how stability, hw decoding for various codecs looks like.
I'll put report on dev ML to inform also others how well ffmpeg7.1 works

@ulmus-scott
Copy link
Contributor Author

Rebased onto master with a few commits reordered and squashed.

@bennettpeter
Copy link
Member

I have rebased and merged into branch devel/ffmpeg-resync.

@kmdewaal
Copy link
Contributor

Works OK for me. No regression found.

I do have a few ATSC3.0 recordings and they do play back with a lot of stuttering and complains about the ac4 audio codec:

2024-12-22 21:44:08.963749 I  AFD: Using vaapi for video decoding
2024-12-22 21:44:08.963757 I  AFD: Opened codec 0x22025f40, id(hevc) type(Video)
2024-12-22 21:44:09.030591 W  avcodec_find_decoder fail for 86119
2024-12-22 21:44:09.030620 I  AFD: codec ac4 has 0 channels
2024-12-22 21:44:09.030644 W  avcodec_find_decoder fail for 86119
2024-12-22 21:44:09.030649 E  AFD: Could not find decoder for codec (ac4), ignoring.

With the current master there is not even an attempt to play the audio but the video plays smooth.

@ulmus-scott
Copy link
Contributor Author

@kmdewaal , I have reverted your fix regarding blocking with VDPAU referencing https://trac.ffmpeg.org/ticket/9532. Could you test if it is still an issue and update the FFmpeg trac as needed?

FFmpeg does not currently have a decoder for AC-4, but a raw muxer and demuxer were added so there is now AV_CODEC_ID_AC4, so it now recognizes the stream as AC-4 but can't decode it.

Is master also using vaapi hardware acceleration? Is ffmpeg/7.1 repeatedly trying to find a decoder for AC-4?

@kmdewaal
Copy link
Contributor

@kmdewaal , I have reverted your fix regarding blocking with VDPAU referencing https://trac.ffmpeg.org/ticket/9532. Could you test if it is still an issue and update the FFmpeg trac as needed?

I do not have an Nvidia/VDPAU system running at the moment so I cannot test it right now. Possibly later. I am OK with having the fix reverted.

FFmpeg does not currently have a decoder for AC-4, but a raw muxer and demuxer were added so there is now AV_CODEC_ID_AC4, so it now recognizes the stream as AC-4 but can't decode it.

Yes, I gathered as much. This should in principle make it possible to use audio pass-through for AC4 so that people with an audio system that can decode AC4 can then use ATSC3.0.

Is master also using vaapi hardware acceleration? Is ffmpeg/7.1 repeatedly trying to find a decoder for AC-4?
The fragment log that I posted yesterday, with messages about not finding the ac4 decoder, is repeated continuously, with intervals in the milliseconds.
Playback is done with vaapi hardware acceleration. This does work.

@bennettpeter
Copy link
Member

I have reset branch devel/ffmpeg-resync and merged the updated pull request into devel/ffmpeg-resync.

@ulmus-scott
Copy link
Contributor Author

@kmdewaal , I am not sure why ffmpeg/7.1 would have stuttery video playback or repeatedly try to find a decoder for AC-4.

Could you provide a more complete log with -v playback:debug,audio:debug,libav:debug --logpath=/tmp?

Could you also upload an ATSC 3.0 sample that is stuttering and repeatedly failing to find the (non-existent) decoder?

@bennettpeter
Copy link
Member

Therfe is a sample here, only a few seconds,
https://trac.ffmpeg.org/attachment/ticket/8349/atsc3.ts

I see the same thing that Klaas describes. I am using ffmpeg decoding, I do not have any accelerated video on my laptop, so it is doing ffmpeg decoding.

@ulmus-scott
Copy link
Contributor Author

AvFormatDecoder::SeekReset() calls GetFrame() in a loop with a 1 ms delay.
GetFrame() calls StreamChangeCheck() which calls ScanStreams() which calls
MythCodecMap::GetCodecContext() to produce the log message
avcodec_find_decoder fail for 86119 and then prints its own
AFD: Could not find decoder for codec (ac4), ignoring..

However, breaking that loop does not effect the log messages.

Something is calling AutoSelectAudioTrack() repeatedly producing
avformatdecoder.cpp:4204 (AutoSelectAudioTrack) - 0 available audio streams.

I'm using FFmpeg software decoding and I get aborts with that sample:

Handling Aborted
Aborted (core dumped)

gdb shows that the assert at libavutil/imgutils.c:350 is failing, related to copying a video frame.

This resolves my aborts:

diff --git a/mythtv/libs/libmythtv/decoders/avformatdecoder.cpp b/mythtv/libs/libmythtv/decoders/avformatdecoder.cpp
index 5cd8e3cb8d..09d022125a 100644
--- a/mythtv/libs/libmythtv/decoders/avformatdecoder.cpp
+++ b/mythtv/libs/libmythtv/decoders/avformatdecoder.cpp
@@ -4836,7 +4836,7 @@ bool AvFormatDecoder::GetFrame(DecodeType decodetype, bool &Retry)
                          avcodec_get_name(curstream->codecpar->codec_id))
                         .arg(curstream->codecpar->codec_id));
                 // Process Stream Change in case we have no audio
-                if (codec_type == AVMEDIA_TYPE_AUDIO && !m_audio->HasAudioIn())
+                if (false && codec_type == AVMEDIA_TYPE_AUDIO && !m_audio->HasAudioIn())
                     m_streamsChanged = true;
             }
             av_packet_unref(pkt);

However, I still get a long list of quickly repeating


2024-12-23 18:46:12.697309 W [308931/309011] Decoder mythavutil.cpp:310 (GetCodecContext) - avcodec_find_decoder fail for 86119
2024-12-23 18:46:12.697312 E [308931/309011] Decoder avformatdecoder.cpp:4832 (GetFrame) - AFD: No codec for stream index 1, type(Audio) id(ac4:86119)

which then stops and then I get a long list of

avformatdecoder.cpp:4204 (AutoSelectAudioTrack) - 0 available audio streams

and then the two mix together.

Video playback is not stuttery. I'll have to look into this more.

@ulmus-scott
Copy link
Contributor Author

mythfrontend -v playback:debug,audio:debug --logpath=/tmp now only shows one call to ScanStreams() and thus one set of log messages about AC-4 since AV_CODEC_ID_AC4 is now ignored.

@bennettpeter
Copy link
Member

@ulmus-scott

I plan to merge the stuff that is already in devel/ffmpeg-resync next week as long as problems are not reported. Those have been tested and are stable, with the exception of the ATSC3 problem. AC-4 audio does not work with the existing system anywway.

It looks like the new commits are for MytTV not ffmpeg. I will merge them separately after ffmpeg has been merged into master.

@ulmus-scott
Copy link
Contributor Author

@bennettpeter OK, this was probably not the pull request to do that new clean up, but I could create a version of d35d23c that does not depend on the rest of the new changes if desired.

@bennettpeter
Copy link
Member

I am ready to merge the pull request, but now there are conflicts due to changes applied in the interim. I am unsure of how to fix them. I get a conflict in libs/libmyth/audio/freesurround_decoder.cpp . Please rebase on to master so that I can merge.

This reverts commit 049fabc.

Reduce changes to FFmpeg.
Instead of the internal FFmpeg header compat/cuda/dynlink_loader.h.  This leaves
only one internal FFmpeg header used by MythTV.

Per the comment on win32_dlopen() in FFmpeg/compat/w32dlfcn.h, Windows will now
additionally search the current directory for nvcuda.dll and nvcuvid.dll, which
FFmpeg considers less secure.

This is an atomic block and the order must be preserved:
 #include "libavutil/log.h"
 #define FFNV_LOG_FUNC(logctx, msg, ...) av_log(logctx, AV_LOG_ERROR, msg,  __VA_ARGS__)
 #define FFNV_DEBUG_LOG_FUNC(logctx, msg, ...) av_log(logctx, AV_LOG_DEBUG, msg,  __VA_ARGS__)
 #include <ffnvcodec/dynlink_loader.h>

 #include "libavutil/hwcontext_cuda.h"
must come after
 #include <ffnvcodec/dynlink_cuda.h>
which is also transitively included by #include <ffnvcodec/dynlink_loader.h>
This reverts commit 5f9b27c.

MythTV no longer uses that file, so remove the modification.
…h (part 1)

Since we don't need the full buffer, we could use avio_read_partial() instead.
ulmus-scott and others added 23 commits December 30, 2024 16:23
The first return ensures (available < count) is always true.

If m_ateof is true, then Count = available, which means (available < count) is
false and all subsequent conditionals are false.

Now, (available < count) is only needed for the while loop condition.
This reverts commit db3dfa505a2a5a7e2b6ff898f9c3ad5d74fbf01a.

I am not sure the "optimization" was correct and removing the customization
appears to have no effect of any kind.  My MHEG sample did not trigger the log
message for "All DSM info found".
The code in libavcodec/mpeg12dec.c will never create a cc_type of 2 for SCTE-20
and cc_type is guaranteed to be less than 2 otherwise by the if condition.

AvFormatDecoder::m_invertScteField is now always 0, so remove it.

AvFormatDecoder::m_lastScteField is now set but unused, so remove it.
…captions

see mpeg_decode_a53_cc()

Differences:
FFmpeg does not have the initial two bytes that MythTV added to signal valid and
length.

FFmpeg will only use the first type of embedded CC that it sees, whereas MythTV
would switch between ATSC and SCTE-20 under certain conditions.  If multiple
types were present MythTV would concatenate them.

FFmpeg allows forcing the subtitle format with the cc_format option in mpeg12dec.c.

DVD CC:
FFmpeg does not check p[4]

ATSC A/53:
FFmpeg checks the process_cc_data_flag in p[5].

SCTE-20:
FFmpeg ignores line_offset, which MythTV ensures is 11 for line 21 data.

FFmpeg does not set one_bit or the 4 reserved bits before cc_valid and cc_type
(cc608hdr vs cap[0]), but these are only to avoid emulating start codes and are
not checked, so it doesn't matter.

p[0] == 0x05 && p[1] == 0x02:
Moved to mpeg_decode_a53_cc().

Originally from:
Fixes #2481, by applying patch from kenny at the-b org. This adds dec…
MythTV@49d3294

Referencing:
https://code.mythtv.org/trac/ticket/2481

Modified in:
Simplify DTV CEA-608 handling a bit & add some range checking.
MythTV@be6a2b4

Some tweaks to cc implementation in mpeg12.c: make the C ANSI compati…
MythTV@77cff32

See also:
https://github.com/CCExtractor/ccextractor/blob/master/src/lib_ccx/es_userdata.c

MythTV:
SCTE-20 encapsulates CTA-608, which is where XDS is defined, so I don't know why
the condition was specific to SCTE-20.
This reverts commit 641109a.

David Hampton confirmed on the mailing list that this change is no longer
necessary on FreeBSD 13/14/15.
…Android)'

Neither header is checked by FFmpeg's configure and neither HAVE_EGL_EGL_H
nor HAVE_GLES2_GL2_H is defined.  FFmpeg/configure has not tested those since
MythTV@018be3f
in 2018.

FFmpeg's configure never tested eglGetProcAddress and thus would never define
HAVE_EGLGETPROCADDRESS.

MythTV's configure does check them and (myth)config.h does define HAVE_EGL_EGL_H,
HAVE_GLES2_GL2_H, and HAVE_EGLGETPROCADDRESS; however, FFmpeg does not use
mythconfig.h, using its own config.h instead, probably since the decoupling of
MythTV and FFmpeg's configure in
MythTV@9563786
in 2018.
by merging the code into AvFormatDecoder::RemoveAudioStreams() and replacing
ff_flush_packet_queue() with avformat_flush().

The original code could have accessed out of bounds if the first stream was an
audio stream, since i would have been -1 on the next iteration.  This would also
cause it to recheck previously checked streams when a stream was removed.

By setting the index first, we don't have to iterate twice to renumber the streams.

Slightly more is reset by avformat_flush(), see libavformat/seek.c, compared to
just ff_flush_packet_queue(), but it is a public function and resetting the
extra values is probably better since it more fully resets the AVFormatContext.
Additionally, the AVFormatContext is now only reset once instead of after
removing each stream.
StreamInfo::m_av_substream_index is only used for kTrackTypeAudio.  It is no
longer set to 0 by:
mythdvddecoder.cpp
AvFormatDecoder::ScanStreams() for AVMEDIA_TYPE_SUBTITLE

StreamInfo::m_orig_num_channels is only used by AvFormatDecoder::GetTrackDesc()
to print the number of audio channels for kAudioTypeNormal.
The following no longer set m_orig_num_channels to 0:
mythdvddecoder.cpp
cc708reader.cpp
decoderbase.cpp
avformatdecoder.cpp

StreamInfo::m_easy_reader and m_wide_aspect_ratio were unused but set by:
AvFormatDecoder::ScanATSCCaptionStreams() (incorrectly, setting
m_orig_num_channels = easy_reader and m_easy_reader = wide_aspect_ratio)

AvFormatDecoder::ScanStreams() for AVMEDIA_TYPE_AUDIO (appeared to set m_easy_reader
in an unused attempt to flag the second StreamInfo for dual mono audio)

decoderbase.cpp: I don't know why one of the default CC608 set m_language_index
to 2, but it is not used for closed captions.
If continue is DVD specific and from
MythTV@8bcef59

AVStream::id will never be negative for MPEG-TS or MPEG-PS.
It was a copy of a value from AVCodecParameters and was only used for printing.

If the channel count from FFmpeg is wrong, add this to GetTrackDesc() to get the
value from libdvdnav:

if (m_ringBuffer->IsDVD())
    channels = m_ringBuffer->DVD()->GetNumAudioChannels(m_tracks[kTrackTypeAudio][TrackNo].m_stream_id);
When attempting to upstream the change before, it was recommended to instead
use AV_CHANNEL_ORDER_CUSTOM with two AV_CHAN_FRONT_CENTER channels. See
https://patchwork.ffmpeg.org/project/ffmpeg/patch/[email protected]/

audiotoolboxdec.c will no longer flag dual_mono since it uses
AV_CHANNEL_ORDER_UNSPEC.

SetupAudioStreamSubIndexes() will create or destroy the second StreamInfo for
the second substream, leaving the duplicate StreamInfo from ScanStreams()
untouched, allowing playing both channels as stereo.
st and st->codecpar will already have been dereferenced by then.
Originally from:
MythTV@6aaf617

references:
https://code.mythtv.org/trac/ticket/5978
I could not find the referenced FFmpeg bugs from the comment or MythTV trac
since FFmpeg has changed its bug tracker. However, I did find
https://samples.ffmpeg.org/ffmpeg-bugs/roundup/issue378/Issue378/
which plays back with subtitles fine.

As far as I can tell the change reduces out_size by len + 6 if that section is
at the end of p.  However, that data is all in the buffer, so I don't see how
that would fix anything.  If this does actually fix something a new FFmpeg bug
report should be created with a sample.
libavcodec/codec_desc.c: move to correct location and correct description
libavcodec/codec_id.h: move to correct location in data codecs
libavformat/mpegts-mythtv.c: reuse teletext_descriptor code for VBI_teletext_descriptor

libmythbase/iso639.h: add used include
libmythbase/stringutil.h: add new funtion split_sv()
libmythtv/decoders/avformatdecoder.cpp:
AvFormatDecoder::ScanTeletextCaptions() was specific to AV_CODEC_ID_DVB_TELETEXT,
looking only at the teletext_descriptor in the PMT.

ScanTeletextCaptions() no longer uses the MythTV exported PMT, using what FFmpeg
already exported when it parsed PMTs.  Both AV_CODEC_ID_DVB_TELETEXT and
AV_CODEC_ID_DVB_VBI (if there is EBU teletext data in the VBI data) now add to
the tracks list and multiple streams are now supported.
update_initial_timestamps(): the AVStream* will already have been dereferenced
higher up the call stack.

compute_frame_duration(): AVStream::codecpar should always be non-NULL, see
avformat_new_stream(), and the AVStream* will already have been dereferenced
higher up the call stack.

handle_new_packet(): av_assert0() guarantees that the AVStream* is valid if they
have been added and removed from the AVFormatContext correctly.
…cc_format()

When Closed Captions are discovered, this method is always invoked.
Therefore, use it to set the property instead of repeating the statement.

Signed-off-by: Marth64 <[email protected]>
(cherry picked from commit 11c703be3176174c2ddb0f3893d10f3ed1accb89)
ATSC A/52:2018 Digital Audio Compression (AC-3, E-AC-3), Annex G
defines stream_type 0x87 for E-AC-3 bit streams.

Signed-off-by: Marton Balint <[email protected]>
(cherry picked from commit be784e95ac5cd720fb0da0f13841b0fdf4f90391)
Signed-off-by: Marton Balint <[email protected]>
(cherry picked from commit 0dceced45c52acc773e690a7aa7ff2e3fb8c560b)
Reviewed-by: Marth64 <[email protected]>
Signed-off-by: Marth64 <[email protected]>
(cherry picked from commit 78c4d6c136e10222a0b0ddff639c836f295a9029)
None of the other external libraries have the warnings disabled.

In addition, the MythTV modifications no longer generate any additional warnings.
Originally from:
MythTV@669955c

Signed-off-by: Zhao Zhili <[email protected]>
(cherry picked from commit 9da1d2e66ab1ac9dcfaa290bbea78b2a4900ac0a)
@ulmus-scott
Copy link
Contributor Author

David Hampton changed the order in freesurround_decoder.cpp, but the change after the commit was still correct.

The changes that were not in devel/ffmpeg-resync were removed and will be in a new pull request after this merges.

@bennettpeter
Copy link
Member

bennettpeter commented Dec 31, 2024

I have rebased and merged this pull request.

@ulmus-scott ulmus-scott mentioned this pull request Dec 31, 2024
6 tasks
@ulmus-scott
Copy link
Contributor Author

@bennettpeter The commits that I removed from this are now at #1009

@ulmus-scott ulmus-scott deleted the ffmpeg/7.1 branch December 31, 2024 19:08
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants