2007-07-05 12:08:28 +03:00
/*
2016-04-27 19:45:23 +02:00
* MPEG - 1 / 2 encoder
2009-01-19 17:46:40 +02:00
* Copyright ( c ) 2000 , 2001 Fabrice Bellard
2007-07-05 12:08:28 +03:00
* Copyright ( c ) 2002 - 2004 Michael Niedermayer < michaelni @ gmx . at >
*
* This file is part of FFmpeg .
*
* FFmpeg is free software ; you can redistribute it and / or
* 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 .
*
* FFmpeg is distributed in the hope that it will be useful ,
* 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
* License along with FFmpeg ; if not , write to the Free Software
* Foundation , Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA
*/
/**
2010-04-20 17:45:34 +03:00
* @ file
2016-04-27 19:45:23 +02:00
* MPEG - 1 / 2 encoder
2007-07-05 12:08:28 +03:00
*/
2013-11-23 23:32:55 +03:00
# include <stdint.h>
2020-12-10 09:50:37 +02:00
# include "config.h"
2022-02-23 14:56:49 +02:00
# include "config_components.h"
2013-09-04 18:17:30 +03:00
# include "libavutil/attributes.h"
2013-09-06 14:00:47 +03:00
# include "libavutil/avassert.h"
2013-09-04 18:17:30 +03:00
# include "libavutil/log.h"
# include "libavutil/opt.h"
2020-11-24 15:39:47 +02:00
# include "libavutil/thread.h"
2013-09-06 14:00:47 +03:00
# include "libavutil/timecode.h"
2013-08-01 17:58:27 +03:00
# include "libavutil/stereo3d.h"
2007-07-05 12:08:28 +03:00
# include "avcodec.h"
2022-03-16 19:18:28 +02:00
# include "codec_internal.h"
2011-10-08 14:56:54 +03:00
# include "mathops.h"
2007-07-05 12:08:28 +03:00
# include "mpeg12.h"
# include "mpeg12data.h"
2022-02-01 09:16:43 +02:00
# include "mpeg12enc.h"
2022-09-30 01:20:07 +02:00
# include "mpeg12vlc.h"
2014-03-17 14:33:33 +03:00
# include "mpegutils.h"
2013-09-04 18:17:30 +03:00
# include "mpegvideo.h"
2022-09-30 03:06:25 +02:00
# include "mpegvideodata.h"
2022-01-28 02:14:42 +02:00
# include "mpegvideoenc.h"
2020-05-30 15:25:32 +02:00
# include "profiles.h"
2022-10-22 02:54:15 +02:00
# include "rl.h"
2007-07-05 12:08:28 +03:00
2020-12-10 09:50:37 +02:00
# if CONFIG_MPEG1VIDEO_ENCODER || CONFIG_MPEG2VIDEO_ENCODER
2013-09-04 18:17:30 +03:00
static const uint8_t svcd_scan_offset_placeholder [ ] = {
0x10 , 0x0E , 0x00 , 0x80 , 0x81 , 0x00 , 0x80 ,
0x81 , 0xff , 0xff , 0xff , 0xff , 0xff , 0xff ,
2007-07-05 12:08:28 +03:00
} ;
2016-01-05 15:41:04 +02:00
static uint8_t mv_penalty [ MAX_FCODE + 1 ] [ MAX_DMV * 2 + 1 ] ;
2013-09-04 18:17:30 +03:00
static uint8_t fcode_tab [ MAX_MV * 2 + 1 ] ;
2007-07-05 12:08:28 +03:00
2013-09-04 18:17:30 +03:00
static uint8_t uni_mpeg1_ac_vlc_len [ 64 * 64 * 2 ] ;
static uint8_t uni_mpeg2_ac_vlc_len [ 64 * 64 * 2 ] ;
2007-07-05 12:08:28 +03:00
2022-10-22 02:54:15 +02:00
static uint8_t mpeg12_max_level [ MAX_LEVEL + 1 ] ;
static uint8_t mpeg12_index_run [ MAX_RUN + 1 ] ;
2013-09-04 18:17:30 +03:00
/* simple include everything table for dc, first byte is bits
* number next 3 are code */
2007-07-05 12:08:28 +03:00
static uint32_t mpeg1_lum_dc_uni [ 512 ] ;
static uint32_t mpeg1_chr_dc_uni [ 512 ] ;
2021-12-22 18:28:20 +02:00
typedef struct MPEG12EncContext {
MpegEncContext mpeg ;
AVRational frame_rate_ext ;
2022-01-25 18:39:19 +02:00
unsigned frame_rate_index ;
2021-12-22 21:25:13 +02:00
2022-01-25 15:44:20 +02:00
int gop_picture_number ; ///< index of the first picture of a GOP based on fake_pic_num
2021-12-22 21:25:13 +02:00
int64_t timecode_frame_start ; ///< GOP timecode frame start number, in non drop frame format
AVTimecode tc ; ///< timecode context
char * tc_opt_str ; ///< timecode option string
int drop_frame_timecode ; ///< timecode is in drop frame format.
int scan_offset ; ///< reserve space for SVCD scan offset user data.
int a53_cc ;
int seq_disp_ext ;
int video_format ;
2022-10-16 22:02:42 +02:00
# define VIDEO_FORMAT_COMPONENT 0
# define VIDEO_FORMAT_PAL 1
# define VIDEO_FORMAT_NTSC 2
# define VIDEO_FORMAT_SECAM 3
# define VIDEO_FORMAT_MAC 4
# define VIDEO_FORMAT_UNSPECIFIED 5
2021-12-22 18:28:20 +02:00
} MPEG12EncContext ;
2019-02-09 00:10:12 +02:00
# define A53_MAX_CC_COUNT 0x1f
2020-12-10 09:50:37 +02:00
# endif /* CONFIG_MPEG1VIDEO_ENCODER || CONFIG_MPEG2VIDEO_ENCODER */
2019-02-09 00:10:12 +02:00
2022-10-22 01:33:11 +02:00
av_cold void ff_mpeg1_init_uni_ac_vlc ( const int8_t max_level [ ] ,
const uint8_t index_run [ ] ,
const uint16_t table_vlc [ ] [ 2 ] ,
uint8_t uni_ac_vlc_len [ ] )
2013-02-01 12:31:59 +03:00
{
2007-07-05 12:08:28 +03:00
int i ;
2013-09-04 18:17:30 +03:00
for ( i = 0 ; i < 128 ; i + + ) {
int level = i - 64 ;
2007-07-05 12:08:28 +03:00
int run ;
2011-11-04 01:56:37 +03:00
if ( ! level )
continue ;
2013-09-04 18:17:30 +03:00
for ( run = 0 ; run < 64 ; run + + ) {
2011-11-04 02:02:17 +03:00
int len , code ;
2013-09-04 18:17:30 +03:00
int alevel = FFABS ( level ) ;
2007-07-05 12:08:28 +03:00
2022-10-22 01:33:11 +02:00
if ( alevel > max_level [ run ] )
2013-09-04 18:17:30 +03:00
code = 111 ; /* rl->n */
2007-07-05 12:08:28 +03:00
else
2022-10-22 01:33:11 +02:00
code = index_run [ run ] + alevel - 1 ;
2007-07-05 12:08:28 +03:00
2013-09-04 18:17:30 +03:00
if ( code < 111 ) { /* rl->n */
/* length of VLC and sign */
2022-10-22 01:33:11 +02:00
len = table_vlc [ code ] [ 1 ] + 1 ;
2007-07-05 12:08:28 +03:00
} else {
2022-10-22 01:33:11 +02:00
len = table_vlc [ MPEG12_RL_NB_ELEMS ] [ 1 ] + 6 ;
2007-07-05 12:08:28 +03:00
2013-09-04 18:17:30 +03:00
if ( alevel < 128 )
2011-11-04 02:02:17 +03:00
len + = 8 ;
2013-09-04 18:17:30 +03:00
else
2011-11-04 02:02:17 +03:00
len + = 16 ;
2007-07-05 12:08:28 +03:00
}
2013-09-04 18:17:30 +03:00
uni_ac_vlc_len [ UNI_AC_ENC_INDEX ( run , i ) ] = len ;
2007-07-05 12:08:28 +03:00
}
}
}
2020-12-10 09:50:37 +02:00
# if CONFIG_MPEG1VIDEO_ENCODER || CONFIG_MPEG2VIDEO_ENCODER
2021-12-22 18:28:20 +02:00
static int find_frame_rate_index ( MPEG12EncContext * mpeg12 )
2013-09-04 18:17:30 +03:00
{
2021-12-22 18:28:20 +02:00
MpegEncContext * const s = & mpeg12 - > mpeg ;
2007-07-05 12:08:28 +03:00
int i ;
2013-09-06 14:00:47 +03:00
AVRational bestq = ( AVRational ) { 0 , 0 } ;
2012-10-04 16:25:30 +03:00
AVRational ext ;
AVRational target = av_inv_q ( s - > avctx - > time_base ) ;
2007-07-05 12:08:28 +03:00
2013-09-04 18:17:30 +03:00
for ( i = 1 ; i < 14 ; i + + ) {
if ( s - > avctx - > strict_std_compliance > FF_COMPLIANCE_UNOFFICIAL & &
i > = 9 )
break ;
2007-07-05 12:08:28 +03:00
2012-10-04 16:25:30 +03:00
for ( ext . num = 1 ; ext . num < = 4 ; ext . num + + ) {
for ( ext . den = 1 ; ext . den < = 32 ; ext . den + + ) {
2012-10-19 14:29:24 +03:00
AVRational q = av_mul_q ( ext , ff_mpeg12_frame_rate_tab [ i ] ) ;
2012-10-04 16:25:30 +03:00
2013-09-06 14:00:47 +03:00
if ( s - > codec_id ! = AV_CODEC_ID_MPEG2VIDEO & & ( ext . den ! = 1 | | ext . num ! = 1 ) )
2012-10-04 16:25:30 +03:00
continue ;
2013-09-06 14:00:47 +03:00
if ( av_gcd ( ext . den , ext . num ) ! = 1 )
2012-10-04 16:25:30 +03:00
continue ;
2013-09-06 14:00:47 +03:00
if ( bestq . num = = 0
2012-10-04 16:25:30 +03:00
| | av_nearer_q ( target , bestq , q ) < 0
2013-09-06 14:00:47 +03:00
| | ext . num = = 1 & & ext . den = = 1 & & av_nearer_q ( target , bestq , q ) = = 0 ) {
bestq = q ;
2022-01-25 18:39:19 +02:00
mpeg12 - > frame_rate_index = i ;
2021-12-22 18:28:20 +02:00
mpeg12 - > frame_rate_ext . num = ext . num ;
mpeg12 - > frame_rate_ext . den = ext . den ;
2012-10-04 16:25:30 +03:00
}
}
2007-07-05 12:08:28 +03:00
}
}
2013-09-04 18:17:30 +03:00
2013-09-06 14:00:47 +03:00
if ( av_cmp_q ( target , bestq ) )
2007-07-05 12:08:28 +03:00
return - 1 ;
else
return 0 ;
}
2008-03-21 05:11:20 +02:00
static av_cold int encode_init ( AVCodecContext * avctx )
2007-07-05 12:08:28 +03:00
{
2021-12-22 18:28:20 +02:00
MPEG12EncContext * const mpeg12 = avctx - > priv_data ;
MpegEncContext * const s = & mpeg12 - > mpeg ;
2020-05-07 12:55:11 +02:00
int ret ;
2021-04-06 15:03:19 +02:00
int max_size = avctx - > codec_id = = AV_CODEC_ID_MPEG2VIDEO ? 16383 : 4095 ;
if ( avctx - > width > max_size | | avctx - > height > max_size ) {
av_log ( avctx , AV_LOG_ERROR , " %s does not support resolutions above %dx%d \n " ,
CONFIG_SMALL ? avctx - > codec - > name : avctx - > codec - > long_name ,
max_size , max_size ) ;
return AVERROR ( EINVAL ) ;
}
2021-04-06 21:33:53 +02:00
if ( ( avctx - > width & 0xFFF ) = = 0 & & ( avctx - > height & 0xFFF ) = = 1 ) {
av_log ( avctx , AV_LOG_ERROR , " Width / Height is invalid for MPEG2 \n " ) ;
return AVERROR ( EINVAL ) ;
}
2007-07-05 12:08:28 +03:00
2021-04-06 21:33:53 +02:00
if ( avctx - > strict_std_compliance > FF_COMPLIANCE_UNOFFICIAL ) {
if ( ( avctx - > width & 0xFFF ) = = 0 | | ( avctx - > height & 0xFFF ) = = 0 ) {
av_log ( avctx , AV_LOG_ERROR , " Width or Height are not allowed to be multiples of 4096 \n "
" add '-strict %d' if you want to use them anyway. \n " , FF_COMPLIANCE_UNOFFICIAL ) ;
2020-05-07 12:55:11 +02:00
return AVERROR ( EINVAL ) ;
2007-07-05 12:08:28 +03:00
}
}
2023-09-02 14:57:41 +02:00
if ( avctx - > profile = = AV_PROFILE_UNKNOWN ) {
if ( avctx - > level ! = AV_LEVEL_UNKNOWN ) {
2007-07-05 12:08:28 +03:00
av_log ( avctx , AV_LOG_ERROR , " Set profile and level \n " ) ;
2020-05-07 12:55:11 +02:00
return AVERROR ( EINVAL ) ;
2007-07-05 12:08:28 +03:00
}
2013-09-04 18:17:30 +03:00
/* Main or 4:2:2 */
2023-09-02 14:57:41 +02:00
avctx - > profile = avctx - > pix_fmt = = AV_PIX_FMT_YUV420P ? AV_PROFILE_MPEG2_MAIN
: AV_PROFILE_MPEG2_422 ;
2007-07-05 12:08:28 +03:00
}
2023-09-02 14:57:41 +02:00
if ( avctx - > level = = AV_LEVEL_UNKNOWN ) {
if ( avctx - > profile = = AV_PROFILE_MPEG2_422 ) { /* 4:2:2 */
2013-09-04 18:17:30 +03:00
if ( avctx - > width < = 720 & & avctx - > height < = 608 )
avctx - > level = 5 ; /* Main */
else
avctx - > level = 2 ; /* High */
} else {
2023-09-02 14:57:41 +02:00
if ( avctx - > profile ! = AV_PROFILE_MPEG2_HIGH & &
2021-04-06 21:33:53 +02:00
avctx - > pix_fmt ! = AV_PIX_FMT_YUV420P ) {
2013-09-04 18:17:30 +03:00
av_log ( avctx , AV_LOG_ERROR ,
" Only High(1) and 4:2:2(0) profiles support 4:2:2 color sampling \n " ) ;
2020-05-07 12:55:11 +02:00
return AVERROR ( EINVAL ) ;
2007-07-05 12:08:28 +03:00
}
2013-09-04 18:17:30 +03:00
if ( avctx - > width < = 720 & & avctx - > height < = 576 )
avctx - > level = 8 ; /* Main */
else if ( avctx - > width < = 1440 )
avctx - > level = 6 ; /* High 1440 */
else
avctx - > level = 4 ; /* High */
2007-07-05 12:08:28 +03:00
}
}
2021-04-06 21:33:53 +02:00
if ( ( ret = ff_mpv_encode_init ( avctx ) ) < 0 )
return ret ;
2013-01-07 17:47:21 +03:00
2021-12-22 18:28:20 +02:00
if ( find_frame_rate_index ( mpeg12 ) < 0 ) {
2022-01-31 15:20:06 +02:00
if ( avctx - > strict_std_compliance > FF_COMPLIANCE_EXPERIMENTAL ) {
2021-04-06 21:33:53 +02:00
av_log ( avctx , AV_LOG_ERROR , " MPEG-1/2 does not support %d/%d fps \n " ,
avctx - > time_base . den , avctx - > time_base . num ) ;
2013-01-07 17:47:21 +03:00
return AVERROR ( EINVAL ) ;
2021-04-06 21:33:53 +02:00
} else {
av_log ( avctx , AV_LOG_INFO ,
" MPEG-1/2 does not support %d/%d fps, there may be AV sync issues \n " ,
avctx - > time_base . den , avctx - > time_base . num ) ;
2013-01-07 17:47:21 +03:00
}
}
2021-12-22 21:25:13 +02:00
if ( mpeg12 - > drop_frame_timecode )
mpeg12 - > tc . flags | = AV_TIMECODE_FLAG_DROPFRAME ;
2022-01-25 18:39:19 +02:00
if ( mpeg12 - > drop_frame_timecode & & mpeg12 - > frame_rate_index ! = 4 ) {
2013-09-04 18:17:30 +03:00
av_log ( avctx , AV_LOG_ERROR ,
" Drop frame time code only allowed with 1001/30000 fps \n " ) ;
2020-05-07 12:55:11 +02:00
return AVERROR ( EINVAL ) ;
2007-07-05 12:08:28 +03:00
}
2021-12-22 21:25:13 +02:00
if ( mpeg12 - > tc_opt_str ) {
2022-01-25 18:39:19 +02:00
AVRational rate = ff_mpeg12_frame_rate_tab [ mpeg12 - > frame_rate_index ] ;
2021-12-22 21:25:13 +02:00
int ret = av_timecode_init_from_string ( & mpeg12 - > tc , rate , mpeg12 - > tc_opt_str , s ) ;
2012-01-16 16:45:46 +03:00
if ( ret < 0 )
return ret ;
2021-12-22 21:25:13 +02:00
mpeg12 - > drop_frame_timecode = ! ! ( mpeg12 - > tc . flags & AV_TIMECODE_FLAG_DROPFRAME ) ;
mpeg12 - > timecode_frame_start = mpeg12 - > tc . start ;
2011-11-16 19:40:00 +03:00
} else {
2021-12-22 21:25:13 +02:00
mpeg12 - > timecode_frame_start = 0 ; // default is -1
2011-07-06 15:44:59 +03:00
}
2016-01-31 14:28:04 +02:00
2007-07-05 12:08:28 +03:00
return 0 ;
}
static void put_header ( MpegEncContext * s , int header )
{
2020-10-26 14:36:06 +02:00
align_put_bits ( & s - > pb ) ;
2013-09-04 18:17:30 +03:00
put_bits ( & s - > pb , 16 , header > > 16 ) ;
2008-07-13 17:27:48 +03:00
put_sbits ( & s - > pb , 16 , header ) ;
2007-07-05 12:08:28 +03:00
}
/* put sequence header if needed */
static void mpeg1_encode_sequence_header ( MpegEncContext * s )
{
2021-12-22 18:28:20 +02:00
MPEG12EncContext * const mpeg12 = ( MPEG12EncContext * ) s ;
2013-09-04 18:17:30 +03:00
unsigned int vbv_buffer_size , fps , v ;
2022-01-25 14:17:32 +02:00
int constraint_parameter_flag ;
2022-01-25 18:39:19 +02:00
AVRational framerate = ff_mpeg12_frame_rate_tab [ mpeg12 - > frame_rate_index ] ;
2013-09-04 18:17:30 +03:00
uint64_t time_code ;
2014-05-16 19:47:42 +03:00
int64_t best_aspect_error = INT64_MAX ;
AVRational aspect_ratio = s - > avctx - > sample_aspect_ratio ;
2022-01-25 14:08:50 +02:00
int aspect_ratio_info ;
2023-04-12 18:58:54 +02:00
if ( ! ( s - > current_picture . f - > flags & AV_FRAME_FLAG_KEY ) )
2022-01-25 14:08:50 +02:00
return ;
2007-07-05 12:08:28 +03:00
2014-05-16 19:47:42 +03:00
if ( aspect_ratio . num = = 0 | | aspect_ratio . den = = 0 )
aspect_ratio = ( AVRational ) { 1 , 1 } ; // pixel aspect 1.1 (VGA)
2007-07-05 12:08:28 +03:00
2022-01-25 14:17:32 +02:00
/* MPEG-1 header repeated every GOP */
put_header ( s , SEQ_START_CODE ) ;
2007-07-05 12:08:28 +03:00
2022-01-25 14:17:32 +02:00
put_sbits ( & s - > pb , 12 , s - > width & 0xFFF ) ;
put_sbits ( & s - > pb , 12 , s - > height & 0xFFF ) ;
2007-07-05 12:08:28 +03:00
2022-01-25 14:17:32 +02:00
for ( int i = 1 ; i < 15 ; i + + ) {
int64_t error = aspect_ratio . num * ( 1LL < < 32 ) / aspect_ratio . den ;
if ( s - > codec_id = = AV_CODEC_ID_MPEG1VIDEO | | i < = 1 )
error - = ( 1LL < < 32 ) / ff_mpeg1_aspect [ i ] ;
else
error - = ( 1LL < < 32 ) * ff_mpeg2_aspect [ i ] . num * s - > height / s - > width / ff_mpeg2_aspect [ i ] . den ;
2007-07-05 12:08:28 +03:00
2022-01-25 14:17:32 +02:00
error = FFABS ( error ) ;
2007-07-05 12:08:28 +03:00
2022-01-25 14:17:32 +02:00
if ( error - 2 < = best_aspect_error ) {
best_aspect_error = error ;
aspect_ratio_info = i ;
2013-09-04 18:17:30 +03:00
}
2022-01-25 14:17:32 +02:00
}
2007-07-05 12:08:28 +03:00
2022-01-25 14:17:32 +02:00
put_bits ( & s - > pb , 4 , aspect_ratio_info ) ;
2022-01-25 18:39:19 +02:00
put_bits ( & s - > pb , 4 , mpeg12 - > frame_rate_index ) ;
2007-07-05 12:08:28 +03:00
2022-01-25 14:17:32 +02:00
if ( s - > avctx - > rc_max_rate ) {
v = ( s - > avctx - > rc_max_rate + 399 ) / 400 ;
if ( v > 0x3ffff & & s - > codec_id = = AV_CODEC_ID_MPEG1VIDEO )
v = 0x3ffff ;
} else {
v = 0x3FFFF ;
}
if ( s - > avctx - > rc_buffer_size )
vbv_buffer_size = s - > avctx - > rc_buffer_size ;
else
/* VBV calculation: Scaled so that a VCD has the proper
* VBV size of 40 kilobytes */
vbv_buffer_size = ( ( 20 * s - > bit_rate ) / ( 1151929 / 2 ) ) * 8 * 1024 ;
vbv_buffer_size = ( vbv_buffer_size + 16383 ) / 16384 ;
put_sbits ( & s - > pb , 18 , v ) ;
put_bits ( & s - > pb , 1 , 1 ) ; // marker
put_sbits ( & s - > pb , 10 , vbv_buffer_size ) ;
constraint_parameter_flag =
s - > width < = 768 & &
s - > height < = 576 & &
s - > mb_width * s - > mb_height < = 396 & &
s - > mb_width * s - > mb_height * framerate . num < = 396 * 25 * framerate . den & &
framerate . num < = framerate . den * 30 & &
s - > avctx - > me_range & &
s - > avctx - > me_range < 128 & &
vbv_buffer_size < = 20 & &
v < = 1856000 / 400 & &
s - > codec_id = = AV_CODEC_ID_MPEG1VIDEO ;
put_bits ( & s - > pb , 1 , constraint_parameter_flag ) ;
ff_write_quant_matrix ( & s - > pb , s - > avctx - > intra_matrix ) ;
ff_write_quant_matrix ( & s - > pb , s - > avctx - > inter_matrix ) ;
if ( s - > codec_id = = AV_CODEC_ID_MPEG2VIDEO ) {
const AVFrameSideData * side_data ;
int width = s - > width ;
int height = s - > height ;
int use_seq_disp_ext ;
put_header ( s , EXT_START_CODE ) ;
put_bits ( & s - > pb , 4 , 1 ) ; // seq ext
2023-09-02 14:57:41 +02:00
put_bits ( & s - > pb , 1 , s - > avctx - > profile = = AV_PROFILE_MPEG2_422 ) ; // escx 1 for 4:2:2 profile
2022-01-25 14:17:32 +02:00
put_bits ( & s - > pb , 3 , s - > avctx - > profile ) ; // profile
put_bits ( & s - > pb , 4 , s - > avctx - > level ) ; // level
put_bits ( & s - > pb , 1 , s - > progressive_sequence ) ;
put_bits ( & s - > pb , 2 , s - > chroma_format ) ;
put_bits ( & s - > pb , 2 , s - > width > > 12 ) ;
put_bits ( & s - > pb , 2 , s - > height > > 12 ) ;
put_bits ( & s - > pb , 12 , v > > 18 ) ; // bitrate ext
put_bits ( & s - > pb , 1 , 1 ) ; // marker
put_bits ( & s - > pb , 8 , vbv_buffer_size > > 10 ) ; // vbv buffer ext
put_bits ( & s - > pb , 1 , s - > low_delay ) ;
put_bits ( & s - > pb , 2 , mpeg12 - > frame_rate_ext . num - 1 ) ; // frame_rate_ext_n
put_bits ( & s - > pb , 5 , mpeg12 - > frame_rate_ext . den - 1 ) ; // frame_rate_ext_d
side_data = av_frame_get_side_data ( s - > current_picture_ptr - > f , AV_FRAME_DATA_PANSCAN ) ;
if ( side_data ) {
const AVPanScan * pan_scan = ( AVPanScan * ) side_data - > data ;
if ( pan_scan - > width & & pan_scan - > height ) {
width = pan_scan - > width > > 4 ;
height = pan_scan - > height > > 4 ;
}
2013-09-04 18:17:30 +03:00
}
2007-07-05 12:08:28 +03:00
2022-01-25 14:17:32 +02:00
use_seq_disp_ext = ( width ! = s - > width | |
height ! = s - > height | |
s - > avctx - > color_primaries ! = AVCOL_PRI_UNSPECIFIED | |
s - > avctx - > color_trc ! = AVCOL_TRC_UNSPECIFIED | |
s - > avctx - > colorspace ! = AVCOL_SPC_UNSPECIFIED | |
mpeg12 - > video_format ! = VIDEO_FORMAT_UNSPECIFIED ) ;
2014-06-14 18:07:27 +03:00
2022-01-25 14:17:32 +02:00
if ( mpeg12 - > seq_disp_ext = = 1 | |
( mpeg12 - > seq_disp_ext = = - 1 & & use_seq_disp_ext ) ) {
2013-09-04 18:17:30 +03:00
put_header ( s , EXT_START_CODE ) ;
2022-01-25 14:17:32 +02:00
put_bits ( & s - > pb , 4 , 2 ) ; // sequence display extension
put_bits ( & s - > pb , 3 , mpeg12 - > video_format ) ; // video_format
put_bits ( & s - > pb , 1 , 1 ) ; // colour_description
put_bits ( & s - > pb , 8 , s - > avctx - > color_primaries ) ; // colour_primaries
put_bits ( & s - > pb , 8 , s - > avctx - > color_trc ) ; // transfer_characteristics
put_bits ( & s - > pb , 8 , s - > avctx - > colorspace ) ; // matrix_coefficients
put_bits ( & s - > pb , 14 , width ) ; // display_horizontal_size
put_bits ( & s - > pb , 1 , 1 ) ; // marker_bit
put_bits ( & s - > pb , 14 , height ) ; // display_vertical_size
put_bits ( & s - > pb , 3 , 0 ) ; // remaining 3 bits are zero padding
2007-07-05 12:08:28 +03:00
}
2022-01-25 14:17:32 +02:00
}
put_header ( s , GOP_START_CODE ) ;
put_bits ( & s - > pb , 1 , mpeg12 - > drop_frame_timecode ) ; // drop frame flag
/* time code: we must convert from the real frame rate to a
* fake MPEG frame rate in case of low frame rate */
fps = ( framerate . num + framerate . den / 2 ) / framerate . den ;
2023-01-28 00:13:05 +02:00
time_code = s - > current_picture_ptr - > coded_picture_number +
2022-01-25 14:17:32 +02:00
mpeg12 - > timecode_frame_start ;
2007-07-05 12:08:28 +03:00
2023-01-28 00:13:05 +02:00
mpeg12 - > gop_picture_number = s - > current_picture_ptr - > coded_picture_number ;
2022-01-25 14:17:32 +02:00
av_assert0 ( mpeg12 - > drop_frame_timecode = = ! ! ( mpeg12 - > tc . flags & AV_TIMECODE_FLAG_DROPFRAME ) ) ;
if ( mpeg12 - > drop_frame_timecode )
time_code = av_timecode_adjust_ntsc_framenum2 ( time_code , fps ) ;
put_bits ( & s - > pb , 5 , ( uint32_t ) ( ( time_code / ( fps * 3600 ) ) % 24 ) ) ;
put_bits ( & s - > pb , 6 , ( uint32_t ) ( ( time_code / ( fps * 60 ) ) % 60 ) ) ;
put_bits ( & s - > pb , 1 , 1 ) ;
put_bits ( & s - > pb , 6 , ( uint32_t ) ( ( time_code / fps ) % 60 ) ) ;
put_bits ( & s - > pb , 6 , ( uint32_t ) ( ( time_code % fps ) ) ) ;
2022-01-25 15:44:20 +02:00
put_bits ( & s - > pb , 1 , ! ! ( s - > avctx - > flags & AV_CODEC_FLAG_CLOSED_GOP ) | |
s - > intra_only | | ! mpeg12 - > gop_picture_number ) ;
2022-01-25 14:17:32 +02:00
put_bits ( & s - > pb , 1 , 0 ) ; // broken link
2007-07-05 12:08:28 +03:00
}
2013-09-04 18:17:30 +03:00
static inline void encode_mb_skip_run ( MpegEncContext * s , int run )
{
2007-07-05 12:08:28 +03:00
while ( run > = 33 ) {
put_bits ( & s - > pb , 11 , 0x008 ) ;
run - = 33 ;
}
2007-07-05 12:21:55 +03:00
put_bits ( & s - > pb , ff_mpeg12_mbAddrIncrTable [ run ] [ 1 ] ,
ff_mpeg12_mbAddrIncrTable [ run ] [ 0 ] ) ;
2007-07-05 12:08:28 +03:00
}
static av_always_inline void put_qscale ( MpegEncContext * s )
{
2015-09-18 00:37:28 +02:00
put_bits ( & s - > pb , 5 , s - > qscale ) ;
2007-07-05 12:08:28 +03:00
}
2013-09-04 18:17:30 +03:00
void ff_mpeg1_encode_slice_header ( MpegEncContext * s )
{
2012-08-07 23:45:46 +03:00
if ( s - > codec_id = = AV_CODEC_ID_MPEG2VIDEO & & s - > height > 2800 ) {
2009-09-20 17:08:44 +03:00
put_header ( s , SLICE_MIN_START_CODE + ( s - > mb_y & 127 ) ) ;
2013-09-04 18:17:30 +03:00
/* slice_vertical_position_extension */
put_bits ( & s - > pb , 3 , s - > mb_y > > 7 ) ;
2009-09-20 17:08:44 +03:00
} else {
2009-09-20 17:09:27 +03:00
put_header ( s , SLICE_MIN_START_CODE + s - > mb_y ) ;
2009-09-20 17:08:44 +03:00
}
2007-07-05 12:08:28 +03:00
put_qscale ( s ) ;
2013-09-04 18:17:30 +03:00
/* slice extra information */
put_bits ( & s - > pb , 1 , 0 ) ;
2007-07-05 12:08:28 +03:00
}
2023-01-22 02:06:56 +02:00
void ff_mpeg1_encode_picture_header ( MpegEncContext * s )
2007-07-05 12:08:28 +03:00
{
2021-12-22 21:25:13 +02:00
MPEG12EncContext * const mpeg12 = ( MPEG12EncContext * ) s ;
2013-08-01 17:58:27 +03:00
AVFrameSideData * side_data ;
2007-07-05 12:08:28 +03:00
mpeg1_encode_sequence_header ( s ) ;
2016-04-27 19:45:23 +02:00
/* MPEG-1 picture header */
2007-07-05 12:08:28 +03:00
put_header ( s , PICTURE_START_CODE ) ;
/* temporal reference */
// RAL: s->picture_number instead of s->fake_picture_number
2013-09-04 18:17:30 +03:00
put_bits ( & s - > pb , 10 ,
2022-01-25 15:44:20 +02:00
( s - > picture_number - mpeg12 - > gop_picture_number ) & 0x3ff ) ;
2007-07-05 12:08:28 +03:00
put_bits ( & s - > pb , 3 , s - > pict_type ) ;
2022-01-26 22:04:53 +02:00
s - > vbv_delay_pos = put_bytes_count ( & s - > pb , 0 ) ;
2013-09-04 18:17:30 +03:00
put_bits ( & s - > pb , 16 , 0xFFFF ) ; /* vbv_delay */
2007-07-05 12:08:28 +03:00
2013-09-04 18:17:30 +03:00
// RAL: Forward f_code also needed for B-frames
if ( s - > pict_type = = AV_PICTURE_TYPE_P | |
s - > pict_type = = AV_PICTURE_TYPE_B ) {
put_bits ( & s - > pb , 1 , 0 ) ; /* half pel coordinates */
if ( s - > codec_id = = AV_CODEC_ID_MPEG1VIDEO )
put_bits ( & s - > pb , 3 , s - > f_code ) ; /* forward_f_code */
2007-07-05 12:08:28 +03:00
else
2013-09-04 18:17:30 +03:00
put_bits ( & s - > pb , 3 , 7 ) ; /* forward_f_code */
2007-07-05 12:08:28 +03:00
}
2013-09-04 18:17:30 +03:00
// RAL: Backward f_code necessary for B-frames
2011-04-28 02:40:44 +03:00
if ( s - > pict_type = = AV_PICTURE_TYPE_B ) {
2013-09-04 18:17:30 +03:00
put_bits ( & s - > pb , 1 , 0 ) ; /* half pel coordinates */
if ( s - > codec_id = = AV_CODEC_ID_MPEG1VIDEO )
put_bits ( & s - > pb , 3 , s - > b_code ) ; /* backward_f_code */
2007-07-05 12:08:28 +03:00
else
2013-09-04 18:17:30 +03:00
put_bits ( & s - > pb , 3 , 7 ) ; /* backward_f_code */
2007-07-05 12:08:28 +03:00
}
2013-09-04 18:17:30 +03:00
put_bits ( & s - > pb , 1 , 0 ) ; /* extra bit picture */
2007-07-05 12:08:28 +03:00
s - > frame_pred_frame_dct = 1 ;
2013-09-04 18:17:30 +03:00
if ( s - > codec_id = = AV_CODEC_ID_MPEG2VIDEO ) {
2007-07-05 12:08:28 +03:00
put_header ( s , EXT_START_CODE ) ;
2013-09-04 18:17:30 +03:00
put_bits ( & s - > pb , 4 , 8 ) ; /* pic ext */
if ( s - > pict_type = = AV_PICTURE_TYPE_P | |
s - > pict_type = = AV_PICTURE_TYPE_B ) {
2007-07-05 12:08:28 +03:00
put_bits ( & s - > pb , 4 , s - > f_code ) ;
put_bits ( & s - > pb , 4 , s - > f_code ) ;
2013-09-04 18:17:30 +03:00
} else {
2007-07-05 12:08:28 +03:00
put_bits ( & s - > pb , 8 , 255 ) ;
}
2011-04-28 02:40:44 +03:00
if ( s - > pict_type = = AV_PICTURE_TYPE_B ) {
2007-07-05 12:08:28 +03:00
put_bits ( & s - > pb , 4 , s - > b_code ) ;
put_bits ( & s - > pb , 4 , s - > b_code ) ;
2013-09-04 18:17:30 +03:00
} else {
2007-07-05 12:08:28 +03:00
put_bits ( & s - > pb , 8 , 255 ) ;
}
put_bits ( & s - > pb , 2 , s - > intra_dc_precision ) ;
2012-07-09 16:38:16 +03:00
av_assert0 ( s - > picture_structure = = PICT_FRAME ) ;
2007-07-05 12:08:28 +03:00
put_bits ( & s - > pb , 2 , s - > picture_structure ) ;
2013-09-04 18:17:30 +03:00
if ( s - > progressive_sequence )
put_bits ( & s - > pb , 1 , 0 ) ; /* no repeat */
else
2023-04-11 20:02:14 +02:00
put_bits ( & s - > pb , 1 , ! ! ( s - > current_picture_ptr - > f - > flags & AV_FRAME_FLAG_TOP_FIELD_FIRST ) ) ;
2013-09-04 18:17:30 +03:00
/* XXX: optimize the generation of this flag with entropy measures */
2007-07-05 12:08:28 +03:00
s - > frame_pred_frame_dct = s - > progressive_sequence ;
put_bits ( & s - > pb , 1 , s - > frame_pred_frame_dct ) ;
put_bits ( & s - > pb , 1 , s - > concealment_motion_vectors ) ;
put_bits ( & s - > pb , 1 , s - > q_scale_type ) ;
put_bits ( & s - > pb , 1 , s - > intra_vlc_format ) ;
put_bits ( & s - > pb , 1 , s - > alternate_scan ) ;
put_bits ( & s - > pb , 1 , s - > repeat_first_field ) ;
s - > progressive_frame = s - > progressive_sequence ;
2013-09-04 18:17:30 +03:00
/* chroma_420_type */
put_bits ( & s - > pb , 1 , s - > chroma_format = =
CHROMA_420 ? s - > progressive_frame : 0 ) ;
2007-07-05 12:08:28 +03:00
put_bits ( & s - > pb , 1 , s - > progressive_frame ) ;
2013-09-04 18:17:30 +03:00
put_bits ( & s - > pb , 1 , 0 ) ; /* composite_display_flag */
2007-07-05 12:08:28 +03:00
}
2021-12-22 21:25:13 +02:00
if ( mpeg12 - > scan_offset ) {
2007-07-05 12:08:28 +03:00
int i ;
put_header ( s , USER_START_CODE ) ;
2013-09-04 18:17:30 +03:00
for ( i = 0 ; i < sizeof ( svcd_scan_offset_placeholder ) ; i + + )
2007-07-05 12:08:28 +03:00
put_bits ( & s - > pb , 8 , svcd_scan_offset_placeholder [ i ] ) ;
}
2014-03-31 20:46:29 +03:00
side_data = av_frame_get_side_data ( s - > current_picture_ptr - > f ,
2013-08-01 17:58:27 +03:00
AV_FRAME_DATA_STEREO3D ) ;
if ( side_data ) {
AVStereo3D * stereo = ( AVStereo3D * ) side_data - > data ;
uint8_t fpa_type ;
switch ( stereo - > type ) {
case AV_STEREO3D_SIDEBYSIDE :
fpa_type = 0x03 ;
break ;
case AV_STEREO3D_TOPBOTTOM :
fpa_type = 0x04 ;
break ;
case AV_STEREO3D_2D :
fpa_type = 0x08 ;
break ;
case AV_STEREO3D_SIDEBYSIDE_QUINCUNX :
fpa_type = 0x23 ;
break ;
default :
fpa_type = 0 ;
break ;
}
if ( fpa_type ! = 0 ) {
put_header ( s , USER_START_CODE ) ;
put_bits ( & s - > pb , 8 , ' J ' ) ; // S3D_video_format_signaling_identifier
put_bits ( & s - > pb , 8 , ' P ' ) ;
put_bits ( & s - > pb , 8 , ' 3 ' ) ;
put_bits ( & s - > pb , 8 , ' D ' ) ;
put_bits ( & s - > pb , 8 , 0x03 ) ; // S3D_video_format_length
put_bits ( & s - > pb , 1 , 1 ) ; // reserved_bit
put_bits ( & s - > pb , 7 , fpa_type ) ; // S3D_video_format_type
put_bits ( & s - > pb , 8 , 0x04 ) ; // reserved_data[0]
put_bits ( & s - > pb , 8 , 0xFF ) ; // reserved_data[1]
}
}
2007-07-05 12:08:28 +03:00
2021-12-22 21:25:13 +02:00
if ( CONFIG_MPEG2VIDEO_ENCODER & & mpeg12 - > a53_cc ) {
2019-02-09 00:10:12 +02:00
side_data = av_frame_get_side_data ( s - > current_picture_ptr - > f ,
AV_FRAME_DATA_A53_CC ) ;
if ( side_data ) {
if ( side_data - > size < = A53_MAX_CC_COUNT * 3 & & side_data - > size % 3 = = 0 ) {
int i = 0 ;
put_header ( s , USER_START_CODE ) ;
put_bits ( & s - > pb , 8 , ' G ' ) ; // user_identifier
put_bits ( & s - > pb , 8 , ' A ' ) ;
put_bits ( & s - > pb , 8 , ' 9 ' ) ;
put_bits ( & s - > pb , 8 , ' 4 ' ) ;
put_bits ( & s - > pb , 8 , 3 ) ; // user_data_type_code
put_bits ( & s - > pb , 8 ,
( side_data - > size / 3 & A53_MAX_CC_COUNT ) | 0x40 ) ; // flags, cc_count
put_bits ( & s - > pb , 8 , 0xff ) ; // em_data
for ( i = 0 ; i < side_data - > size ; i + + )
put_bits ( & s - > pb , 8 , side_data - > data [ i ] ) ;
put_bits ( & s - > pb , 8 , 0xff ) ; // marker_bits
} else {
av_log ( s - > avctx , AV_LOG_WARNING ,
2021-04-14 14:59:32 +02:00
" Closed Caption size (% " SIZE_SPECIFIER " ) can not exceed "
" 93 bytes and must be a multiple of 3 \n " , side_data - > size ) ;
2019-02-09 00:10:12 +02:00
}
}
}
2013-09-04 18:17:30 +03:00
s - > mb_y = 0 ;
2007-07-05 12:08:28 +03:00
ff_mpeg1_encode_slice_header ( s ) ;
}
static inline void put_mb_modes ( MpegEncContext * s , int n , int bits ,
int has_mv , int field_motion )
{
put_bits ( & s - > pb , n , bits ) ;
if ( ! s - > frame_pred_frame_dct ) {
if ( has_mv )
2013-09-04 18:17:30 +03:00
/* motion_type: frame/field */
put_bits ( & s - > pb , 2 , 2 - field_motion ) ;
2007-07-05 12:08:28 +03:00
put_bits ( & s - > pb , 1 , s - > interlaced_dct ) ;
}
}
2013-09-04 18:17:29 +03:00
// RAL: Parameter added: f_or_b_code
static void mpeg1_encode_motion ( MpegEncContext * s , int val , int f_or_b_code )
{
if ( val = = 0 ) {
2021-12-25 04:25:02 +02:00
/* zero vector, corresponds to ff_mpeg12_mbMotionVectorTable[0] */
put_bits ( & s - > pb , 1 , 0x01 ) ;
2013-09-04 18:17:29 +03:00
} else {
int code , sign , bits ;
int bit_size = f_or_b_code - 1 ;
2013-09-04 18:17:30 +03:00
int range = 1 < < bit_size ;
2013-09-04 18:17:29 +03:00
/* modulo encoding */
val = sign_extend ( val , 5 + bit_size ) ;
if ( val > = 0 ) {
val - - ;
code = ( val > > bit_size ) + 1 ;
bits = val & ( range - 1 ) ;
sign = 0 ;
} else {
val = - val ;
val - - ;
code = ( val > > bit_size ) + 1 ;
bits = val & ( range - 1 ) ;
sign = 1 ;
}
2013-09-06 13:34:09 +03:00
av_assert2 ( code > 0 & & code < = 16 ) ;
2013-09-04 18:17:29 +03:00
put_bits ( & s - > pb ,
ff_mpeg12_mbMotionVectorTable [ code ] [ 1 ] ,
ff_mpeg12_mbMotionVectorTable [ code ] [ 0 ] ) ;
put_bits ( & s - > pb , 1 , sign ) ;
2013-09-04 18:17:30 +03:00
if ( bit_size > 0 )
2013-09-04 18:17:29 +03:00
put_bits ( & s - > pb , bit_size , bits ) ;
}
}
static inline void encode_dc ( MpegEncContext * s , int diff , int component )
{
2014-11-04 16:36:57 +02:00
unsigned int diff_u = diff + 255 ;
if ( diff_u > = 511 ) {
2013-09-04 18:17:29 +03:00
int index ;
2013-09-04 18:17:30 +03:00
if ( diff < 0 ) {
index = av_log2_16bit ( - 2 * diff ) ;
2013-09-04 18:17:29 +03:00
diff - - ;
2013-09-04 18:17:30 +03:00
} else {
index = av_log2_16bit ( 2 * diff ) ;
2013-09-04 18:17:29 +03:00
}
2013-09-04 18:17:30 +03:00
if ( component = = 0 )
put_bits ( & s - > pb ,
ff_mpeg12_vlc_dc_lum_bits [ index ] + index ,
( ff_mpeg12_vlc_dc_lum_code [ index ] < < index ) +
2015-04-22 03:28:21 +02:00
av_mod_uintp2 ( diff , index ) ) ;
2013-09-04 18:17:30 +03:00
else
put_bits ( & s - > pb ,
ff_mpeg12_vlc_dc_chroma_bits [ index ] + index ,
( ff_mpeg12_vlc_dc_chroma_code [ index ] < < index ) +
2015-04-22 03:28:21 +02:00
av_mod_uintp2 ( diff , index ) ) ;
2013-09-04 18:17:29 +03:00
} else {
2013-09-04 18:17:30 +03:00
if ( component = = 0 )
put_bits ( & s - > pb ,
mpeg1_lum_dc_uni [ diff + 255 ] & 0xFF ,
mpeg1_lum_dc_uni [ diff + 255 ] > > 8 ) ;
else
put_bits ( & s - > pb ,
mpeg1_chr_dc_uni [ diff + 255 ] & 0xFF ,
mpeg1_chr_dc_uni [ diff + 255 ] > > 8 ) ;
2013-09-04 18:17:29 +03:00
}
}
2013-09-04 18:17:30 +03:00
static void mpeg1_encode_block ( MpegEncContext * s , int16_t * block , int n )
2013-09-04 18:17:29 +03:00
{
int alevel , level , last_non_zero , dc , diff , i , j , run , last_index , sign ;
int code , component ;
2022-10-21 23:49:55 +02:00
const uint16_t ( * table_vlc ) [ 2 ] = ff_mpeg1_vlc_table ;
2013-09-04 18:17:29 +03:00
last_index = s - > block_last_index [ n ] ;
/* DC coef */
if ( s - > mb_intra ) {
2013-09-04 18:17:30 +03:00
component = ( n < = 3 ? 0 : ( n & 1 ) + 1 ) ;
dc = block [ 0 ] ; /* overflow is impossible */
diff = dc - s - > last_dc [ component ] ;
2013-09-04 18:17:29 +03:00
encode_dc ( s , diff , component ) ;
s - > last_dc [ component ] = dc ;
i = 1 ;
if ( s - > intra_vlc_format )
2022-10-21 23:49:55 +02:00
table_vlc = ff_mpeg2_vlc_table ;
2013-09-04 18:17:29 +03:00
} else {
2013-09-04 18:17:30 +03:00
/* encode the first coefficient: needs to be done here because
* it is handled slightly differently */
2013-09-04 18:17:29 +03:00
level = block [ 0 ] ;
if ( abs ( level ) = = 1 ) {
2013-09-04 18:17:30 +03:00
code = ( ( uint32_t ) level > > 31 ) ; /* the sign bit */
put_bits ( & s - > pb , 2 , code | 0x02 ) ;
i = 1 ;
2013-09-04 18:17:29 +03:00
} else {
2013-09-04 18:17:30 +03:00
i = 0 ;
2013-09-04 18:17:29 +03:00
last_non_zero = - 1 ;
goto next_coef ;
}
}
/* now quantify & encode AC coefs */
last_non_zero = i - 1 ;
2013-09-04 18:17:30 +03:00
for ( ; i < = last_index ; i + + ) {
j = s - > intra_scantable . permutated [ i ] ;
2013-09-04 18:17:29 +03:00
level = block [ j ] ;
2013-09-04 18:17:30 +03:00
next_coef :
2013-09-04 18:17:29 +03:00
/* encode using VLC */
if ( level ! = 0 ) {
run = i - last_non_zero - 1 ;
2013-09-04 18:17:30 +03:00
alevel = level ;
2013-09-04 18:17:29 +03:00
MASK_ABS ( sign , alevel ) ;
2013-09-04 18:17:30 +03:00
sign & = 1 ;
2013-09-04 18:17:29 +03:00
2022-10-22 02:54:15 +02:00
if ( alevel < = mpeg12_max_level [ run ] ) {
code = mpeg12_index_run [ run ] + alevel - 1 ;
2013-09-04 18:17:30 +03:00
/* store the VLC & sign at once */
put_bits ( & s - > pb , table_vlc [ code ] [ 1 ] + 1 ,
( table_vlc [ code ] [ 0 ] < < 1 ) + sign ) ;
2013-09-04 18:17:29 +03:00
} else {
2021-12-25 04:25:02 +02:00
/* Escape seems to be pretty rare <5% so I do not optimize it;
* the following value is the common escape value for both
* possible tables ( i . e . table_vlc [ 111 ] ) . */
put_bits ( & s - > pb , 6 , 0x01 ) ;
2013-09-04 18:17:29 +03:00
/* escape: only clip in this case */
put_bits ( & s - > pb , 6 , run ) ;
2013-09-04 18:17:30 +03:00
if ( s - > codec_id = = AV_CODEC_ID_MPEG1VIDEO ) {
2013-09-04 18:17:29 +03:00
if ( alevel < 128 ) {
put_sbits ( & s - > pb , 8 , level ) ;
} else {
2013-09-04 18:17:30 +03:00
if ( level < 0 )
2013-09-04 18:17:29 +03:00
put_bits ( & s - > pb , 16 , 0x8001 + level + 255 ) ;
2013-09-04 18:17:30 +03:00
else
2013-09-04 18:17:29 +03:00
put_sbits ( & s - > pb , 16 , level ) ;
}
2013-09-04 18:17:30 +03:00
} else {
2013-09-04 18:17:29 +03:00
put_sbits ( & s - > pb , 12 , level ) ;
}
}
last_non_zero = i ;
}
}
/* end of block */
put_bits ( & s - > pb , table_vlc [ 112 ] [ 1 ] , table_vlc [ 112 ] [ 0 ] ) ;
}
2007-07-05 12:08:28 +03:00
static av_always_inline void mpeg1_encode_mb_internal ( MpegEncContext * s ,
2014-04-28 02:11:04 +03:00
int16_t block [ 8 ] [ 64 ] ,
2013-09-04 18:17:30 +03:00
int motion_x , int motion_y ,
2021-12-25 03:07:53 +02:00
int mb_block_count ,
int chroma_y_shift )
2007-07-05 12:08:28 +03:00
{
2021-12-25 03:38:59 +02:00
/* MPEG-1 is always 420. */
# define IS_MPEG1(s) (chroma_y_shift == 1 && (s)->codec_id == AV_CODEC_ID_MPEG1VIDEO)
2007-07-05 12:08:28 +03:00
int i , cbp ;
2013-09-04 18:17:30 +03:00
const int mb_x = s - > mb_x ;
const int mb_y = s - > mb_y ;
const int first_mb = mb_x = = s - > resync_mb_x & & mb_y = = s - > resync_mb_y ;
2007-07-05 12:08:28 +03:00
/* compute cbp */
cbp = 0 ;
2013-09-04 18:17:30 +03:00
for ( i = 0 ; i < mb_block_count ; i + + )
2007-07-05 12:08:28 +03:00
if ( s - > block_last_index [ i ] > = 0 )
cbp | = 1 < < ( mb_block_count - 1 - i ) ;
if ( cbp = = 0 & & ! first_mb & & s - > mv_type = = MV_TYPE_16X16 & &
2013-09-04 18:17:30 +03:00
( mb_x ! = s - > mb_width - 1 | |
2021-12-25 03:38:59 +02:00
( mb_y ! = s - > end_mb_y - 1 & & IS_MPEG1 ( s ) ) ) & &
2011-04-28 02:40:44 +03:00
( ( s - > pict_type = = AV_PICTURE_TYPE_P & & ( motion_x | motion_y ) = = 0 ) | |
2013-09-04 18:17:30 +03:00
( s - > pict_type = = AV_PICTURE_TYPE_B & & s - > mv_dir = = s - > last_mv_dir & &
( ( ( s - > mv_dir & MV_DIR_FORWARD )
? ( ( s - > mv [ 0 ] [ 0 ] [ 0 ] - s - > last_mv [ 0 ] [ 0 ] [ 0 ] ) |
( s - > mv [ 0 ] [ 0 ] [ 1 ] - s - > last_mv [ 0 ] [ 0 ] [ 1 ] ) ) : 0 ) |
( ( s - > mv_dir & MV_DIR_BACKWARD )
? ( ( s - > mv [ 1 ] [ 0 ] [ 0 ] - s - > last_mv [ 1 ] [ 0 ] [ 0 ] ) |
( s - > mv [ 1 ] [ 0 ] [ 1 ] - s - > last_mv [ 1 ] [ 0 ] [ 1 ] ) ) : 0 ) ) = = 0 ) ) ) {
2007-07-05 12:08:28 +03:00
s - > mb_skip_run + + ;
s - > qscale - = s - > dquant ;
s - > skip_count + + ;
s - > misc_bits + + ;
s - > last_bits + + ;
2013-09-04 18:17:30 +03:00
if ( s - > pict_type = = AV_PICTURE_TYPE_P ) {
s - > last_mv [ 0 ] [ 0 ] [ 0 ] =
s - > last_mv [ 0 ] [ 0 ] [ 1 ] =
s - > last_mv [ 0 ] [ 1 ] [ 0 ] =
s - > last_mv [ 0 ] [ 1 ] [ 1 ] = 0 ;
2007-07-05 12:08:28 +03:00
}
} else {
2013-09-04 18:17:30 +03:00
if ( first_mb ) {
2012-07-09 16:38:16 +03:00
av_assert0 ( s - > mb_skip_run = = 0 ) ;
2007-07-05 12:08:28 +03:00
encode_mb_skip_run ( s , s - > mb_x ) ;
2013-09-04 18:17:30 +03:00
} else {
2007-07-05 12:08:28 +03:00
encode_mb_skip_run ( s , s - > mb_skip_run ) ;
}
2011-04-28 02:40:44 +03:00
if ( s - > pict_type = = AV_PICTURE_TYPE_I ) {
2013-09-04 18:17:30 +03:00
if ( s - > dquant & & cbp ) {
/* macroblock_type: macroblock_quant = 1 */
put_mb_modes ( s , 2 , 1 , 0 , 0 ) ;
2007-07-05 12:08:28 +03:00
put_qscale ( s ) ;
2013-09-04 18:17:30 +03:00
} else {
/* macroblock_type: macroblock_quant = 0 */
put_mb_modes ( s , 1 , 1 , 0 , 0 ) ;
2007-07-05 12:08:28 +03:00
s - > qscale - = s - > dquant ;
}
2013-09-04 18:17:30 +03:00
s - > misc_bits + = get_bits_diff ( s ) ;
2007-07-05 12:08:28 +03:00
s - > i_count + + ;
} else if ( s - > mb_intra ) {
2013-09-04 18:17:30 +03:00
if ( s - > dquant & & cbp ) {
2007-07-05 12:08:28 +03:00
put_mb_modes ( s , 6 , 0x01 , 0 , 0 ) ;
put_qscale ( s ) ;
2013-09-04 18:17:30 +03:00
} else {
2007-07-05 12:08:28 +03:00
put_mb_modes ( s , 5 , 0x03 , 0 , 0 ) ;
s - > qscale - = s - > dquant ;
}
2013-09-04 18:17:30 +03:00
s - > misc_bits + = get_bits_diff ( s ) ;
2007-07-05 12:08:28 +03:00
s - > i_count + + ;
memset ( s - > last_mv , 0 , sizeof ( s - > last_mv ) ) ;
2011-04-28 02:40:44 +03:00
} else if ( s - > pict_type = = AV_PICTURE_TYPE_P ) {
2013-09-04 18:17:30 +03:00
if ( s - > mv_type = = MV_TYPE_16X16 ) {
2007-07-05 12:08:28 +03:00
if ( cbp ! = 0 ) {
2013-09-04 18:17:30 +03:00
if ( ( motion_x | motion_y ) = = 0 ) {
if ( s - > dquant ) {
/* macroblock_pattern & quant */
put_mb_modes ( s , 5 , 1 , 0 , 0 ) ;
2007-07-05 12:08:28 +03:00
put_qscale ( s ) ;
2013-09-04 18:17:30 +03:00
} else {
/* macroblock_pattern only */
put_mb_modes ( s , 2 , 1 , 0 , 0 ) ;
2007-07-05 12:08:28 +03:00
}
2013-09-04 18:17:30 +03:00
s - > misc_bits + = get_bits_diff ( s ) ;
2007-07-05 12:08:28 +03:00
} else {
2013-09-04 18:17:30 +03:00
if ( s - > dquant ) {
put_mb_modes ( s , 5 , 2 , 1 , 0 ) ; /* motion + cbp */
2007-07-05 12:08:28 +03:00
put_qscale ( s ) ;
2013-09-04 18:17:30 +03:00
} else {
put_mb_modes ( s , 1 , 1 , 1 , 0 ) ; /* motion + cbp */
2007-07-05 12:08:28 +03:00
}
2013-09-04 18:17:30 +03:00
s - > misc_bits + = get_bits_diff ( s ) ;
// RAL: f_code parameter added
mpeg1_encode_motion ( s ,
motion_x - s - > last_mv [ 0 ] [ 0 ] [ 0 ] ,
s - > f_code ) ;
// RAL: f_code parameter added
mpeg1_encode_motion ( s ,
motion_y - s - > last_mv [ 0 ] [ 0 ] [ 1 ] ,
s - > f_code ) ;
s - > mv_bits + = get_bits_diff ( s ) ;
2007-07-05 12:08:28 +03:00
}
} else {
2013-09-04 18:17:30 +03:00
put_bits ( & s - > pb , 3 , 1 ) ; /* motion only */
2007-07-05 12:08:28 +03:00
if ( ! s - > frame_pred_frame_dct )
2013-09-04 18:17:30 +03:00
put_bits ( & s - > pb , 2 , 2 ) ; /* motion_type: frame */
s - > misc_bits + = get_bits_diff ( s ) ;
// RAL: f_code parameter added
mpeg1_encode_motion ( s ,
motion_x - s - > last_mv [ 0 ] [ 0 ] [ 0 ] ,
s - > f_code ) ;
// RAL: f_code parameter added
mpeg1_encode_motion ( s ,
motion_y - s - > last_mv [ 0 ] [ 0 ] [ 1 ] ,
s - > f_code ) ;
s - > qscale - = s - > dquant ;
s - > mv_bits + = get_bits_diff ( s ) ;
2007-07-05 12:08:28 +03:00
}
2013-09-04 18:17:30 +03:00
s - > last_mv [ 0 ] [ 1 ] [ 0 ] = s - > last_mv [ 0 ] [ 0 ] [ 0 ] = motion_x ;
s - > last_mv [ 0 ] [ 1 ] [ 1 ] = s - > last_mv [ 0 ] [ 0 ] [ 1 ] = motion_y ;
} else {
2012-07-09 16:38:16 +03:00
av_assert2 ( ! s - > frame_pred_frame_dct & & s - > mv_type = = MV_TYPE_FIELD ) ;
2007-07-05 12:08:28 +03:00
if ( cbp ) {
2013-09-04 18:17:30 +03:00
if ( s - > dquant ) {
put_mb_modes ( s , 5 , 2 , 1 , 1 ) ; /* motion + cbp */
2007-07-05 12:08:28 +03:00
put_qscale ( s ) ;
2013-09-04 18:17:30 +03:00
} else {
put_mb_modes ( s , 1 , 1 , 1 , 1 ) ; /* motion + cbp */
2007-07-05 12:08:28 +03:00
}
} else {
2013-09-04 18:17:30 +03:00
put_bits ( & s - > pb , 3 , 1 ) ; /* motion only */
put_bits ( & s - > pb , 2 , 1 ) ; /* motion_type: field */
2007-07-05 12:08:28 +03:00
s - > qscale - = s - > dquant ;
}
2013-09-04 18:17:30 +03:00
s - > misc_bits + = get_bits_diff ( s ) ;
for ( i = 0 ; i < 2 ; i + + ) {
2007-07-05 12:08:28 +03:00
put_bits ( & s - > pb , 1 , s - > field_select [ 0 ] [ i ] ) ;
2013-09-04 18:17:30 +03:00
mpeg1_encode_motion ( s ,
s - > mv [ 0 ] [ i ] [ 0 ] - s - > last_mv [ 0 ] [ i ] [ 0 ] ,
s - > f_code ) ;
mpeg1_encode_motion ( s ,
s - > mv [ 0 ] [ i ] [ 1 ] - ( s - > last_mv [ 0 ] [ i ] [ 1 ] > > 1 ) ,
s - > f_code ) ;
s - > last_mv [ 0 ] [ i ] [ 0 ] = s - > mv [ 0 ] [ i ] [ 0 ] ;
s - > last_mv [ 0 ] [ i ] [ 1 ] = 2 * s - > mv [ 0 ] [ i ] [ 1 ] ;
2007-07-05 12:08:28 +03:00
}
2013-09-04 18:17:30 +03:00
s - > mv_bits + = get_bits_diff ( s ) ;
2007-07-05 12:08:28 +03:00
}
2013-09-04 18:17:30 +03:00
if ( cbp ) {
2021-12-25 03:07:53 +02:00
if ( chroma_y_shift ) {
2013-09-04 18:17:30 +03:00
put_bits ( & s - > pb ,
ff_mpeg12_mbPatTable [ cbp ] [ 1 ] ,
ff_mpeg12_mbPatTable [ cbp ] [ 0 ] ) ;
2007-07-05 12:08:28 +03:00
} else {
2013-09-04 18:17:30 +03:00
put_bits ( & s - > pb ,
ff_mpeg12_mbPatTable [ cbp > > 2 ] [ 1 ] ,
ff_mpeg12_mbPatTable [ cbp > > 2 ] [ 0 ] ) ;
2008-07-13 17:27:48 +03:00
put_sbits ( & s - > pb , 2 , cbp ) ;
2007-07-05 12:08:28 +03:00
}
}
2013-09-04 18:17:30 +03:00
} else {
if ( s - > mv_type = = MV_TYPE_16X16 ) {
if ( cbp ) { // With coded bloc pattern
2007-07-05 12:08:28 +03:00
if ( s - > dquant ) {
2013-09-04 18:17:30 +03:00
if ( s - > mv_dir = = MV_DIR_FORWARD )
2007-07-05 12:08:28 +03:00
put_mb_modes ( s , 6 , 3 , 1 , 0 ) ;
else
2013-09-04 18:17:30 +03:00
put_mb_modes ( s , 8 - s - > mv_dir , 2 , 1 , 0 ) ;
2007-07-05 12:08:28 +03:00
put_qscale ( s ) ;
} else {
2013-09-04 18:17:30 +03:00
put_mb_modes ( s , 5 - s - > mv_dir , 3 , 1 , 0 ) ;
2007-07-05 12:08:28 +03:00
}
2013-09-04 18:17:30 +03:00
} else { // No coded bloc pattern
put_bits ( & s - > pb , 5 - s - > mv_dir , 2 ) ;
2007-07-05 12:08:28 +03:00
if ( ! s - > frame_pred_frame_dct )
put_bits ( & s - > pb , 2 , 2 ) ; /* motion_type: frame */
s - > qscale - = s - > dquant ;
}
s - > misc_bits + = get_bits_diff ( s ) ;
2013-09-04 18:17:30 +03:00
if ( s - > mv_dir & MV_DIR_FORWARD ) {
mpeg1_encode_motion ( s ,
s - > mv [ 0 ] [ 0 ] [ 0 ] - s - > last_mv [ 0 ] [ 0 ] [ 0 ] ,
s - > f_code ) ;
mpeg1_encode_motion ( s ,
s - > mv [ 0 ] [ 0 ] [ 1 ] - s - > last_mv [ 0 ] [ 0 ] [ 1 ] ,
s - > f_code ) ;
s - > last_mv [ 0 ] [ 0 ] [ 0 ] =
s - > last_mv [ 0 ] [ 1 ] [ 0 ] = s - > mv [ 0 ] [ 0 ] [ 0 ] ;
s - > last_mv [ 0 ] [ 0 ] [ 1 ] =
s - > last_mv [ 0 ] [ 1 ] [ 1 ] = s - > mv [ 0 ] [ 0 ] [ 1 ] ;
2007-07-05 12:08:28 +03:00
}
2013-09-04 18:17:30 +03:00
if ( s - > mv_dir & MV_DIR_BACKWARD ) {
mpeg1_encode_motion ( s ,
s - > mv [ 1 ] [ 0 ] [ 0 ] - s - > last_mv [ 1 ] [ 0 ] [ 0 ] ,
s - > b_code ) ;
mpeg1_encode_motion ( s ,
s - > mv [ 1 ] [ 0 ] [ 1 ] - s - > last_mv [ 1 ] [ 0 ] [ 1 ] ,
s - > b_code ) ;
s - > last_mv [ 1 ] [ 0 ] [ 0 ] =
s - > last_mv [ 1 ] [ 1 ] [ 0 ] = s - > mv [ 1 ] [ 0 ] [ 0 ] ;
s - > last_mv [ 1 ] [ 0 ] [ 1 ] =
s - > last_mv [ 1 ] [ 1 ] [ 1 ] = s - > mv [ 1 ] [ 0 ] [ 1 ] ;
2007-07-05 12:08:28 +03:00
}
2013-09-04 18:17:30 +03:00
} else {
2012-07-09 16:38:16 +03:00
av_assert2 ( s - > mv_type = = MV_TYPE_FIELD ) ;
av_assert2 ( ! s - > frame_pred_frame_dct ) ;
2013-09-04 18:17:30 +03:00
if ( cbp ) { // With coded bloc pattern
2007-07-05 12:08:28 +03:00
if ( s - > dquant ) {
2013-09-04 18:17:30 +03:00
if ( s - > mv_dir = = MV_DIR_FORWARD )
2007-07-05 12:08:28 +03:00
put_mb_modes ( s , 6 , 3 , 1 , 1 ) ;
else
2013-09-04 18:17:30 +03:00
put_mb_modes ( s , 8 - s - > mv_dir , 2 , 1 , 1 ) ;
2007-07-05 12:08:28 +03:00
put_qscale ( s ) ;
} else {
2013-09-04 18:17:30 +03:00
put_mb_modes ( s , 5 - s - > mv_dir , 3 , 1 , 1 ) ;
2007-07-05 12:08:28 +03:00
}
2013-09-04 18:17:30 +03:00
} else { // No coded bloc pattern
put_bits ( & s - > pb , 5 - s - > mv_dir , 2 ) ;
put_bits ( & s - > pb , 2 , 1 ) ; /* motion_type: field */
2007-07-05 12:08:28 +03:00
s - > qscale - = s - > dquant ;
}
s - > misc_bits + = get_bits_diff ( s ) ;
2013-09-04 18:17:30 +03:00
if ( s - > mv_dir & MV_DIR_FORWARD ) {
for ( i = 0 ; i < 2 ; i + + ) {
2007-07-05 12:08:28 +03:00
put_bits ( & s - > pb , 1 , s - > field_select [ 0 ] [ i ] ) ;
2013-09-04 18:17:30 +03:00
mpeg1_encode_motion ( s ,
s - > mv [ 0 ] [ i ] [ 0 ] - s - > last_mv [ 0 ] [ i ] [ 0 ] ,
s - > f_code ) ;
mpeg1_encode_motion ( s ,
s - > mv [ 0 ] [ i ] [ 1 ] - ( s - > last_mv [ 0 ] [ i ] [ 1 ] > > 1 ) ,
s - > f_code ) ;
s - > last_mv [ 0 ] [ i ] [ 0 ] = s - > mv [ 0 ] [ i ] [ 0 ] ;
s - > last_mv [ 0 ] [ i ] [ 1 ] = s - > mv [ 0 ] [ i ] [ 1 ] * 2 ;
2007-07-05 12:08:28 +03:00
}
}
2013-09-04 18:17:30 +03:00
if ( s - > mv_dir & MV_DIR_BACKWARD ) {
for ( i = 0 ; i < 2 ; i + + ) {
2007-07-05 12:08:28 +03:00
put_bits ( & s - > pb , 1 , s - > field_select [ 1 ] [ i ] ) ;
2013-09-04 18:17:30 +03:00
mpeg1_encode_motion ( s ,
s - > mv [ 1 ] [ i ] [ 0 ] - s - > last_mv [ 1 ] [ i ] [ 0 ] ,
s - > b_code ) ;
mpeg1_encode_motion ( s ,
s - > mv [ 1 ] [ i ] [ 1 ] - ( s - > last_mv [ 1 ] [ i ] [ 1 ] > > 1 ) ,
s - > b_code ) ;
s - > last_mv [ 1 ] [ i ] [ 0 ] = s - > mv [ 1 ] [ i ] [ 0 ] ;
s - > last_mv [ 1 ] [ i ] [ 1 ] = s - > mv [ 1 ] [ i ] [ 1 ] * 2 ;
2007-07-05 12:08:28 +03:00
}
}
}
s - > mv_bits + = get_bits_diff ( s ) ;
2013-09-04 18:17:30 +03:00
if ( cbp ) {
2021-12-25 03:07:53 +02:00
if ( chroma_y_shift ) {
2013-09-04 18:17:30 +03:00
put_bits ( & s - > pb ,
ff_mpeg12_mbPatTable [ cbp ] [ 1 ] ,
ff_mpeg12_mbPatTable [ cbp ] [ 0 ] ) ;
2007-07-05 12:08:28 +03:00
} else {
2013-09-04 18:17:30 +03:00
put_bits ( & s - > pb ,
ff_mpeg12_mbPatTable [ cbp > > 2 ] [ 1 ] ,
ff_mpeg12_mbPatTable [ cbp > > 2 ] [ 0 ] ) ;
2008-07-13 17:27:48 +03:00
put_sbits ( & s - > pb , 2 , cbp ) ;
2007-07-05 12:08:28 +03:00
}
}
}
2013-09-04 18:17:30 +03:00
for ( i = 0 ; i < mb_block_count ; i + + )
if ( cbp & ( 1 < < ( mb_block_count - 1 - i ) ) )
2007-07-05 12:08:28 +03:00
mpeg1_encode_block ( s , block [ i ] , i ) ;
s - > mb_skip_run = 0 ;
2013-09-04 18:17:30 +03:00
if ( s - > mb_intra )
s - > i_tex_bits + = get_bits_diff ( s ) ;
2007-07-05 12:08:28 +03:00
else
2013-09-04 18:17:30 +03:00
s - > p_tex_bits + = get_bits_diff ( s ) ;
2007-07-05 12:08:28 +03:00
}
}
2014-04-28 02:11:04 +03:00
void ff_mpeg1_encode_mb ( MpegEncContext * s , int16_t block [ 8 ] [ 64 ] ,
2013-09-04 18:17:30 +03:00
int motion_x , int motion_y )
2007-07-05 12:08:28 +03:00
{
2013-09-04 18:17:30 +03:00
if ( s - > chroma_format = = CHROMA_420 )
2021-12-25 03:07:53 +02:00
mpeg1_encode_mb_internal ( s , block , motion_x , motion_y , 6 , 1 ) ;
2013-09-04 18:17:30 +03:00
else
2021-12-25 03:07:53 +02:00
mpeg1_encode_mb_internal ( s , block , motion_x , motion_y , 8 , 0 ) ;
2007-07-05 12:08:28 +03:00
}
2020-11-24 15:39:47 +02:00
static av_cold void mpeg12_encode_init_static ( void )
2007-07-05 12:08:28 +03:00
{
2022-10-22 02:54:15 +02:00
ff_rl_init_level_run ( mpeg12_max_level , mpeg12_index_run ,
ff_mpeg12_run , ff_mpeg12_level , MPEG12_RL_NB_ELEMS ) ;
2007-07-05 12:08:28 +03:00
2022-10-22 02:54:15 +02:00
ff_mpeg1_init_uni_ac_vlc ( mpeg12_max_level , mpeg12_index_run ,
2022-10-22 01:33:11 +02:00
ff_mpeg1_vlc_table , uni_mpeg1_ac_vlc_len ) ;
2022-10-22 02:54:15 +02:00
ff_mpeg1_init_uni_ac_vlc ( mpeg12_max_level , mpeg12_index_run ,
2022-10-22 01:33:11 +02:00
ff_mpeg2_vlc_table , uni_mpeg2_ac_vlc_len ) ;
2007-07-05 12:08:28 +03:00
2020-11-24 15:43:46 +02:00
/* build unified dc encoding tables */
for ( int i = - 255 ; i < 256 ; i + + ) {
int adiff , index ;
int bits , code ;
int diff = i ;
2007-07-05 12:08:28 +03:00
2020-11-24 15:43:46 +02:00
adiff = FFABS ( diff ) ;
if ( diff < 0 )
diff - - ;
index = av_log2 ( 2 * adiff ) ;
2007-07-05 12:08:28 +03:00
2020-11-24 15:43:46 +02:00
bits = ff_mpeg12_vlc_dc_lum_bits [ index ] + index ;
code = ( ff_mpeg12_vlc_dc_lum_code [ index ] < < index ) +
av_mod_uintp2 ( diff , index ) ;
mpeg1_lum_dc_uni [ i + 255 ] = bits + ( code < < 8 ) ;
2007-07-05 12:08:28 +03:00
2020-11-24 15:43:46 +02:00
bits = ff_mpeg12_vlc_dc_chroma_bits [ index ] + index ;
code = ( ff_mpeg12_vlc_dc_chroma_code [ index ] < < index ) +
av_mod_uintp2 ( diff , index ) ;
mpeg1_chr_dc_uni [ i + 255 ] = bits + ( code < < 8 ) ;
}
for ( int f_code = 1 ; f_code < = MAX_FCODE ; f_code + + )
for ( int mv = - MAX_DMV ; mv < = MAX_DMV ; mv + + ) {
int len ;
if ( mv = = 0 ) {
2021-12-25 04:25:02 +02:00
len = 1 ; /* ff_mpeg12_mbMotionVectorTable[0][1] */
2020-11-24 15:43:46 +02:00
} else {
int val , bit_size , code ;
bit_size = f_code - 1 ;
val = mv ;
if ( val < 0 )
val = - val ;
val - - ;
code = ( val > > bit_size ) + 1 ;
if ( code < 17 )
len = ff_mpeg12_mbMotionVectorTable [ code ] [ 1 ] +
1 + bit_size ;
else
2021-12-25 04:25:02 +02:00
len = 10 /* ff_mpeg12_mbMotionVectorTable[16][1] */ +
2020-11-24 15:43:46 +02:00
2 + bit_size ;
2007-07-05 12:08:28 +03:00
}
2020-11-24 15:43:46 +02:00
mv_penalty [ f_code ] [ mv + MAX_DMV ] = len ;
}
2007-07-05 12:08:28 +03:00
2020-11-24 15:43:46 +02:00
for ( int f_code = MAX_FCODE ; f_code > 0 ; f_code - - )
for ( int mv = - ( 8 < < f_code ) ; mv < ( 8 < < f_code ) ; mv + + )
fcode_tab [ mv + MAX_MV ] = f_code ;
2020-11-24 15:39:47 +02:00
}
av_cold void ff_mpeg1_encode_init ( MpegEncContext * s )
{
static AVOnce init_static_once = AV_ONCE_INIT ;
2022-09-30 03:06:25 +02:00
s - > y_dc_scale_table =
2022-10-28 12:06:38 +02:00
s - > c_dc_scale_table = ff_mpeg12_dc_scale_table [ s - > intra_dc_precision ] ;
2020-11-24 15:39:47 +02:00
2013-09-04 18:17:30 +03:00
s - > me . mv_penalty = mv_penalty ;
s - > fcode_tab = fcode_tab ;
if ( s - > codec_id = = AV_CODEC_ID_MPEG1VIDEO ) {
s - > min_qcoeff = - 255 ;
s - > max_qcoeff = 255 ;
} else {
s - > min_qcoeff = - 2047 ;
s - > max_qcoeff = 2047 ;
2022-02-17 10:02:27 +02:00
s - > mpeg_quant = 1 ;
2007-07-05 12:08:28 +03:00
}
if ( s - > intra_vlc_format ) {
2013-09-04 18:17:30 +03:00
s - > intra_ac_vlc_length =
s - > intra_ac_vlc_last_length = uni_mpeg2_ac_vlc_len ;
2007-07-05 12:08:28 +03:00
} else {
2013-09-04 18:17:30 +03:00
s - > intra_ac_vlc_length =
s - > intra_ac_vlc_last_length = uni_mpeg1_ac_vlc_len ;
2007-07-05 12:08:28 +03:00
}
2013-09-04 18:17:30 +03:00
s - > inter_ac_vlc_length =
s - > inter_ac_vlc_last_length = uni_mpeg1_ac_vlc_len ;
2020-11-24 15:39:47 +02:00
ff_thread_once ( & init_static_once , mpeg12_encode_init_static ) ;
2007-07-05 12:08:28 +03:00
}
2021-12-22 21:25:13 +02:00
# define OFFSET(x) offsetof(MPEG12EncContext, x)
2011-08-27 09:26:14 +03:00
# define VE AV_OPT_FLAG_ENCODING_PARAM | AV_OPT_FLAG_VIDEO_PARAM
2013-09-04 18:17:30 +03:00
# define COMMON_OPTS \
2016-01-31 14:28:04 +02:00
{ " gop_timecode " , " MPEG GOP Timecode in hh:mm:ss[:;.]ff format. Overrides timecode_frame_start. " , \
2020-03-17 23:46:36 +02:00
OFFSET ( tc_opt_str ) , AV_OPT_TYPE_STRING , { . str = NULL } , 0 , 0 , VE } , \
2013-09-04 18:17:30 +03:00
{ " drop_frame_timecode " , " Timecode is in drop frame format. " , \
2015-11-21 23:04:39 +02:00
OFFSET ( drop_frame_timecode ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , VE } , \
2013-09-04 18:17:30 +03:00
{ " scan_offset " , " Reserve space for SVCD scan offset user data. " , \
2016-01-31 14:28:04 +02:00
OFFSET ( scan_offset ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , VE } , \
2015-09-13 23:45:24 +02:00
{ " timecode_frame_start " , " GOP timecode frame start number, in non-drop-frame format " , \
2016-01-31 14:28:04 +02:00
OFFSET ( timecode_frame_start ) , AV_OPT_TYPE_INT64 , { . i64 = - 1 } , - 1 , INT64_MAX , VE } , \
2021-04-06 19:32:07 +02:00
FF_MPV_COMMON_BFRAME_OPTS
2011-08-27 10:57:36 +03:00
static const AVOption mpeg1_options [ ] = {
COMMON_OPTS
2012-02-27 14:12:19 +03:00
FF_MPV_COMMON_OPTS
2022-02-07 15:50:34 +02:00
FF_MPV_COMMON_MOTION_EST_OPTS
2011-08-27 10:57:36 +03:00
{ NULL } ,
} ;
static const AVOption mpeg2_options [ ] = {
COMMON_OPTS
2020-11-24 16:32:04 +02:00
{ " intra_vlc " , " Use MPEG-2 intra VLC table. " ,
2021-12-22 21:25:13 +02:00
FF_MPV_OFFSET ( intra_vlc_format ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , VE } ,
{ " non_linear_quant " , " Use nonlinear quantizer. " , FF_MPV_OFFSET ( q_scale_type ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , VE } ,
{ " alternate_scan " , " Enable alternate scantable. " , FF_MPV_OFFSET ( alternate_scan ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , VE } ,
2021-04-06 17:53:20 +02:00
{ " a53cc " , " Use A53 Closed Captions (if available) " , OFFSET ( a53_cc ) , AV_OPT_TYPE_BOOL , { . i64 = 1 } , 0 , 1 , VE } ,
2024-02-11 16:41:05 +02:00
{ " seq_disp_ext " , " Write sequence_display_extension blocks. " , OFFSET ( seq_disp_ext ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , 1 , VE , . unit = " seq_disp_ext " } ,
{ " auto " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = - 1 } , 0 , 0 , VE , . unit = " seq_disp_ext " } ,
{ " never " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = 0 } , 0 , 0 , VE , . unit = " seq_disp_ext " } ,
{ " always " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = 1 } , 0 , 0 , VE , . unit = " seq_disp_ext " } ,
{ " video_format " , " Video_format in the sequence_display_extension indicating the source of the video. " , OFFSET ( video_format ) , AV_OPT_TYPE_INT , { . i64 = VIDEO_FORMAT_UNSPECIFIED } , 0 , 7 , VE , . unit = " video_format " } ,
{ " component " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = VIDEO_FORMAT_COMPONENT } , 0 , 0 , VE , . unit = " video_format " } ,
{ " pal " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = VIDEO_FORMAT_PAL } , 0 , 0 , VE , . unit = " video_format " } ,
{ " ntsc " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = VIDEO_FORMAT_NTSC } , 0 , 0 , VE , . unit = " video_format " } ,
{ " secam " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = VIDEO_FORMAT_SECAM } , 0 , 0 , VE , . unit = " video_format " } ,
{ " mac " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = VIDEO_FORMAT_MAC } , 0 , 0 , VE , . unit = " video_format " } ,
{ " unspecified " , NULL , 0 , AV_OPT_TYPE_CONST , { . i64 = VIDEO_FORMAT_UNSPECIFIED } , 0 , 0 , VE , . unit = " video_format " } ,
# define LEVEL(name, value) name, NULL, 0, AV_OPT_TYPE_CONST, { .i64 = value }, 0, 0, VE, .unit = "avctx.level"
2020-08-02 15:07:04 +02:00
{ LEVEL ( " high " , 4 ) } ,
{ LEVEL ( " high1440 " , 6 ) } ,
{ LEVEL ( " main " , 8 ) } ,
{ LEVEL ( " low " , 10 ) } ,
# undef LEVEL
2012-02-27 14:12:19 +03:00
FF_MPV_COMMON_OPTS
2022-02-07 15:50:34 +02:00
FF_MPV_COMMON_MOTION_EST_OPTS
2020-05-30 15:25:32 +02:00
FF_MPEG2_PROFILE_OPTS
2011-08-27 09:26:14 +03:00
{ NULL } ,
2011-07-06 15:44:59 +03:00
} ;
2013-09-04 18:17:30 +03:00
# define mpeg12_class(x) \
static const AVClass mpeg # # x # # _class = { \
. class_name = " mpeg " # x " video encoder " , \
2024-01-19 14:33:28 +02:00
. item_name = av_default_item_name , \
2013-09-04 18:17:30 +03:00
. option = mpeg # # x # # _options , \
. version = LIBAVUTIL_VERSION_INT , \
2011-08-27 09:26:14 +03:00
} ;
mpeg12_class ( 1 )
mpeg12_class ( 2 )
2022-03-16 22:09:54 +02:00
const FFCodec ff_mpeg1video_encoder = {
. p . name = " mpeg1video " ,
2022-08-29 13:38:02 +02:00
CODEC_LONG_NAME ( " MPEG-1 video " ) ,
2022-03-16 22:09:54 +02:00
. p . type = AVMEDIA_TYPE_VIDEO ,
. p . id = AV_CODEC_ID_MPEG1VIDEO ,
2021-12-22 18:28:20 +02:00
. priv_data_size = sizeof ( MPEG12EncContext ) ,
2012-04-06 19:19:39 +03:00
. init = encode_init ,
2022-03-30 23:28:24 +02:00
FF_CODEC_ENCODE_CB ( ff_mpv_encode_picture ) ,
2014-08-10 18:25:12 +03:00
. close = ff_mpv_encode_end ,
2022-03-16 22:09:54 +02:00
. p . supported_framerates = ff_mpeg12_frame_rate_tab + 1 ,
. p . pix_fmts = ( const enum AVPixelFormat [ ] ) { AV_PIX_FMT_YUV420P ,
2013-09-04 18:17:30 +03:00
AV_PIX_FMT_NONE } ,
2023-01-04 18:27:07 +02:00
. p . capabilities = AV_CODEC_CAP_DELAY | AV_CODEC_CAP_SLICE_THREADS |
AV_CODEC_CAP_ENCODER_REORDERED_OPAQUE ,
2022-07-10 00:05:45 +02:00
. caps_internal = FF_CODEC_CAP_INIT_CLEANUP ,
2022-03-16 22:09:54 +02:00
. p . priv_class = & mpeg1_class ,
2007-07-05 12:08:28 +03:00
} ;
2022-03-16 22:09:54 +02:00
const FFCodec ff_mpeg2video_encoder = {
. p . name = " mpeg2video " ,
2022-08-29 13:38:02 +02:00
CODEC_LONG_NAME ( " MPEG-2 video " ) ,
2022-03-16 22:09:54 +02:00
. p . type = AVMEDIA_TYPE_VIDEO ,
. p . id = AV_CODEC_ID_MPEG2VIDEO ,
2021-12-22 18:28:20 +02:00
. priv_data_size = sizeof ( MPEG12EncContext ) ,
2012-04-06 19:19:39 +03:00
. init = encode_init ,
2022-03-30 23:28:24 +02:00
FF_CODEC_ENCODE_CB ( ff_mpv_encode_picture ) ,
2014-08-10 18:25:12 +03:00
. close = ff_mpv_encode_end ,
2022-03-16 22:09:54 +02:00
. p . supported_framerates = ff_mpeg2_frame_rate_tab ,
. p . pix_fmts = ( const enum AVPixelFormat [ ] ) { AV_PIX_FMT_YUV420P ,
2013-09-04 18:17:30 +03:00
AV_PIX_FMT_YUV422P ,
AV_PIX_FMT_NONE } ,
2023-01-04 18:27:07 +02:00
. p . capabilities = AV_CODEC_CAP_DELAY | AV_CODEC_CAP_SLICE_THREADS |
AV_CODEC_CAP_ENCODER_REORDERED_OPAQUE ,
2022-07-10 00:05:45 +02:00
. caps_internal = FF_CODEC_CAP_INIT_CLEANUP ,
2022-03-16 22:09:54 +02:00
. p . priv_class = & mpeg2_class ,
2007-07-05 12:08:28 +03:00
} ;
2020-12-10 09:50:37 +02:00
# endif /* CONFIG_MPEG1VIDEO_ENCODER || CONFIG_MPEG2VIDEO_ENCODER */