1
0
mirror of https://github.com/FFmpeg/FFmpeg.git synced 2025-01-24 13:56:33 +02:00

ffmpeg: Don't require a known device to pass a frames context to an encoder

The previous code here did not handle passing a frames context when
ffmpeg itself did not know about the device it came from (for example,
because it was created by device derivation inside a filter graph), which
would break encoders requiring that input.  Fix that by checking for HW
frames and device context methods independently, and prefer to use a
frames context method if possible.  At the same time, revert the encoding
additions to the device matching function because the additional
complexity was not relevant to decoding.

Also fixes #8637, which is the same case but with the device creation
hidden in the ad-hoc libmfx setup code.
This commit is contained in:
Mark Thompson 2020-04-28 23:56:42 +01:00
parent 1a9684c08a
commit 706ed34ce7

View File

@ -19,6 +19,7 @@
#include <string.h> #include <string.h>
#include "libavutil/avstring.h" #include "libavutil/avstring.h"
#include "libavutil/pixdesc.h"
#include "libavfilter/buffersink.h" #include "libavfilter/buffersink.h"
#include "ffmpeg.h" #include "ffmpeg.h"
@ -282,10 +283,7 @@ void hw_device_free_all(void)
nb_hw_devices = 0; nb_hw_devices = 0;
} }
static HWDevice *hw_device_match_by_codec(const AVCodec *codec, static HWDevice *hw_device_match_by_codec(const AVCodec *codec)
enum AVPixelFormat format,
int possible_methods,
int *matched_methods)
{ {
const AVCodecHWConfig *config; const AVCodecHWConfig *config;
HWDevice *dev; HWDevice *dev;
@ -294,18 +292,11 @@ static HWDevice *hw_device_match_by_codec(const AVCodec *codec,
config = avcodec_get_hw_config(codec, i); config = avcodec_get_hw_config(codec, i);
if (!config) if (!config)
return NULL; return NULL;
if (format != AV_PIX_FMT_NONE && if (!(config->methods & AV_CODEC_HW_CONFIG_METHOD_HW_DEVICE_CTX))
config->pix_fmt != AV_PIX_FMT_NONE &&
config->pix_fmt != format)
continue;
if (!(config->methods & possible_methods))
continue; continue;
dev = hw_device_get_by_type(config->device_type); dev = hw_device_get_by_type(config->device_type);
if (dev) { if (dev)
if (matched_methods)
*matched_methods = config->methods & possible_methods;
return dev; return dev;
}
} }
} }
@ -351,9 +342,7 @@ int hw_device_setup_for_decode(InputStream *ist)
if (!dev) if (!dev)
err = hw_device_init_from_type(type, NULL, &dev); err = hw_device_init_from_type(type, NULL, &dev);
} else { } else {
dev = hw_device_match_by_codec(ist->dec, AV_PIX_FMT_NONE, dev = hw_device_match_by_codec(ist->dec);
AV_CODEC_HW_CONFIG_METHOD_HW_DEVICE_CTX,
NULL);
if (!dev) { if (!dev) {
// No device for this codec, but not using generic hwaccel // No device for this codec, but not using generic hwaccel
// and therefore may well not need one - ignore. // and therefore may well not need one - ignore.
@ -429,37 +418,57 @@ int hw_device_setup_for_decode(InputStream *ist)
int hw_device_setup_for_encode(OutputStream *ost) int hw_device_setup_for_encode(OutputStream *ost)
{ {
HWDevice *dev; const AVCodecHWConfig *config;
AVBufferRef *frames_ref; HWDevice *dev = NULL;
int methods = AV_CODEC_HW_CONFIG_METHOD_HW_DEVICE_CTX; AVBufferRef *frames_ref = NULL;
int matched_methods; int i;
if (ost->filter) { if (ost->filter) {
frames_ref = av_buffersink_get_hw_frames_ctx(ost->filter->filter); frames_ref = av_buffersink_get_hw_frames_ctx(ost->filter->filter);
if (frames_ref && if (frames_ref &&
((AVHWFramesContext*)frames_ref->data)->format == ((AVHWFramesContext*)frames_ref->data)->format ==
ost->enc_ctx->pix_fmt) ost->enc_ctx->pix_fmt) {
methods |= AV_CODEC_HW_CONFIG_METHOD_HW_FRAMES_CTX; // Matching format, will try to use hw_frames_ctx.
} else {
frames_ref = NULL;
}
} }
dev = hw_device_match_by_codec(ost->enc, ost->enc_ctx->pix_fmt, for (i = 0;; i++) {
methods, &matched_methods); config = avcodec_get_hw_config(ost->enc, i);
if (dev) { if (!config)
if (matched_methods & AV_CODEC_HW_CONFIG_METHOD_HW_DEVICE_CTX) { break;
ost->enc_ctx->hw_device_ctx = av_buffer_ref(dev->device_ref);
if (!ost->enc_ctx->hw_device_ctx) if (frames_ref &&
return AVERROR(ENOMEM); config->methods & AV_CODEC_HW_CONFIG_METHOD_HW_FRAMES_CTX &&
} (config->pix_fmt == AV_PIX_FMT_NONE ||
if (matched_methods & AV_CODEC_HW_CONFIG_METHOD_HW_FRAMES_CTX) { config->pix_fmt == ost->enc_ctx->pix_fmt)) {
av_log(ost->enc_ctx, AV_LOG_VERBOSE, "Using input "
"frames context (format %s) with %s encoder.\n",
av_get_pix_fmt_name(ost->enc_ctx->pix_fmt),
ost->enc->name);
ost->enc_ctx->hw_frames_ctx = av_buffer_ref(frames_ref); ost->enc_ctx->hw_frames_ctx = av_buffer_ref(frames_ref);
if (!ost->enc_ctx->hw_frames_ctx) if (!ost->enc_ctx->hw_frames_ctx)
return AVERROR(ENOMEM); return AVERROR(ENOMEM);
return 0;
} }
return 0;
if (!dev &&
config->methods & AV_CODEC_HW_CONFIG_METHOD_HW_DEVICE_CTX)
dev = hw_device_get_by_type(config->device_type);
}
if (dev) {
av_log(ost->enc_ctx, AV_LOG_VERBOSE, "Using device %s "
"(type %s) with %s encoder.\n", dev->name,
av_hwdevice_get_type_name(dev->type), ost->enc->name);
ost->enc_ctx->hw_device_ctx = av_buffer_ref(dev->device_ref);
if (!ost->enc_ctx->hw_device_ctx)
return AVERROR(ENOMEM);
} else { } else {
// No device required, or no device available. // No device required, or no device available.
return 0;
} }
return 0;
} }
static int hwaccel_retrieve_data(AVCodecContext *avctx, AVFrame *input) static int hwaccel_retrieve_data(AVCodecContext *avctx, AVFrame *input)