2015-03-13 11:20:31 +02:00
/*
* Intel MediaSDK QSV encoder utility functions
*
* copyright ( c ) 2013 Yukinori Yamazoe
*
2015-03-28 02:33:27 +02:00
* This file is part of FFmpeg .
2015-03-13 11:20:31 +02:00
*
2015-03-28 02:33:27 +02:00
* FFmpeg is free software ; you can redistribute it and / or
2015-03-13 11:20:31 +02: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 02:33:27 +02:00
* FFmpeg is distributed in the hope that it will be useful ,
2015-03-13 11:20:31 +02: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 02:33:27 +02:00
* License along with FFmpeg ; if not , write to the Free Software
2015-03-13 11:20:31 +02:00
* Foundation , Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA
*/
# ifndef AVCODEC_QSVENC_H
# define AVCODEC_QSVENC_H
# include <stdint.h>
# include <sys/types.h>
2020-09-08 05:17:27 +02:00
# include <mfxvideo.h>
2015-03-13 11:20:31 +02:00
2021-01-04 04:46:14 +02:00
# include "libavutil/common.h"
# include "libavutil/hwcontext.h"
# include "libavutil/hwcontext_qsv.h"
2015-03-13 11:20:31 +02:00
# include "libavutil/avutil.h"
2015-07-14 18:16:26 +02:00
# include "libavutil/fifo.h"
2015-03-13 11:20:31 +02:00
# include "avcodec.h"
2020-04-13 17:33:16 +02:00
# include "hwconfig.h"
2015-03-13 11:20:31 +02:00
# include "qsv_internal.h"
2022-01-13 07:45:25 +02:00
# define QSV_HAVE_EXT_VP9_TILES QSV_VERSION_ATLEAST(1, 29)
2022-10-13 10:25:23 +02:00
# define QSV_HAVE_EXT_AV1_PARAM QSV_VERSION_ATLEAST(2, 5)
2019-10-26 16:18:31 +02:00
2018-06-15 16:52:28 +02:00
# if defined(_WIN32) || defined(__CYGWIN__)
2022-05-22 14:19:11 +02:00
# define QSV_HAVE_AVBR 1
# define QSV_HAVE_VCM 1
2018-04-02 15:17:23 +02:00
# define QSV_HAVE_MF 0
2017-12-07 08:44:26 +02:00
# else
2018-01-30 12:07:23 +02:00
# define QSV_HAVE_AVBR 0
2017-12-07 08:44:26 +02:00
# define QSV_HAVE_VCM 0
2020-08-19 06:41:16 +02:00
# define QSV_HAVE_MF !QSV_ONEVPL
2017-12-12 16:27:52 +02:00
# endif
2015-11-07 21:50:41 +02:00
# define QSV_COMMON_OPTS \
2018-09-03 00:23:26 +02:00
{ " async_depth " , " Maximum processing parallelism " , OFFSET ( qsv . async_depth ) , AV_OPT_TYPE_INT , { . i64 = ASYNC_DEPTH_DEFAULT } , 1 , INT_MAX , VE } , \
2022-09-26 11:36:30 +02:00
{ " preset " , NULL , OFFSET ( qsv . preset ) , AV_OPT_TYPE_INT , { . i64 = MFX_TARGETUSAGE_UNKNOWN } , MFX_TARGETUSAGE_UNKNOWN , MFX_TARGETUSAGE_BEST_SPEED , VE , " preset " } , \
2015-11-24 15:48:28 +02:00
{ " veryfast " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_TARGETUSAGE_BEST_SPEED } , INT_MIN , INT_MAX , VE , " preset " } , \
{ " faster " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_TARGETUSAGE_6 } , INT_MIN , INT_MAX , VE , " preset " } , \
{ " fast " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_TARGETUSAGE_5 } , INT_MIN , INT_MAX , VE , " preset " } , \
{ " medium " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_TARGETUSAGE_BALANCED } , INT_MIN , INT_MAX , VE , " preset " } , \
{ " slow " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_TARGETUSAGE_3 } , INT_MIN , INT_MAX , VE , " preset " } , \
{ " slower " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_TARGETUSAGE_2 } , INT_MIN , INT_MAX , VE , " preset " } , \
{ " veryslow " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_TARGETUSAGE_BEST_QUALITY } , INT_MIN , INT_MAX , VE , " preset " } , \
2022-05-19 21:54:16 +02:00
{ " forced_idr " , " Forcing I frames as IDR frames " , OFFSET ( qsv . forced_idr ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , VE } , \
{ " low_power " , " enable low power mode(experimental: many limitations by mfx version, BRC modes, etc.) " , OFFSET ( qsv . low_power ) , AV_OPT_TYPE_BOOL , { . i64 = - 1 } , - 1 , 1 , VE } ,
# define QSV_OPTION_RDO \
{ " rdo " , " Enable rate distortion optimization " , OFFSET ( qsv . rdo ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , 1 , VE } ,
# define QSV_OPTION_MAX_FRAME_SIZE \
2022-03-21 10:33:14 +02:00
{ " max_frame_size " , " Maximum encoded frame size in bytes " , OFFSET ( qsv . max_frame_size ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , INT_MAX , VE } , \
{ " max_frame_size_i " , " Maximum encoded I frame size in bytes " , OFFSET ( qsv . max_frame_size_i ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , INT_MAX , VE } , \
2022-05-19 21:54:16 +02:00
{ " max_frame_size_p " , " Maximum encoded P frame size in bytes " , OFFSET ( qsv . max_frame_size_p ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , INT_MAX , VE } ,
# define QSV_OPTION_MAX_SLICE_SIZE \
{ " max_slice_size " , " Maximum encoded slice size in bytes " , OFFSET ( qsv . max_slice_size ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , INT_MAX , VE } ,
# define QSV_OPTION_BITRATE_LIMIT \
{ " bitrate_limit " , " Toggle bitrate limitations " , OFFSET ( qsv . bitrate_limit ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , 1 , VE } ,
# define QSV_OPTION_MBBRC \
{ " mbbrc " , " MB level bitrate control " , OFFSET ( qsv . mbbrc ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , 1 , VE } ,
# define QSV_OPTION_EXTBRC \
{ " extbrc " , " Extended bitrate control " , OFFSET ( qsv . extbrc ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , 1 , VE } ,
# define QSV_OPTION_ADAPTIVE_I \
{ " adaptive_i " , " Adaptive I-frame placement " , OFFSET ( qsv . adaptive_i ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , 1 , VE } ,
# define QSV_OPTION_ADAPTIVE_B \
{ " adaptive_b " , " Adaptive B-frame placement " , OFFSET ( qsv . adaptive_b ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , 1 , VE } ,
# define QSV_OPTION_P_STRATEGY \
{ " p_strategy " , " Enable P-pyramid: 0-default 1-simple 2-pyramid(bf need to be set to 0). " , OFFSET ( qsv . p_strategy ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , 2 , VE } ,
# define QSV_OPTION_B_STRATEGY \
{ " b_strategy " , " Strategy to choose between I/P/B-frames " , OFFSET ( qsv . b_strategy ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , 1 , VE } ,
# define QSV_OPTION_DBLK_IDC \
{ " dblk_idc " , " This option disable deblocking. It has value in range 0~2. " , OFFSET ( qsv . dblk_idc ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , 2 , VE } ,
# define QSV_OPTION_LOW_DELAY_BRC \
{ " low_delay_brc " , " Allow to strictly obey avg frame size " , OFFSET ( qsv . low_delay_brc ) , AV_OPT_TYPE_BOOL , { . i64 = - 1 } , - 1 , 1 , VE } ,
2015-11-07 21:50:41 +02:00
2022-05-25 15:13:41 +02:00
# define QSV_OPTION_MAX_MIN_QP \
{ " max_qp_i " , " Maximum video quantizer scale for I frame " , OFFSET ( qsv . max_qp_i ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , 51 , VE } , \
{ " min_qp_i " , " Minimum video quantizer scale for I frame " , OFFSET ( qsv . min_qp_i ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , 51 , VE } , \
{ " max_qp_p " , " Maximum video quantizer scale for P frame " , OFFSET ( qsv . max_qp_p ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , 51 , VE } , \
{ " min_qp_p " , " Minimum video quantizer scale for P frame " , OFFSET ( qsv . min_qp_p ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , 51 , VE } , \
{ " max_qp_b " , " Maximum video quantizer scale for B frame " , OFFSET ( qsv . max_qp_b ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , 51 , VE } , \
{ " min_qp_b " , " Minimum video quantizer scale for B frame " , OFFSET ( qsv . min_qp_b ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , 51 , VE } ,
2022-03-23 07:38:12 +02:00
# define QSV_OPTION_SCENARIO \
{ " scenario " , " A hint to encoder about the scenario for the encoding session " , OFFSET ( qsv . scenario ) , AV_OPT_TYPE_INT , { . i64 = MFX_SCENARIO_UNKNOWN } , \
MFX_SCENARIO_UNKNOWN , MFX_SCENARIO_REMOTE_GAMING , VE , " scenario " } , \
{ " unknown " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_SCENARIO_UNKNOWN } , . flags = VE , " scenario " } , \
{ " displayremoting " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_SCENARIO_DISPLAY_REMOTING } , . flags = VE , " scenario " } , \
{ " videoconference " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_SCENARIO_VIDEO_CONFERENCE } , . flags = VE , " scenario " } , \
{ " archive " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_SCENARIO_ARCHIVE } , . flags = VE , " scenario " } , \
{ " livestreaming " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_SCENARIO_LIVE_STREAMING } , . flags = VE , " scenario " } , \
{ " cameracapture " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_SCENARIO_CAMERA_CAPTURE } , . flags = VE , " scenario " } , \
{ " videosurveillance " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_SCENARIO_VIDEO_SURVEILLANCE } , . flags = VE , " scenario " } , \
{ " gamestreaming " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_SCENARIO_GAME_STREAMING } , . flags = VE , " scenario " } , \
{ " remotegaming " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = MFX_SCENARIO_REMOTE_GAMING } , . flags = VE , " scenario " } ,
2022-09-01 04:12:57 +02:00
# define QSV_OPTION_AVBR \
{ " avbr_accuracy " , " Accuracy of the AVBR ratecontrol (unit of tenth of percent) " , OFFSET ( qsv . avbr_accuracy ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , UINT16_MAX , VE } , \
{ " avbr_convergence " , " Convergence of the AVBR ratecontrol (unit of 100 frames) " , OFFSET ( qsv . avbr_convergence ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , UINT16_MAX , VE } ,
2022-11-02 10:11:51 +02:00
# define QSV_OPTION_SKIP_FRAME \
{ " skip_frame " , " Allow frame skipping " , OFFSET ( qsv . skip_frame ) , AV_OPT_TYPE_INT , { . i64 = MFX_SKIPFRAME_NO_SKIP } , \
MFX_SKIPFRAME_NO_SKIP , MFX_SKIPFRAME_BRC_ONLY , VE , " skip_frame " } , \
{ " no_skip " , " Frame skipping is disabled " , \
0 , AV_OPT_TYPE_CONST , { . i64 = MFX_SKIPFRAME_NO_SKIP } , . flags = VE , " skip_frame " } , \
{ " insert_dummy " , " Encoder inserts into bitstream frame where all macroblocks are encoded as skipped " , \
0 , AV_OPT_TYPE_CONST , { . i64 = MFX_SKIPFRAME_INSERT_DUMMY } , . flags = VE , " skip_frame " } , \
{ " insert_nothing " , " Encoder inserts nothing into bitstream " , \
0 , AV_OPT_TYPE_CONST , { . i64 = MFX_SKIPFRAME_INSERT_NOTHING } , . flags = VE , " skip_frame " } , \
{ " brc_only " , " skip_frame metadata indicates the number of missed frames before the current frame " , \
0 , AV_OPT_TYPE_CONST , { . i64 = MFX_SKIPFRAME_BRC_ONLY } , . flags = VE , " skip_frame " } ,
2020-11-08 20:51:15 +02:00
extern const AVCodecHWConfigInternal * const ff_qsv_enc_hw_configs [ ] ;
2020-04-13 17:33:16 +02:00
2015-10-27 19:08:45 +02:00
typedef int SetEncodeCtrlCB ( AVCodecContext * avctx ,
const AVFrame * frame , mfxEncodeCtrl * enc_ctrl ) ;
2015-03-13 11:20:31 +02:00
typedef struct QSVEncContext {
AVCodecContext * avctx ;
QSVFrame * work_frames ;
mfxSession session ;
2019-09-19 22:45:26 +02:00
QSVSession internal_qs ;
2015-03-13 11:20:31 +02:00
int packet_size ;
2015-06-16 18:22:11 +02:00
int width_align ;
2017-05-06 06:31:30 +02:00
int height_align ;
2015-03-13 11:20:31 +02:00
mfxVideoParam param ;
mfxFrameAllocRequest req ;
mfxExtCodingOption extco ;
2015-08-21 10:17:35 +02:00
mfxExtCodingOption2 extco2 ;
2019-01-23 13:16:17 +02:00
mfxExtCodingOption3 extco3 ;
2018-04-02 15:17:23 +02:00
# if QSV_HAVE_MF
mfxExtMultiFrameParam extmfp ;
mfxExtMultiFrameControl extmfc ;
# endif
2019-11-26 05:56:18 +02:00
mfxExtHEVCTiles exthevctiles ;
2019-11-26 05:53:16 +02:00
mfxExtVP9Param extvp9param ;
2022-10-13 10:25:23 +02:00
# if QSV_HAVE_EXT_AV1_PARAM
mfxExtAV1TileParam extav1tileparam ;
mfxExtAV1BitstreamParam extav1bsparam ;
# endif
2019-10-26 16:18:31 +02:00
2020-08-19 03:43:12 +02:00
# if QSV_HAVE_OPAQUE
2015-10-11 22:02:26 +02:00
mfxExtOpaqueSurfaceAlloc opaque_alloc ;
mfxFrameSurface1 * * opaque_surfaces ;
AVBufferRef * opaque_alloc_buf ;
2020-08-19 03:43:12 +02:00
# endif
2015-10-11 22:02:26 +02:00
2021-07-28 10:15:47 +02:00
mfxExtVideoSignalInfo extvsi ;
2022-10-13 10:25:23 +02:00
mfxExtBuffer * extparam_internal [ 5 + ( QSV_HAVE_MF * 2 ) + QSV_HAVE_EXT_AV1_PARAM * 2 ] ;
2015-10-11 22:02:26 +02:00
int nb_extparam_internal ;
2015-08-05 15:22:34 +02:00
mfxExtBuffer * * extparam ;
2015-03-13 11:20:31 +02:00
2022-01-07 12:41:42 +02:00
AVFifo * async_fifo ;
2015-07-14 18:16:26 +02:00
2016-10-21 19:57:12 +02:00
QSVFramesContext frames_ctx ;
2019-03-26 01:40:54 +02:00
mfxVersion ver ;
2019-03-26 20:56:24 +02:00
int hevc_vps ;
2015-03-13 11:20:31 +02:00
// options set by the caller
int async_depth ;
int idr_interval ;
int profile ;
2022-11-01 19:58:06 +02:00
int tier ;
2015-03-13 11:20:31 +02:00
int preset ;
int avbr_accuracy ;
int avbr_convergence ;
2015-08-19 10:17:40 +02:00
int pic_timing_sei ;
2015-08-21 10:17:35 +02:00
int look_ahead ;
int look_ahead_depth ;
int look_ahead_downsampling ;
2015-11-08 07:44:17 +02:00
int vcm ;
2015-11-08 08:58:49 +02:00
int rdo ;
int max_frame_size ;
2022-03-21 10:33:14 +02:00
int max_frame_size_i ;
int max_frame_size_p ;
2015-11-08 08:58:49 +02:00
int max_slice_size ;
2022-01-17 07:11:20 +02:00
int dblk_idc ;
2022-03-23 07:38:12 +02:00
int scenario ;
2015-11-08 08:58:49 +02:00
2019-11-26 05:56:18 +02:00
int tile_cols ;
int tile_rows ;
2018-03-15 12:02:12 +02:00
int aud ;
2015-11-08 08:58:49 +02:00
int single_sei_nal_unit ;
int max_dec_frame_buffering ;
int bitrate_limit ;
int mbbrc ;
int extbrc ;
int adaptive_i ;
int adaptive_b ;
2015-09-09 04:18:26 +02:00
int b_strategy ;
2022-01-17 07:11:21 +02:00
int p_strategy ;
2016-01-13 22:27:00 +02:00
int cavlc ;
2015-11-08 08:58:49 +02:00
int int_ref_type ;
int int_ref_cycle_size ;
int int_ref_qp_delta ;
2022-01-25 09:12:37 +02:00
int int_ref_cycle_dist ;
2015-11-08 08:58:49 +02:00
int recovery_point_sei ;
2015-06-16 18:22:11 +02:00
2018-10-25 13:14:16 +02:00
int repeat_pps ;
2018-11-29 08:14:22 +02:00
int low_power ;
2019-01-11 09:09:07 +02:00
int gpb ;
2022-01-17 07:11:22 +02:00
int transform_skip ;
2018-10-25 13:14:16 +02:00
2015-10-27 19:08:45 +02:00
int a53_cc ;
2018-04-14 01:34:23 +02:00
2018-04-02 15:17:23 +02:00
# if QSV_HAVE_MF
int mfmode ;
# endif
2015-06-16 18:22:11 +02:00
char * load_plugins ;
2015-10-27 19:08:45 +02:00
SetEncodeCtrlCB * set_encode_ctrl_cb ;
2018-11-29 10:29:00 +02:00
int forced_idr ;
2022-03-21 10:33:14 +02:00
int low_delay_brc ;
2022-05-22 14:19:11 +02:00
int co2_idx ;
int co3_idx ;
int exthevctiles_idx ;
int vp9_idx ;
2022-05-25 15:13:41 +02:00
int max_qp_i ;
int min_qp_i ;
int max_qp_p ;
int min_qp_p ;
int max_qp_b ;
int min_qp_b ;
2022-07-14 10:43:25 +02:00
// These are used for qp reset
int old_global_quality ;
float old_i_quant_factor ;
float old_i_quant_offset ;
float old_b_quant_factor ;
float old_b_quant_offset ;
2022-09-06 11:22:53 +02:00
// This is used for max_frame_size reset
int old_max_frame_size ;
2022-09-06 11:22:54 +02:00
// This is used for gop reset
int old_gop_size ;
2022-09-06 11:22:56 +02:00
// These are used for intra refresh reset
int old_int_ref_type ;
int old_int_ref_cycle_size ;
int old_int_ref_qp_delta ;
int old_int_ref_cycle_dist ;
2022-09-06 11:22:57 +02:00
// These are used for max/min qp reset;
int old_qmax ;
int old_qmin ;
int old_max_qp_i ;
int old_min_qp_i ;
int old_max_qp_p ;
int old_min_qp_p ;
int old_max_qp_b ;
int old_min_qp_b ;
2022-09-06 11:22:58 +02:00
// This is used for low_delay_brc reset
int old_low_delay_brc ;
2022-09-23 04:44:10 +02:00
// This is used for framerate reset
AVRational old_framerate ;
2022-09-23 04:44:11 +02:00
// These are used for bitrate control reset
int old_bit_rate ;
int old_rc_buffer_size ;
int old_rc_initial_buffer_occupancy ;
int old_rc_max_rate ;
2022-09-23 04:44:12 +02:00
// This is used for SEI Timing reset
int old_pic_timing_sei ;
2022-11-02 10:11:51 +02:00
int skip_frame ;
2015-03-13 11:20:31 +02:00
} QSVEncContext ;
int ff_qsv_enc_init ( AVCodecContext * avctx , QSVEncContext * q ) ;
int ff_qsv_encode ( AVCodecContext * avctx , QSVEncContext * q ,
AVPacket * pkt , const AVFrame * frame , int * got_packet ) ;
int ff_qsv_enc_close ( AVCodecContext * avctx , QSVEncContext * q ) ;
# endif /* AVCODEC_QSVENC_H */