Rewrite FFmpeg decoder selection logic to dynamically discover compatible decoders

This avoids having to maintain a hardcoded list of all known H.264/HEVC/AV1 decoders
and ensures new/unknown decoders will just work automatically (assuming we have
a renderer that can process their output pixel format).
This commit is contained in:
Cameron Gutman 2023-07-15 15:12:58 -05:00
parent fc5e1ed5d8
commit cd4a6f3e17
2 changed files with 144 additions and 124 deletions

View file

@ -47,12 +47,8 @@
#define FAILED_DECODES_RESET_THRESHOLD 20 #define FAILED_DECODES_RESET_THRESHOLD 20
typedef struct { static const QMap<QString, int> k_NonHwaccelCodecInfo = {
const char* codec; // H.264
int capabilities;
} codec_info_t;
static const QList<codec_info_t> k_NonHwaccelH264Codecs = {
#ifdef HAVE_MMAL #ifdef HAVE_MMAL
{"h264_mmal", 0}, {"h264_mmal", 0},
#endif #endif
@ -68,17 +64,15 @@ static const QList<codec_info_t> k_NonHwaccelH264Codecs = {
{"h264_v4l2m2m", 0}, {"h264_v4l2m2m", 0},
#endif #endif
{"h264_omx", 0}, {"h264_omx", 0},
};
static const QList<codec_info_t> k_NonHwaccelHEVCCodecs = { // HEVC
{"hevc_rkmpp", 0}, {"hevc_rkmpp", 0},
{"hevc_nvv4l2", CAPABILITY_REFERENCE_FRAME_INVALIDATION_HEVC}, {"hevc_nvv4l2", CAPABILITY_REFERENCE_FRAME_INVALIDATION_HEVC},
{"hevc_nvmpi", 0}, {"hevc_nvmpi", 0},
{"hevc_v4l2m2m", 0}, {"hevc_v4l2m2m", 0},
{"hevc_omx", 0}, {"hevc_omx", 0},
};
static const QList<codec_info_t> k_NonHwaccelAV1Codecs = { // AV1
}; };
bool FFmpegVideoDecoder::isHardwareAccelerated() bool FFmpegVideoDecoder::isHardwareAccelerated()
@ -131,26 +125,11 @@ int FFmpegVideoDecoder::getDecoderCapabilities()
// We have a non-hwaccel hardware decoder. This will always // We have a non-hwaccel hardware decoder. This will always
// be using SDLRenderer/DrmRenderer so we will pick decoder // be using SDLRenderer/DrmRenderer so we will pick decoder
// capabilities based on the decoder name. // capabilities based on the decoder name.
for (const codec_info_t& codecInfo : k_NonHwaccelH264Codecs) { capabilities = k_NonHwaccelCodecInfo.value(m_VideoDecoderCtx->codec->name, 0);
if (strcmp(m_VideoDecoderCtx->codec->name, codecInfo.codec) == 0) { SDL_LogInfo(SDL_LOG_CATEGORY_APPLICATION,
capabilities = codecInfo.capabilities; "Using capabilities table for decoder: %s -> %d",
SDL_LogInfo(SDL_LOG_CATEGORY_APPLICATION, m_VideoDecoderCtx->codec->name,
"Found capabilities for non-hwaccel H.264 decoder: %s -> %d", capabilities);
m_VideoDecoderCtx->codec->name,
capabilities);
break;
}
}
for (const codec_info_t& codecInfo : k_NonHwaccelHEVCCodecs) {
if (strcmp(m_VideoDecoderCtx->codec->name, codecInfo.codec) == 0) {
capabilities = codecInfo.capabilities;
SDL_LogInfo(SDL_LOG_CATEGORY_APPLICATION,
"Found capabilities for non-hwaccel HEVC decoder: %s -> %d",
m_VideoDecoderCtx->codec->name,
capabilities);
break;
}
}
} }
} }
@ -879,26 +858,28 @@ bool FFmpegVideoDecoder::tryInitializeRenderer(const AVCodec* decoder,
} \ } \
} }
bool FFmpegVideoDecoder::tryInitializeRendererForDecoderByName(const char *decoderName, bool FFmpegVideoDecoder::tryInitializeRendererForUnknownDecoder(const AVCodec* decoder,
PDECODER_PARAMETERS params) PDECODER_PARAMETERS params,
bool tryHwAccel)
{ {
const AVCodec* decoder = avcodec_find_decoder_by_name(decoderName); if (!decoder) {
if (decoder == nullptr) {
return false; return false;
} }
// This might be a hwaccel decoder, so try any hw configs first if (tryHwAccel) {
for (int i = 0;; i++) { // This might be a hwaccel decoder, so try any hw configs first
const AVCodecHWConfig *config = avcodec_get_hw_config(decoder, i); for (int i = 0;; i++) {
if (!config) { const AVCodecHWConfig *config = avcodec_get_hw_config(decoder, i);
// No remaining hwaccel options if (!config) {
break; // No remaining hwaccel options
} break;
}
// Initialize the hardware codec and submit a test frame if the renderer needs it // Initialize the hardware codec and submit a test frame if the renderer needs it
if (tryInitializeRenderer(decoder, params, config, if (tryInitializeRenderer(decoder, params, config,
[config]() -> IFFmpegRenderer* { return createHwAccelRenderer(config, 0); })) { [config]() -> IFFmpegRenderer* { return createHwAccelRenderer(config, 0); })) {
return true; return true;
}
} }
} }
@ -923,7 +904,7 @@ bool FFmpegVideoDecoder::tryInitializeRendererForDecoderByName(const char *decod
#ifdef HAVE_MMAL #ifdef HAVE_MMAL
// HACK: Avoid using YUV420P on h264_mmal. It can cause a deadlock inside the MMAL libraries. // HACK: Avoid using YUV420P on h264_mmal. It can cause a deadlock inside the MMAL libraries.
// Even if it didn't completely deadlock us, the performance would likely be atrocious. // Even if it didn't completely deadlock us, the performance would likely be atrocious.
if (strcmp(decoderName, "h264_mmal") == 0) { if (strcmp(decoder->name, "h264_mmal") == 0) {
for (int i = 0; decoder->pix_fmts[i] != AV_PIX_FMT_NONE; i++) { for (int i = 0; decoder->pix_fmts[i] != AV_PIX_FMT_NONE; i++) {
TRY_PREFERRED_PIXEL_FORMAT(MmalRenderer); TRY_PREFERRED_PIXEL_FORMAT(MmalRenderer);
} }
@ -990,7 +971,7 @@ bool FFmpegVideoDecoder::initialize(PDECODER_PARAMETERS params)
QString h264DecoderHint = qgetenv("H264_DECODER_HINT"); QString h264DecoderHint = qgetenv("H264_DECODER_HINT");
if (!h264DecoderHint.isEmpty() && (params->videoFormat & VIDEO_FORMAT_MASK_H264)) { if (!h264DecoderHint.isEmpty() && (params->videoFormat & VIDEO_FORMAT_MASK_H264)) {
QByteArray decoderString = h264DecoderHint.toLocal8Bit(); QByteArray decoderString = h264DecoderHint.toLocal8Bit();
if (tryInitializeRendererForDecoderByName(decoderString.constData(), params)) { if (tryInitializeRendererForUnknownDecoder(avcodec_find_decoder_by_name(decoderString.constData()), params, true)) {
SDL_LogWarn(SDL_LOG_CATEGORY_APPLICATION, SDL_LogWarn(SDL_LOG_CATEGORY_APPLICATION,
"Using custom H.264 decoder (H264_DECODER_HINT): %s", "Using custom H.264 decoder (H264_DECODER_HINT): %s",
decoderString.constData()); decoderString.constData());
@ -1007,7 +988,7 @@ bool FFmpegVideoDecoder::initialize(PDECODER_PARAMETERS params)
QString hevcDecoderHint = qgetenv("HEVC_DECODER_HINT"); QString hevcDecoderHint = qgetenv("HEVC_DECODER_HINT");
if (!hevcDecoderHint.isEmpty() && (params->videoFormat & VIDEO_FORMAT_MASK_H265)) { if (!hevcDecoderHint.isEmpty() && (params->videoFormat & VIDEO_FORMAT_MASK_H265)) {
QByteArray decoderString = hevcDecoderHint.toLocal8Bit(); QByteArray decoderString = hevcDecoderHint.toLocal8Bit();
if (tryInitializeRendererForDecoderByName(decoderString.constData(), params)) { if (tryInitializeRendererForUnknownDecoder(avcodec_find_decoder_by_name(decoderString.constData()), params, true)) {
SDL_LogWarn(SDL_LOG_CATEGORY_APPLICATION, SDL_LogWarn(SDL_LOG_CATEGORY_APPLICATION,
"Using custom HEVC decoder (HEVC_DECODER_HINT): %s", "Using custom HEVC decoder (HEVC_DECODER_HINT): %s",
decoderString.constData()); decoderString.constData());
@ -1024,7 +1005,7 @@ bool FFmpegVideoDecoder::initialize(PDECODER_PARAMETERS params)
QString av1DecoderHint = qgetenv("AV1_DECODER_HINT"); QString av1DecoderHint = qgetenv("AV1_DECODER_HINT");
if (!av1DecoderHint.isEmpty() && (params->videoFormat & VIDEO_FORMAT_MASK_AV1)) { if (!av1DecoderHint.isEmpty() && (params->videoFormat & VIDEO_FORMAT_MASK_AV1)) {
QByteArray decoderString = av1DecoderHint.toLocal8Bit(); QByteArray decoderString = av1DecoderHint.toLocal8Bit();
if (tryInitializeRendererForDecoderByName(decoderString.constData(), params)) { if (tryInitializeRendererForUnknownDecoder(avcodec_find_decoder_by_name(decoderString.constData()), params, true)) {
SDL_LogWarn(SDL_LOG_CATEGORY_APPLICATION, SDL_LogWarn(SDL_LOG_CATEGORY_APPLICATION,
"Using custom AV1 decoder (AV1_DECODER_HINT): %s", "Using custom AV1 decoder (AV1_DECODER_HINT): %s",
decoderString.constData()); decoderString.constData());
@ -1039,98 +1020,136 @@ bool FFmpegVideoDecoder::initialize(PDECODER_PARAMETERS params)
} }
const AVCodec* decoder; const AVCodec* decoder;
void* codecIterator;
if (params->videoFormat & VIDEO_FORMAT_MASK_H264) {
decoder = avcodec_find_decoder(AV_CODEC_ID_H264);
}
else if (params->videoFormat & VIDEO_FORMAT_MASK_H265) {
decoder = avcodec_find_decoder(AV_CODEC_ID_HEVC);
}
else if (params->videoFormat & VIDEO_FORMAT_MASK_AV1) {
decoder = avcodec_find_decoder(AV_CODEC_ID_AV1);
}
else {
Q_ASSERT(false);
decoder = nullptr;
}
if (!decoder) {
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
"Unable to find decoder for format: %x",
params->videoFormat);
return false;
}
// Look for a hardware decoder first unless software-only // Look for a hardware decoder first unless software-only
if (params->vds != StreamingPreferences::VDS_FORCE_SOFTWARE) { if (params->vds != StreamingPreferences::VDS_FORCE_SOFTWARE) {
// Look for the first matching hwaccel hardware decoder (pass 0) // Iterate through tier-1 hwaccel decoders
for (int i = 0;; i++) { codecIterator = NULL;
const AVCodecHWConfig *config = avcodec_get_hw_config(decoder, i); while ((decoder = av_codec_iterate(&codecIterator))) {
if (!config) { // Skip codecs that aren't decoders
// No remaining hwaccel options if (!av_codec_is_decoder(decoder)) {
break; continue;
} }
// Initialize the hardware codec and submit a test frame if the renderer needs it // Skip decoders that don't match our codec
if (tryInitializeRenderer(decoder, params, config, if (((params->videoFormat & VIDEO_FORMAT_MASK_H264) && decoder->id != AV_CODEC_ID_H264) ||
[config]() -> IFFmpegRenderer* { return createHwAccelRenderer(config, 0); })) { ((params->videoFormat & VIDEO_FORMAT_MASK_H265) && decoder->id != AV_CODEC_ID_HEVC) ||
((params->videoFormat & VIDEO_FORMAT_MASK_AV1) && decoder->id != AV_CODEC_ID_AV1)) {
continue;
}
// Look for the first matching hwaccel hardware decoder (pass 0)
for (int i = 0;; i++) {
const AVCodecHWConfig *config = avcodec_get_hw_config(decoder, i);
if (!config) {
// No remaining hwaccel options
break;
}
// Initialize the hardware codec and submit a test frame if the renderer needs it
if (tryInitializeRenderer(decoder, params, config,
[config]() -> IFFmpegRenderer* { return createHwAccelRenderer(config, 0); })) {
return true;
}
}
}
// Iterate through non-hwaccel hardware decoders
codecIterator = NULL;
while ((decoder = av_codec_iterate(&codecIterator))) {
// Skip codecs that aren't decoders
if (!av_codec_is_decoder(decoder)) {
continue;
}
// Skip decoders that don't match our codec
if (((params->videoFormat & VIDEO_FORMAT_MASK_H264) && decoder->id != AV_CODEC_ID_H264) ||
((params->videoFormat & VIDEO_FORMAT_MASK_H265) && decoder->id != AV_CODEC_ID_HEVC) ||
((params->videoFormat & VIDEO_FORMAT_MASK_AV1) && decoder->id != AV_CODEC_ID_AV1)) {
continue;
}
// Skip hwaccel and software/hybrid decoders
if (avcodec_get_hw_config(decoder, 0) || !(decoder->capabilities & AV_CODEC_CAP_HARDWARE)) {
continue;
}
// Try this decoder
if (tryInitializeRendererForUnknownDecoder(decoder, params, false)) {
return true; return true;
} }
} }
// Continue with special non-hwaccel hardware decoders // Iterate through tier-2 hwaccel decoders
if (params->videoFormat & VIDEO_FORMAT_MASK_H264) { codecIterator = NULL;
for (const codec_info_t& codecInfo : k_NonHwaccelH264Codecs) { while ((decoder = av_codec_iterate(&codecIterator))) {
if (tryInitializeRendererForDecoderByName(codecInfo.codec, params)) { // Skip codecs that aren't decoders
return true; if (!av_codec_is_decoder(decoder)) {
} continue;
}
}
else if (params->videoFormat & VIDEO_FORMAT_MASK_H265) {
for (const codec_info_t& codecInfo : k_NonHwaccelHEVCCodecs) {
if (tryInitializeRendererForDecoderByName(codecInfo.codec, params)) {
return true;
}
}
}
else if (params->videoFormat & VIDEO_FORMAT_MASK_AV1) {
for (const codec_info_t& codecInfo : k_NonHwaccelAV1Codecs) {
if (tryInitializeRendererForDecoderByName(codecInfo.codec, params)) {
return true;
}
}
}
else {
Q_ASSERT(false);
}
// Look for the first matching hwaccel hardware decoder (pass 1)
// This picks up "second-tier" hwaccels like CUDA.
for (int i = 0;; i++) {
const AVCodecHWConfig *config = avcodec_get_hw_config(decoder, i);
if (!config) {
// No remaining hwaccel options
break;
} }
// Initialize the hardware codec and submit a test frame if the renderer needs it // Skip decoders that don't match our codec
if (tryInitializeRenderer(decoder, params, config, if (((params->videoFormat & VIDEO_FORMAT_MASK_H264) && decoder->id != AV_CODEC_ID_H264) ||
[config]() -> IFFmpegRenderer* { return createHwAccelRenderer(config, 1); })) { ((params->videoFormat & VIDEO_FORMAT_MASK_H265) && decoder->id != AV_CODEC_ID_HEVC) ||
return true; ((params->videoFormat & VIDEO_FORMAT_MASK_AV1) && decoder->id != AV_CODEC_ID_AV1)) {
continue;
}
// Look for the first matching hwaccel hardware decoder (pass 1)
// This picks up "second-tier" hwaccels like CUDA.
for (int i = 0;; i++) {
const AVCodecHWConfig *config = avcodec_get_hw_config(decoder, i);
if (!config) {
// No remaining hwaccel options
break;
}
// Initialize the hardware codec and submit a test frame if the renderer needs it
if (tryInitializeRenderer(decoder, params, config,
[config]() -> IFFmpegRenderer* { return createHwAccelRenderer(config, 1); })) {
return true;
}
} }
} }
} }
// Fallback to software if no matching hardware decoder was found // Iterate through all software decoders if allowed
// and if software fallback is allowed
if (params->vds != StreamingPreferences::VDS_FORCE_HARDWARE) { if (params->vds != StreamingPreferences::VDS_FORCE_HARDWARE) {
if (tryInitializeRenderer(decoder, params, nullptr, codecIterator = NULL;
[]() -> IFFmpegRenderer* { return new SdlRenderer(); })) { while ((decoder = av_codec_iterate(&codecIterator))) {
return true; // Skip codecs that aren't decoders
if (!av_codec_is_decoder(decoder)) {
continue;
}
// Skip decoders that don't match our codec
if (((params->videoFormat & VIDEO_FORMAT_MASK_H264) && decoder->id != AV_CODEC_ID_H264) ||
((params->videoFormat & VIDEO_FORMAT_MASK_H265) && decoder->id != AV_CODEC_ID_HEVC) ||
((params->videoFormat & VIDEO_FORMAT_MASK_AV1) && decoder->id != AV_CODEC_ID_AV1)) {
continue;
}
// Skip hardware decoders
//
// NB: We can't skip hwaccel decoders here because they can be both
// hardware and software depending on whether an hwaccel is supplied.
// Instead, we tell tryInitializeRendererForUnknownDecoder() not to
// try hwaccel for this decoder.
if (decoder->capabilities & AV_CODEC_CAP_HARDWARE) {
continue;
}
// Try this decoder without hwaccel
if (tryInitializeRendererForUnknownDecoder(decoder, params, false)) {
return true;
}
} }
} }
// No decoder worked SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
"Unable to find working decoder for format: %x",
params->videoFormat);
return false; return false;
} }

View file

@ -40,8 +40,9 @@ private:
bool createFrontendRenderer(PDECODER_PARAMETERS params, bool useAlternateFrontend); bool createFrontendRenderer(PDECODER_PARAMETERS params, bool useAlternateFrontend);
bool tryInitializeRendererForDecoderByName(const char* decoderName, bool tryInitializeRendererForUnknownDecoder(const AVCodec* decoder,
PDECODER_PARAMETERS params); PDECODER_PARAMETERS params,
bool tryHwAccel);
bool tryInitializeRenderer(const AVCodec* decoder, bool tryInitializeRenderer(const AVCodec* decoder,
PDECODER_PARAMETERS params, PDECODER_PARAMETERS params,