используя ffmpeg hwaccel из C ++ - PullRequest
19 голосов
/ 13 мая 2011

Как мне декодировать файл с аппаратным ускорением с помощью ffmpeg?

Я написал рабочий видеоплеер, который использует ffmpeg. Я проверил поддержку с помощью "av_hwaccel_next" и нашел mpeg2_dxva.

Однако, когда я загружаю файл mpeg2 (как обычно), я не получаю никакого аппаратного ускорения. AVCodecContext->hwaccel и AVCodecContext->hwaccelcontext равны нулю.

Нужно ли где-нибудь передавать флаг, чтобы включить hw-ускорение?

Мне не удалось найти какую-либо информацию по этому поводу, кто-нибудь знает хороший источник?

1 Ответ

23 голосов
/ 13 мая 2011

Начните с чтения документации ffmpeg: https://trac.ffmpeg.org/wiki/HWAccelIntro и получше ответ Как использовать аппаратное ускорение с ffmpeg (и для страницы проверки linux https://wiki.archlinux.org/index.php/Hardware_video_acceleration)

Когдаиспользуя инструмент FFmpeg, HW-ассистированное декодирование включается с помощью опции -hwaccel, которая включает определенный декодер. Каждый декодер может иметь определенные ограничения (например, H.264 декодер может поддерживать только базовый профиль). HW-ассистированное кодированиеактивируется посредством использования определенного кодера (например, h264_nvenc). Фильтрация Обработка с поддержкой HW поддерживается только в нескольких фильтрах. Существует несколько стандартов API аппаратного ускорения, некоторые из которых поддерживаются для некоторыхэкстент от FFmpeg.

