@ -148,14 +148,6 @@ typedef struct VAAPIEncodeH264Context {
// Rate control configuration.
int send_timing_sei ;
struct {
VAEncMiscParameterBuffer misc ;
VAEncMiscParameterRateControl rc ;
} rc_params ;
struct {
VAEncMiscParameterBuffer misc ;
VAEncMiscParameterHRD hrd ;
} hrd_params ;
# if VA_CHECK_VERSION(0, 36, 0)
// Speed-quality tradeoff setting.
@ -797,16 +789,16 @@ static int vaapi_encode_h264_init_sequence_params(AVCodecContext *avctx)
vseq - > seq_fields . bits . log2_max_frame_num_minus4 = 4 ;
vseq - > seq_fields . bits . pic_order_cnt_type = 0 ;
if ( ctx - > input_ width ! = ctx - > aligned _width | |
ctx - > input_ height ! = ctx - > aligned _height) {
if ( av ctx- > width ! = ctx - > surface _width | |
av ctx- > height ! = ctx - > surface _height) {
vseq - > frame_cropping_flag = 1 ;
vseq - > frame_crop_left_offset = 0 ;
vseq - > frame_crop_right_offset =
( ctx - > aligned _width - ctx - > input_ width) / 2 ;
( ctx - > surface _width - av ctx- > width ) / 2 ;
vseq - > frame_crop_top_offset = 0 ;
vseq - > frame_crop_bottom_offset =
( ctx - > aligned _height - ctx - > input_ height) / 2 ;
( ctx - > surface _height - av ctx- > height ) / 2 ;
} else {
vseq - > frame_cropping_flag = 0 ;
}
@ -866,9 +858,9 @@ static int vaapi_encode_h264_init_sequence_params(AVCodecContext *avctx)
( avctx - > bit_rate > > mseq - > bit_rate_scale + 6 ) - 1 ;
mseq - > cpb_size_scale =
av_clip_uintp2 ( av_log2 ( priv - > hrd_params . hrd . buffer_size ) - 15 - 4 , 4 ) ;
av_clip_uintp2 ( av_log2 ( ctx - > hrd_params . hrd . buffer_size ) - 15 - 4 , 4 ) ;
mseq - > cpb_size_value_minus1 [ 0 ] =
( priv - > hrd_params . hrd . buffer_size > > mseq - > cpb_size_scale + 4 ) - 1 ;
( ctx - > hrd_params . hrd . buffer_size > > mseq - > cpb_size_scale + 4 ) - 1 ;
// CBR mode isn't actually available here, despite naming.
mseq - > cbr_flag [ 0 ] = 0 ;
@ -880,8 +872,8 @@ static int vaapi_encode_h264_init_sequence_params(AVCodecContext *avctx)
// This calculation can easily overflow 32 bits.
mseq - > initial_cpb_removal_delay = 90000 *
( uint64_t ) priv - > hrd_params . hrd . initial_buffer_fullness /
priv - > hrd_params . hrd . buffer_size ;
( uint64_t ) ctx - > hrd_params . hrd . initial_buffer_fullness /
ctx - > hrd_params . hrd . buffer_size ;
mseq - > initial_cpb_removal_delay_offset = 0 ;
} else {
@ -1083,100 +1075,94 @@ static int vaapi_encode_h264_init_slice_params(AVCodecContext *avctx,
return 0 ;
}
static av_cold int vaapi_encode_h264_init_constant_bitrat e ( AVCodecContext * avctx )
static av_cold int vaapi_encode_h264_configur e ( AVCodecContext * avctx )
{
VAAPIEncodeContext * ctx = avctx - > priv_data ;
VAAPIEncodeH264Context * priv = ctx - > priv_data ;
int hrd_buffer_size ;
int hrd_initial_buffer_fullness ;
VAAPIEncodeH264Options * opt = ctx - > codec_options ;
if ( avctx - > bit_rate > INT32_MAX ) {
av_log ( avctx , AV_LOG_ERROR , " Target bitrate of 2^31 bps or "
" higher is not supported. \n " ) ;
return AVERROR ( EINVAL ) ;
priv - > mb_width = FFALIGN ( avctx - > width , 16 ) / 16 ;
priv - > mb_height = FFALIGN ( avctx - > height , 16 ) / 16 ;
if ( ctx - > va_rc_mode = = VA_RC_CQP ) {
priv - > fixed_qp_p = opt - > qp ;
if ( avctx - > i_quant_factor > 0.0 )
priv - > fixed_qp_idr = ( int ) ( ( priv - > fixed_qp_p * avctx - > i_quant_factor +
avctx - > i_quant_offset ) + 0.5 ) ;
else
priv - > fixed_qp_idr = priv - > fixed_qp_p ;
if ( avctx - > b_quant_factor > 0.0 )
priv - > fixed_qp_b = ( int ) ( ( priv - > fixed_qp_p * avctx - > b_quant_factor +
avctx - > b_quant_offset ) + 0.5 ) ;
else
priv - > fixed_qp_b = priv - > fixed_qp_p ;
av_log ( avctx , AV_LOG_DEBUG , " Using fixed QP = "
" %d / %d / %d for IDR- / P- / B-frames. \n " ,
priv - > fixed_qp_idr , priv - > fixed_qp_p , priv - > fixed_qp_b ) ;
} else if ( ctx - > va_rc_mode = = VA_RC_CBR ) {
// These still need to be set for pic_init_qp/slice_qp_delta.
priv - > fixed_qp_idr = 26 ;
priv - > fixed_qp_p = 26 ;
priv - > fixed_qp_b = 26 ;
av_log ( avctx , AV_LOG_DEBUG , " Using constant-bitrate = %d bps. \n " ,
avctx - > bit_rate ) ;
} else {
av_assert0 ( 0 & & " Invalid RC mode. " ) ;
}
if ( opt - > quality > 0 ) {
# if VA_CHECK_VERSION(0, 36, 0)
priv - > quality_params . misc . type =
VAEncMiscParameterTypeQualityLevel ;
priv - > quality_params . quality . quality_level = opt - > quality ;
ctx - > global_params [ ctx - > nb_global_params ] =
& priv - > quality_params . misc ;
ctx - > global_params_size [ ctx - > nb_global_params + + ] =
sizeof ( priv - > quality_params ) ;
# else
av_log ( avctx , AV_LOG_WARNING , " The encode quality option is not "
" supported with this VAAPI version. \n " ) ;
# endif
}
if ( avctx - > rc_buffer_size )
hrd_buffer_size = avctx - > rc_buffer_size ;
else
hrd_buffer_size = avctx - > bit_rate ;
if ( avctx - > rc_initial_buffer_occupancy )
hrd_initial_buffer_fullness = avctx - > rc_initial_buffer_occupancy ;
else
hrd_initial_buffer_fullness = hrd_buffer_size * 3 / 4 ;
priv - > rc_params . misc . type = VAEncMiscParameterTypeRateControl ;
priv - > rc_params . rc = ( VAEncMiscParameterRateControl ) {
. bits_per_second = avctx - > bit_rate ,
. target_percentage = 66 ,
. window_size = 1000 ,
. initial_qp = ( avctx - > qmax > = 0 ? avctx - > qmax : 40 ) ,
. min_qp = ( avctx - > qmin > = 0 ? avctx - > qmin : 18 ) ,
. basic_unit_size = 0 ,
} ;
ctx - > global_params [ ctx - > nb_global_params ] =
& priv - > rc_params . misc ;
ctx - > global_params_size [ ctx - > nb_global_params + + ] =
sizeof ( priv - > rc_params ) ;
priv - > hrd_params . misc . type = VAEncMiscParameterTypeHRD ;
priv - > hrd_params . hrd = ( VAEncMiscParameterHRD ) {
. initial_buffer_fullness = hrd_initial_buffer_fullness ,
. buffer_size = hrd_buffer_size ,
} ;
ctx - > global_params [ ctx - > nb_global_params ] =
& priv - > hrd_params . misc ;
ctx - > global_params_size [ ctx - > nb_global_params + + ] =
sizeof ( priv - > hrd_params ) ;
// These still need to be set for pic_init_qp/slice_qp_delta.
priv - > fixed_qp_idr = 26 ;
priv - > fixed_qp_p = 26 ;
priv - > fixed_qp_b = 26 ;
av_log ( avctx , AV_LOG_DEBUG , " Using constant-bitrate = % " PRId64 " bps. \n " ,
avctx - > bit_rate ) ;
return 0 ;
}
static av_cold int vaapi_encode_h264_init_fixed_qp ( AVCodecContext * avctx )
{
VAAPIEncodeContext * ctx = avctx - > priv_data ;
VAAPIEncodeH264Context * priv = ctx - > priv_data ;
VAAPIEncodeH264Options * opt = ctx - > codec_options ;
static const VAAPIEncodeType vaapi_encode_type_h264 = {
. priv_data_size = sizeof ( VAAPIEncodeH264Context ) ,
priv - > fixed_qp_p = opt - > qp ;
if ( avctx - > i_quant_factor > 0.0 )
priv - > fixed_qp_idr = ( int ) ( ( priv - > fixed_qp_p * avctx - > i_quant_factor +
avctx - > i_quant_offset ) + 0.5 ) ;
else
priv - > fixed_qp_idr = priv - > fixed_qp_p ;
if ( avctx - > b_quant_factor > 0.0 )
priv - > fixed_qp_b = ( int ) ( ( priv - > fixed_qp_p * avctx - > b_quant_factor +
avctx - > b_quant_offset ) + 0.5 ) ;
else
priv - > fixed_qp_b = priv - > fixed_qp_p ;
. configure = & vaapi_encode_h264_configure ,
av_log ( avctx , AV_LOG_DEBUG , " Using fixed QP = "
" %d / %d / %d for IDR- / P- / B-frames. \n " ,
priv - > fixed_qp_idr , priv - > fixed_qp_p , priv - > fixed_qp_b ) ;
return 0 ;
}
. sequence_params_size = sizeof ( VAEncSequenceParameterBufferH264 ) ,
. init_sequence_params = & vaapi_encode_h264_init_sequence_params ,
. picture_params_size = sizeof ( VAEncPictureParameterBufferH264 ) ,
. init_picture_params = & vaapi_encode_h264_init_picture_params ,
static av_cold int vaapi_encode_h264_init_internal ( AVCodecContext * avctx )
. slice_params_size = sizeof ( VAEncSliceParameterBufferH264 ) ,
. init_slice_params = & vaapi_encode_h264_init_slice_params ,
. sequence_header_type = VAEncPackedHeaderSequence ,
. write_sequence_header = & vaapi_encode_h264_write_sequence_header ,
. slice_header_type = VAEncPackedHeaderH264_Slice ,
. write_slice_header = & vaapi_encode_h264_write_slice_header ,
. write_extra_header = & vaapi_encode_h264_write_extra_header ,
} ;
static av_cold int vaapi_encode_h264_init ( AVCodecContext * avctx )
{
static const VAConfigAttrib default_config_attributes [ ] = {
{ . type = VAConfigAttribRTFormat ,
. value = VA_RT_FORMAT_YUV420 } ,
{ . type = VAConfigAttribEncPackedHeaders ,
. value = ( VA_ENC_PACKED_HEADER_SEQUENCE |
VA_ENC_PACKED_HEADER_SLICE ) } ,
} ;
VAAPIEncodeContext * ctx = avctx - > priv_data ;
VAAPIEncodeH264Options * opt =
( VAAPIEncodeH264Options * ) ctx - > codec_options_data ;
VAAPIEncodeContext * ctx = avctx - > priv_data ;
VAAPIEncodeH264Context * priv = ctx - > priv_data ;
VAAPIEncodeH264Options * opt = ctx - > codec_options ;
int i , err ;
ctx - > codec = & vaapi_encode_type_h264 ;
switch ( avctx - > profile ) {
case FF_PROFILE_H264_CONSTRAINED_BASELINE :
@ -1216,7 +1202,7 @@ static av_cold int vaapi_encode_h264_init_internal(AVCodecContext *avctx)
return AVERROR ( EINVAL ) ;
}
if ( opt - > low_power ) {
# if VA_CHECK_VERSION(0, 39, 1 )
# if VA_CHECK_VERSION(0, 39, 2 )
ctx - > va_entrypoint = VAEntrypointEncSliceLP ;
# else
av_log ( avctx , AV_LOG_ERROR , " Low-power encoding is not "
@ -1227,80 +1213,19 @@ static av_cold int vaapi_encode_h264_init_internal(AVCodecContext *avctx)
ctx - > va_entrypoint = VAEntrypointEncSlice ;
}
ctx - > input_width = avctx - > width ;
ctx - > input_height = avctx - > height ;
ctx - > aligned_width = FFALIGN ( ctx - > input_width , 16 ) ;
ctx - > aligned_height = FFALIGN ( ctx - > input_height , 16 ) ;
priv - > mb_width = ctx - > aligned_width / 16 ;
priv - > mb_height = ctx - > aligned_height / 16 ;
// Only 8-bit encode is supported.
ctx - > va_rt_format = VA_RT_FORMAT_YUV420 ;
for ( i = 0 ; i < FF_ARRAY_ELEMS ( default_config_attributes ) ; i + + ) {
ctx - > config_attributes [ ctx - > nb_config_attributes + + ] =
default_config_attributes [ i ] ;
}
if ( avctx - > bit_rate > 0 ) {
if ( avctx - > bit_rate > 0 )
ctx - > va_rc_mode = VA_RC_CBR ;
err = vaapi_encode_h264_init_constant_bitrate ( avctx ) ;
} else {
else
ctx - > va_rc_mode = VA_RC_CQP ;
err = vaapi_encode_h264_init_fixed_qp ( avctx ) ;
}
if ( err < 0 )
return err ;
ctx - > config_attributes [ ctx - > nb_config_attributes + + ] = ( VAConfigAttrib ) {
. type = VAConfigAttribRateControl ,
. value = ctx - > va_rc_mode ,
} ;
if ( opt - > quality > 0 ) {
# if VA_CHECK_VERSION(0, 36, 0)
priv - > quality_params . misc . type =
VAEncMiscParameterTypeQualityLevel ;
priv - > quality_params . quality . quality_level = opt - > quality ;
ctx - > global_params [ ctx - > nb_global_params ] =
& priv - > quality_params . misc ;
ctx - > global_params_size [ ctx - > nb_global_params + + ] =
sizeof ( priv - > quality_params ) ;
# else
av_log ( avctx , AV_LOG_WARNING , " The encode quality option is not "
" supported with this VAAPI version. \n " ) ;
# endif
}
ctx - > nb_recon_frames = 20 ;
return 0 ;
}
static VAAPIEncodeType vaapi_encode_type_h264 = {
. priv_data_size = sizeof ( VAAPIEncodeH264Context ) ,
. init = & vaapi_encode_h264_init_internal ,
. sequence_params_size = sizeof ( VAEncSequenceParameterBufferH264 ) ,
. init_sequence_params = & vaapi_encode_h264_init_sequence_params ,
ctx - > surface_width = FFALIGN ( avctx - > width , 16 ) ;
ctx - > surface_height = FFALIGN ( avctx - > height , 16 ) ;
. picture_params_size = sizeof ( VAEncPictureParameterBufferH264 ) ,
. init_picture_params = & vaapi_encode_h264_init_picture_params ,
. slice_params_size = sizeof ( VAEncSliceParameterBufferH264 ) ,
. init_slice_params = & vaapi_encode_h264_init_slice_params ,
. sequence_header_type = VAEncPackedHeaderSequence ,
. write_sequence_header = & vaapi_encode_h264_write_sequence_header ,
. slice_header_type = VAEncPackedHeaderH264_Slice ,
. write_slice_header = & vaapi_encode_h264_write_slice_header ,
. write_extra_header = & vaapi_encode_h264_write_extra_header ,
} ;
static av_cold int vaapi_encode_h264_init ( AVCodecContext * avctx )
{
return ff_vaapi_encode_init ( avctx , & vaapi_encode_type_h264 ) ;
return ff_vaapi_encode_init ( avctx ) ;
}
# define OFFSET(x) (offsetof(VAAPIEncodeContext, codec_options_data) + \