2015-03-13 10:20:31 +01:00
/*
2018-08-30 08:56:18 +02:00
* Intel MediaSDK QSV based H .264 encoder
2015-03-13 10:20:31 +01:00
*
* copyright ( c ) 2013 Yukinori Yamazoe
*
2015-03-28 01:33:27 +01:00
* This file is part of FFmpeg .
2015-03-13 10:20:31 +01:00
*
2015-03-28 01:33:27 +01:00
* FFmpeg is free software ; you can redistribute it and / or
2015-03-13 10:20:31 +01:00
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation ; either
* version 2.1 of the License , or ( at your option ) any later version .
*
2015-03-28 01:33:27 +01:00
* FFmpeg is distributed in the hope that it will be useful ,
2015-03-13 10:20:31 +01:00
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
* Lesser General Public License for more details .
*
* You should have received a copy of the GNU Lesser General Public
2015-03-28 01:33:27 +01:00
* License along with FFmpeg ; if not , write to the Free Software
2015-03-13 10:20:31 +01:00
* Foundation , Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA
*/
# include <stdint.h>
# include <sys/types.h>
2020-09-08 11:17:27 +08:00
# include <mfxvideo.h>
2015-03-13 10:20:31 +01:00
2016-05-18 10:22:24 +02:00
# include "libavutil/common.h"
2015-03-13 10:20:31 +01:00
# include "libavutil/opt.h"
# include "avcodec.h"
2022-03-16 18:18:28 +01:00
# include "codec_internal.h"
2015-03-13 10:20:31 +01:00
# include "qsv.h"
# include "qsvenc.h"
2020-08-15 14:43:11 -03:00
# include "atsc_a53.h"
2015-03-13 10:20:31 +01:00
typedef struct QSVH264EncContext {
AVClass * class ;
QSVEncContext qsv ;
} QSVH264EncContext ;
2015-10-27 12:08:45 -05:00
static int qsv_h264_set_encode_ctrl ( AVCodecContext * avctx ,
const AVFrame * frame , mfxEncodeCtrl * enc_ctrl )
{
2016-06-19 12:04:03 -03:00
QSVH264EncContext * qh264 = avctx - > priv_data ;
QSVEncContext * q = & qh264 - > qsv ;
2015-10-27 12:08:45 -05:00
if ( q - > a53_cc & & frame ) {
2016-06-04 21:11:52 +03:00
mfxPayload * payload ;
mfxU8 * sei_data ;
size_t sei_size ;
int res ;
res = ff_alloc_a53_sei ( frame , sizeof ( mfxPayload ) + 2 , ( void * * ) & payload , & sei_size ) ;
2016-10-05 09:34:25 -04:00
if ( res < 0 | | ! payload )
2016-06-04 21:11:52 +03:00
return res ;
sei_data = ( mfxU8 * ) ( payload + 1 ) ;
// SEI header
sei_data [ 0 ] = 4 ;
sei_data [ 1 ] = ( mfxU8 ) sei_size ; // size of SEI data
// SEI data filled in by ff_alloc_a53_sei
payload - > BufSize = sei_size + 2 ;
payload - > NumBit = payload - > BufSize * 8 ;
payload - > Type = 4 ;
payload - > Data = sei_data ;
enc_ctrl - > NumExtParam = 0 ;
enc_ctrl - > NumPayload = 1 ;
enc_ctrl - > Payload [ 0 ] = payload ;
2015-10-27 12:08:45 -05:00
}
return 0 ;
}
2015-03-13 10:20:31 +01:00
static av_cold int qsv_enc_init ( AVCodecContext * avctx )
{
QSVH264EncContext * q = avctx - > priv_data ;
2015-10-27 12:08:45 -05:00
q - > qsv . set_encode_ctrl_cb = qsv_h264_set_encode_ctrl ;
2015-03-13 10:20:31 +01:00
return ff_qsv_enc_init ( avctx , & q - > qsv ) ;
}
static int qsv_enc_frame ( AVCodecContext * avctx , AVPacket * pkt ,
const AVFrame * frame , int * got_packet )
{
QSVH264EncContext * q = avctx - > priv_data ;
return ff_qsv_encode ( avctx , & q - > qsv , pkt , frame , got_packet ) ;
}
static av_cold int qsv_enc_close ( AVCodecContext * avctx )
{
QSVH264EncContext * q = avctx - > priv_data ;
return ff_qsv_enc_close ( avctx , & q - > qsv ) ;
}
# define OFFSET(x) offsetof(QSVH264EncContext, x)
# define VE AV_OPT_FLAG_VIDEO_PARAM | AV_OPT_FLAG_ENCODING_PARAM
static const AVOption options [ ] = {
2015-11-07 20:50:41 +01:00
QSV_COMMON_OPTS
2022-05-19 12:54:16 -07:00
QSV_OPTION_RDO
QSV_OPTION_MAX_FRAME_SIZE
QSV_OPTION_MAX_SLICE_SIZE
QSV_OPTION_BITRATE_LIMIT
QSV_OPTION_MBBRC
QSV_OPTION_EXTBRC
QSV_OPTION_ADAPTIVE_I
QSV_OPTION_ADAPTIVE_B
QSV_OPTION_P_STRATEGY
QSV_OPTION_B_STRATEGY
QSV_OPTION_DBLK_IDC
QSV_OPTION_LOW_DELAY_BRC
2022-05-25 21:13:41 +08:00
QSV_OPTION_MAX_MIN_QP
2022-03-23 13:38:12 +08:00
QSV_OPTION_SCENARIO
2022-09-01 10:12:57 +08:00
QSV_OPTION_AVBR
2022-11-02 16:11:51 +08:00
QSV_OPTION_SKIP_FRAME
2022-12-05 20:18:21 +00:00
# if QSV_HAVE_HE
QSV_HE_OPTIONS
# endif
2015-11-07 20:50:41 +01:00
2021-12-25 11:48:59 +08:00
{ " cavlc " , " Enable CAVLC " , OFFSET ( qsv . cavlc ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , VE } ,
2018-06-07 16:51:54 +08:00
# if QSV_HAVE_VCM
2021-12-25 11:48:59 +08:00
{ " vcm " , " Use the video conferencing mode ratecontrol " , OFFSET ( qsv . vcm ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , VE } ,
2018-06-07 16:51:54 +08:00
# endif
2015-03-13 10:20:31 +01:00
{ " idr_interval " , " Distance (in I-frames) between IDR frames " , OFFSET ( qsv . idr_interval ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , INT_MAX , VE } ,
2021-12-25 11:48:59 +08:00
{ " pic_timing_sei " , " Insert picture timing SEI with pic_struct_syntax element " , OFFSET ( qsv . pic_timing_sei ) , AV_OPT_TYPE_BOOL , { . i64 = 1 } , 0 , 1 , VE } ,
2015-11-08 07:58:49 +01:00
{ " single_sei_nal_unit " , " Put all the SEI messages into one NALU " , OFFSET ( qsv . single_sei_nal_unit ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , 1 , VE } ,
{ " max_dec_frame_buffering " , " Maximum number of frames buffered in the DPB " , OFFSET ( qsv . max_dec_frame_buffering ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , UINT16_MAX , VE } ,
2015-03-13 10:20:31 +01:00
2021-12-25 11:48:59 +08:00
{ " look_ahead " , " Use VBR algorithm with look ahead " , OFFSET ( qsv . look_ahead ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , VE } ,
2015-08-21 09:17:35 +01:00
{ " look_ahead_depth " , " Depth of look ahead in number frames " , OFFSET ( qsv . look_ahead_depth ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , 100 , VE } ,
2017-12-14 19:04:37 -03:00
{ " look_ahead_downsampling " , " Downscaling factor for the frames saved for the lookahead analysis " , OFFSET ( qsv . look_ahead_downsampling ) ,
2024-02-11 15:41:05 +01:00
AV_OPT_TYPE_INT , { . i64 = MFX_LOOKAHEAD_DS_UNKNOWN } , MFX_LOOKAHEAD_DS_UNKNOWN , MFX_LOOKAHEAD_DS_4x , VE , . unit = " look_ahead_downsampling " } ,
{ " unknown " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_LOOKAHEAD_DS_UNKNOWN } , INT_MIN , INT_MAX , VE , . unit = " look_ahead_downsampling " } ,
{ " auto " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_LOOKAHEAD_DS_UNKNOWN } , INT_MIN , INT_MAX , VE , . unit = " look_ahead_downsampling " } ,
{ " off " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_LOOKAHEAD_DS_OFF } , INT_MIN , INT_MAX , VE , . unit = " look_ahead_downsampling " } ,
{ " 2x " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_LOOKAHEAD_DS_2x } , INT_MIN , INT_MAX , VE , . unit = " look_ahead_downsampling " } ,
{ " 4x " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_LOOKAHEAD_DS_4x } , INT_MIN , INT_MAX , VE , . unit = " look_ahead_downsampling " } ,
{ " int_ref_type " , " Intra refresh type. B frames should be set to 0. " , OFFSET ( qsv . int_ref_type ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , UINT16_MAX , VE , . unit = " int_ref_type " } ,
{ " none " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = 0 } , . flags = VE , . unit = " int_ref_type " } ,
{ " vertical " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = 1 } , . flags = VE , . unit = " int_ref_type " } ,
{ " horizontal " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = 2 } , . flags = VE , . unit = " int_ref_type " } ,
{ " slice " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = 3 } , . flags = VE , . unit = " int_ref_type " } ,
2015-11-08 07:58:49 +01:00
{ " int_ref_cycle_size " , " Number of frames in the intra refresh cycle " , OFFSET ( qsv . int_ref_cycle_size ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , UINT16_MAX , VE } ,
{ " int_ref_qp_delta " , " QP difference for the refresh MBs " , OFFSET ( qsv . int_ref_qp_delta ) , AV_OPT_TYPE_INT , { . i64 = INT16_MIN } , INT16_MIN , INT16_MAX , VE } ,
{ " recovery_point_sei " , " Insert recovery point SEI messages " , OFFSET ( qsv . recovery_point_sei ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , 1 , VE } ,
2022-01-25 15:12:37 +08:00
{ " int_ref_cycle_dist " , " Distance between the beginnings of the intra-refresh cycles in frames " , OFFSET ( qsv . int_ref_cycle_dist ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , INT16_MAX , VE } ,
2024-02-11 15:41:05 +01:00
{ " profile " , NULL , OFFSET ( qsv . profile ) , AV_OPT_TYPE_INT , { . i64 = MFX_PROFILE_UNKNOWN } , 0 , INT_MAX , VE , . unit = " profile " } ,
{ " unknown " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_PROFILE_UNKNOWN } , INT_MIN , INT_MAX , VE , . unit = " profile " } ,
{ " baseline " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_PROFILE_AVC_BASELINE } , INT_MIN , INT_MAX , VE , . unit = " profile " } ,
{ " main " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_PROFILE_AVC_MAIN } , INT_MIN , INT_MAX , VE , . unit = " profile " } ,
{ " high " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_PROFILE_AVC_HIGH } , INT_MIN , INT_MAX , VE , . unit = " profile " } ,
2015-03-13 10:20:31 +01:00
2021-12-25 11:48:59 +08:00
{ " a53cc " , " Use A53 Closed Captions (if available) " , OFFSET ( qsv . a53_cc ) , AV_OPT_TYPE_BOOL , { . i64 = 1 } , 0 , 1 , VE } ,
2018-03-29 20:34:43 -03:00
2021-12-25 11:48:59 +08:00
{ " aud " , " Insert the Access Unit Delimiter NAL " , OFFSET ( qsv . aud ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , VE } ,
2018-03-15 18:02:12 +08:00
2018-04-02 15:17:23 +02:00
# if QSV_HAVE_MF
2024-02-11 15:41:05 +01:00
{ " mfmode " , " Multi-Frame Mode " , OFFSET ( qsv . mfmode ) , AV_OPT_TYPE_INT , { . i64 = MFX_MF_AUTO } , MFX_MF_DEFAULT , MFX_MF_AUTO , VE , . unit = " mfmode " } ,
{ " off " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_MF_DISABLED } , INT_MIN , INT_MAX , VE , . unit = " mfmode " } ,
{ " auto " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_MF_AUTO } , INT_MIN , INT_MAX , VE , . unit = " mfmode " } ,
2018-04-02 15:17:23 +02:00
# endif
2018-10-25 19:14:16 +08:00
{ " repeat_pps " , " repeat pps for every frame " , OFFSET ( qsv . repeat_pps ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , VE } ,
2015-03-13 10:20:31 +01:00
{ NULL } ,
} ;
static const AVClass class = {
. class_name = " h264_qsv encoder " ,
2024-01-19 13:33:28 +01:00
. item_name = av_default_item_name ,
2015-03-13 10:20:31 +01:00
. option = options ,
. version = LIBAVUTIL_VERSION_INT ,
} ;
2022-03-16 21:26:11 +01:00
static const FFCodecDefault qsv_enc_defaults [ ] = {
2024-02-01 16:35:35 +08:00
{ " b " , " 0 " } ,
2015-03-13 10:20:31 +01:00
{ " refs " , " 0 " } ,
2022-09-26 17:36:29 +08:00
{ " g " , " -1 " } ,
{ " bf " , " -1 " } ,
2018-08-27 16:23:31 +08:00
{ " qmin " , " -1 " } ,
{ " qmax " , " -1 " } ,
2019-03-27 19:33:37 +08:00
{ " trellis " , " -1 " } ,
2015-03-13 10:20:31 +01:00
{ " flags " , " +cgop " } ,
{ NULL } ,
} ;
2022-03-16 21:09:54 +01:00
const FFCodec ff_h264_qsv_encoder = {
. p . name = " h264_qsv " ,
2022-08-29 13:38:02 +02:00
CODEC_LONG_NAME ( " H.264 / AVC / MPEG-4 AVC / MPEG-4 part 10 (Intel Quick Sync Video acceleration) " ) ,
2015-03-13 10:20:31 +01:00
. priv_data_size = sizeof ( QSVH264EncContext ) ,
2022-03-16 21:09:54 +01:00
. p . type = AVMEDIA_TYPE_VIDEO ,
. p . id = AV_CODEC_ID_H264 ,
2015-03-13 10:20:31 +01:00
. init = qsv_enc_init ,
2022-03-30 23:28:24 +02:00
FF_CODEC_ENCODE_CB ( qsv_enc_frame ) ,
2015-03-13 10:20:31 +01:00
. close = qsv_enc_close ,
2022-03-16 21:09:54 +01:00
. p . capabilities = AV_CODEC_CAP_DELAY | AV_CODEC_CAP_HYBRID ,
. p . pix_fmts = ( const enum AVPixelFormat [ ] ) { AV_PIX_FMT_NV12 ,
2015-03-13 10:20:31 +01:00
AV_PIX_FMT_QSV ,
AV_PIX_FMT_NONE } ,
2022-03-16 21:09:54 +01:00
. p . priv_class = & class ,
2015-03-13 10:20:31 +01:00
. defaults = qsv_enc_defaults ,
2022-07-09 22:25:41 +02:00
. caps_internal = FF_CODEC_CAP_NOT_INIT_THREADSAFE |
FF_CODEC_CAP_INIT_CLEANUP ,
2022-03-16 21:09:54 +01:00
. p . wrapper_name = " qsv " ,
2020-04-13 16:33:16 +01:00
. hw_configs = ff_qsv_enc_hw_configs ,
2015-03-13 10:20:31 +01:00
} ;