Активация hwaccel контролировалась с помощью кода, подобного (бит переформатирован после 2013 https://github.com/FFmpeg/FFmpeg/commit/08303d774132775d49d4ba767092de5d426f089d)

avctx->hwaccel = ff_find_hwaccel(avctx->codec->id, avctx->pix_fmt);

Например, в libavcodec / mpeg12dec.c https://github.com/FFmpeg/FFmpeg/blob/6c7254722ad43712db5686feee8bf75c74d8635b/libavcodec/mpeg12dec.c

avctx->pix_fmt = mpeg_get_pixelformat(avctx);
avctx->hwaccel = ff_find_hwaccel(avctx->codec->id, avctx->pix_fmt);

ff_find_hwaccel проверяет пару кодеков и пикселей в формате изображения и всех доступных hwaccelerator.

AVHWAccel *ff_find_hwaccel(enum CodecID codec_id, enum PixelFormat pix_fmt)
{
    AVHWAccel *hwaccel=NULL;

    while((hwaccel= av_hwaccel_next(hwaccel))){
        if (   hwaccel->id      == codec_id
            && hwaccel->pix_fmt == pix_fmt)
            return hwaccel;
    }
    return NULL;
}

Например, dxva2 (https://en.wikipedia.org/wiki/DirectX_Video_Acceleration) имеет:

AVHWAccel mpeg2_dxva2_hwaccel = {
    .name           = "mpeg2_dxva2",
    .type           = AVMEDIA_TYPE_VIDEO,
    .id             = CODEC_ID_MPEG2VIDEO,
    .pix_fmt        = PIX_FMT_DXVA2_VLD,
    .capabilities   = 0,
    .start_frame    = start_frame,
    .decode_slice   = decode_slice,
    .end_frame      = end_frame,
    .priv_data_size = sizeof(struct dxva2_picture_context),
};

и libavutil/pixfmt.h перечисляет все поддерживаемые hw-декодеры / ускорители по их форматам пикселей https://ffmpeg.org/doxygen/3.2/pixfmt_8h.html

AV_PIX_FMT_XVMC_MPEG2_MC    - XVideo Motion Acceleration via common packet passing.
AV_PIX_FMT_XVMC_MPEG2_IDCT  - undocumented
AV_PIX_FMT_XVMC         - undocumented
AV_PIX_FMT_VDPAU_H264   - H.264 HW decoding with VDPAU, data[0] contains a vdpau_render_state struct which contains the bitstream of the slices as well as various fields extracted from headers.
AV_PIX_FMT_VDPAU_MPEG1  - MPEG-1 HW decoding with VDPAU, data[0] contains a vdpau_render_state struct which contains the bitstream of the slices as well as various fields extracted from headers.
AV_PIX_FMT_VDPAU_MPEG2  - MPEG-2 HW decoding with VDPAU, data[0] contains a vdpau_render_state struct which contains the bitstream of the slices as well as various fields extracted from headers.
AV_PIX_FMT_VDPAU_WMV3   - WMV3 HW decoding with VDPAU, data[0] contains a vdpau_render_state struct which contains the bitstream of the slices as well as various fields extracted from headers.
AV_PIX_FMT_VDPAU_VC1    - VC-1 HW decoding with VDPAU, data[0] contains a vdpau_render_state struct which contains the bitstream of the slices as well as various fields extracted from headers. 
AV_PIX_FMT_VAAPI_MOCO   - HW acceleration through VA API at motion compensation entry-point, Picture.data[3] contains a vaapi_render_state struct which contains macroblocks as well as various fields extracted from headers.
AV_PIX_FMT_VAAPI_IDCT   - HW acceleration through VA API at IDCT entry-point, Picture.data[3] contains a vaapi_render_state struct which contains fields extracted from headers.
AV_PIX_FMT_VAAPI_VLD    - HW decoding through VA API, Picture.data[3] contains a VASurfaceID. 
AV_PIX_FMT_VDPAU_MPEG4  - MPEG-4 HW decoding with VDPAU, data[0] contains a vdpau_render_state struct which contains the bitstream of the slices as well as various fields extracted from headers.
AV_PIX_FMT_DXVA2_VLD    - HW decoding through DXVA2, Picture.data[3] contains a LPDIRECT3DSURFACE9 pointer. 
AV_PIX_FMT_VDPAU        - HW acceleration through VDPAU, Picture.data[3] contains a VdpVideoSurface. 
AV_PIX_FMT_VDA          - HW acceleration through VDA, data[3] contains a CVPixelBufferRef. 
AV_PIX_FMT_QSV          - HW acceleration through QSV, data[3] contains a pointer to the mfxFrameSurface1 structure.
AV_PIX_FMT_MMAL         - HW acceleration though MMAL, data[3] contains a pointer to the MMAL_BUFFER_HEADER_T structure.
AV_PIX_FMT_D3D11VA_VLD  - HW decoding through Direct3D11, Picture.data[3] contains a ID3D11VideoDecoderOutputView pointer.
AV_PIX_FMT_CUDA         - HW acceleration through CUDA. data[i] contain CUdeviceptr pointers exactly as for system memory frames. 

Фактический выбор форматов пикселей в функции, вызываемой до ff_find_hwaccel, static enum PixelFormat mpeg_get_pixelformat(AVCodecContext *avctx) из libavcodec/mpeg12dec.c для mpeg1 / 2.В более ранних версиях ffmpeg / libavcodec он проверяет avctx->xvmc_acceleration и / или avctx->codec->capabilities&CODEC_CAP_HWACCEL_VDPAU или вызывает avctx->get_format(avctx,ff_hwaccel_pixfmt_list_420); для включения hw-декодирования в некоторых случаях.

В последней версии (2017) он и несколько соседних функций выполняют выбор hw-кодера https://github.com/FFmpeg/FFmpeg/blob/aff8cf18cb0b1fa4f2e3d163c3da2f25aa6d1906/libavcodec/mpeg12dec.c#L1189.

В основном: аппаратный декодер и его API (устаревший XVMC * 1050)*, VDPAU , VA API , MS DXVA или MS Direct3D11 или videotoolbox) должны быть включены в вашей сборке ffmpeg и поддерживается вашим оборудованием и его драйверами / библиотеками (многие из них являются проприетарными и должны быть загружены отдельно).Иногда -hwaccel опция должна быть указана для ffmpeg или для загрузки плагина.В linux вы можете использовать команды vainfo и vdpauinfo для проверки доступности и поддерживаемых профилей с наиболее популярными стандартными API-интерфейсами декодирования видео.

Входной файл (для mpeg1 / 2) должен быть не Grayscale, он должен иметьs->chroma_format меньше 2 (4: 2: 0 Подвыбор цветности , что обычно для ISO / IEC MPEG и ITU-T VCEG H.26x; но не для некоторых MPEG-4 Part 2 и не длявысокая 4: 4: 4 варианта H.264 / MPEG-4 AVC).

static const enum AVPixelFormat mpeg2_hwaccel_pixfmt_list_420[] = {
#if CONFIG_MPEG2_XVMC_HWACCEL
    AV_PIX_FMT_XVMC,
#endif
#if CONFIG_MPEG_VDPAU_DECODER && FF_API_VDPAU
    AV_PIX_FMT_VDPAU_MPEG2,
#endif
#if CONFIG_MPEG2_VDPAU_HWACCEL
    AV_PIX_FMT_VDPAU,
#endif
#if CONFIG_MPEG2_DXVA2_HWACCEL
    AV_PIX_FMT_DXVA2_VLD,
#endif
#if CONFIG_MPEG2_D3D11VA_HWACCEL
    AV_PIX_FMT_D3D11VA_VLD,
#endif
#if CONFIG_MPEG2_VAAPI_HWACCEL
    AV_PIX_FMT_VAAPI,
#endif
#if CONFIG_MPEG2_VIDEOTOOLBOX_HWACCEL
    AV_PIX_FMT_VIDEOTOOLBOX,
#endif
    AV_PIX_FMT_YUV420P,
    AV_PIX_FMT_NONE
};

static const enum AVPixelFormat mpeg12_pixfmt_list_422[] = {
    AV_PIX_FMT_YUV422P,
    AV_PIX_FMT_NONE
};

static const enum AVPixelFormat mpeg12_pixfmt_list_444[] = {
    AV_PIX_FMT_YUV444P,
    AV_PIX_FMT_NONE
};

#if FF_API_VDPAU
static inline int uses_vdpau(AVCodecContext *avctx) {
    return avctx->pix_fmt == AV_PIX_FMT_VDPAU_MPEG1 || avctx->pix_fmt == AV_PIX_FMT_VDPAU_MPEG2;
}
#endif

static enum AVPixelFormat mpeg_get_pixelformat(AVCodecContext *avctx)
{
    Mpeg1Context *s1  = avctx->priv_data;
    MpegEncContext *s = &s1->mpeg_enc_ctx;
    const enum AVPixelFormat *pix_fmts;

    if (CONFIG_GRAY && (avctx->flags & AV_CODEC_FLAG_GRAY))
        return AV_PIX_FMT_GRAY8;

    if (s->chroma_format < 2)
        pix_fmts = avctx->codec_id == AV_CODEC_ID_MPEG1VIDEO ?
                                mpeg1_hwaccel_pixfmt_list_420 :
                                mpeg2_hwaccel_pixfmt_list_420;
    else if (s->chroma_format == 2)
        pix_fmts = mpeg12_pixfmt_list_422;
    else
        pix_fmts = mpeg12_pixfmt_list_444;

    return ff_thread_get_format(avctx, pix_fmts);
}

static void setup_hwaccel_for_pixfmt(AVCodecContext *avctx)
{
    // until then pix_fmt may be changed right after codec init
    if (avctx->hwaccel
#if FF_API_VDPAU
        || uses_vdpau(avctx)
#endif
        )
        if (avctx->idct_algo == FF_IDCT_AUTO)
            avctx->idct_algo = FF_IDCT_SIMPLE;

    if (avctx->hwaccel && avctx->pix_fmt == AV_PIX_FMT_XVMC) {
        Mpeg1Context *s1 = avctx->priv_data;
        MpegEncContext *s = &s1->mpeg_enc_ctx;

        s->pack_pblocks = 1;
#if FF_API_XVMC
FF_DISABLE_DEPRECATION_WARNINGS
        avctx->xvmc_acceleration = 2;
FF_ENABLE_DEPRECATION_WARNINGS
#endif /* FF_API_XVMC */
    }
}
...