h264.c
资源名称:chapter15.rar [点击查看]
上传用户:hjq518
上传日期:2021-12-09
资源大小:5084k
文件大小:214k
源码类别:
Audio
开发平台:
Visual C++
- if(h->nal_ref_idc==0 && abs_frame_num > 0)
- abs_frame_num--;
- expected_delta_per_poc_cycle = 0;
- for(i=0; i < h->sps.poc_cycle_length; i++)
- expected_delta_per_poc_cycle += h->sps.offset_for_ref_frame[ i ]; //FIXME integrate during sps parse
- if(abs_frame_num > 0){
- int poc_cycle_cnt = (abs_frame_num - 1) / h->sps.poc_cycle_length;
- int frame_num_in_poc_cycle = (abs_frame_num - 1) % h->sps.poc_cycle_length;
- expectedpoc = poc_cycle_cnt * expected_delta_per_poc_cycle;
- for(i = 0; i <= frame_num_in_poc_cycle; i++)
- expectedpoc = expectedpoc + h->sps.offset_for_ref_frame[ i ];
- } else
- expectedpoc = 0;
- if(h->nal_ref_idc == 0)
- expectedpoc = expectedpoc + h->sps.offset_for_non_ref_pic;
- field_poc[0] = expectedpoc + h->delta_poc[0];
- field_poc[1] = field_poc[0] + h->sps.offset_for_top_to_bottom_field;
- if(s->picture_structure == PICT_FRAME)
- field_poc[1] += h->delta_poc[1];
- }else{
- int poc;
- if(h->nal_unit_type == NAL_IDR_SLICE){
- poc= 0;
- }else{
- if(h->nal_ref_idc) poc= 2*(h->frame_num_offset + h->frame_num);
- else poc= 2*(h->frame_num_offset + h->frame_num) - 1;
- }
- field_poc[0]= poc;
- field_poc[1]= poc;
- }
- if(s->picture_structure != PICT_BOTTOM_FIELD)
- s->current_picture_ptr->field_poc[0]= field_poc[0];
- if(s->picture_structure != PICT_TOP_FIELD)
- s->current_picture_ptr->field_poc[1]= field_poc[1];
- if(s->picture_structure == PICT_FRAME) // FIXME field pix?
- s->current_picture_ptr->poc= FFMIN(field_poc[0], field_poc[1]);
- return 0;
- }
- /**
- * decodes a slice header.
- * this will allso call MPV_common_init() and frame_start() as needed
- */
- static int decode_slice_header(H264Context *h){
- MpegEncContext * const s = &h->s;
- int first_mb_in_slice, pps_id;
- int num_ref_idx_active_override_flag;
- static const uint8_t slice_type_map[5]= {P_TYPE, B_TYPE, I_TYPE, SP_TYPE, SI_TYPE};
- s->current_picture.reference= h->nal_ref_idc != 0;
- first_mb_in_slice= get_ue_golomb(&s->gb);
- h->slice_type= get_ue_golomb(&s->gb);
- if(h->slice_type > 9){
- av_log(h->s.avctx, AV_LOG_ERROR, "slice type too large (%d) at %d %dn", h->slice_type, s->mb_x, s->mb_y);
- }
- if(h->slice_type > 4){
- h->slice_type -= 5;
- h->slice_type_fixed=1;
- }else
- h->slice_type_fixed=0;
- h->slice_type= slice_type_map[ h->slice_type ];
- s->pict_type= h->slice_type; // to make a few old func happy, its wrong though
- pps_id= get_ue_golomb(&s->gb);
- if(pps_id>255){
- av_log(h->s.avctx, AV_LOG_ERROR, "pps_id out of rangen");
- return -1;
- }
- h->pps= h->pps_buffer[pps_id];
- if(h->pps.slice_group_count == 0){
- av_log(h->s.avctx, AV_LOG_ERROR, "non existing PPS referencedn");
- return -1;
- }
- h->sps= h->sps_buffer[ h->pps.sps_id ];
- if(h->sps.log2_max_frame_num == 0){
- av_log(h->s.avctx, AV_LOG_ERROR, "non existing SPS referencedn");
- return -1;
- }
- s->mb_width= h->sps.mb_width;
- s->mb_height= h->sps.mb_height;
- h->b_stride= s->mb_width*4;
- h->b8_stride= s->mb_width*2;
- s->mb_x = first_mb_in_slice % s->mb_width;
- s->mb_y = first_mb_in_slice / s->mb_width; //FIXME AFFW
- s->width = 16*s->mb_width - 2*(h->sps.crop_left + h->sps.crop_right );
- if(h->sps.frame_mbs_only_flag)
- s->height= 16*s->mb_height - 2*(h->sps.crop_top + h->sps.crop_bottom);
- else
- s->height= 16*s->mb_height - 4*(h->sps.crop_top + h->sps.crop_bottom); //FIXME recheck
- if (s->context_initialized
- && ( s->width != s->avctx->width || s->height != s->avctx->height)) {
- free_tables(h);
- MPV_common_end(s);
- }
- if (!s->context_initialized) {
- if (MPV_common_init(s) < 0)
- return -1;
- alloc_tables(h);
- s->avctx->width = s->width;
- s->avctx->height = s->height;
- s->avctx->sample_aspect_ratio= h->sps.sar;
- }
- if(first_mb_in_slice == 0){
- frame_start(h);
- }
- s->current_picture_ptr->frame_num= //FIXME frame_num cleanup
- h->frame_num= get_bits(&s->gb, h->sps.log2_max_frame_num);
- if(h->sps.frame_mbs_only_flag){
- s->picture_structure= PICT_FRAME;
- }else{
- if(get_bits1(&s->gb)) //field_pic_flag
- s->picture_structure= PICT_TOP_FIELD + get_bits1(&s->gb); //bottom_field_flag
- else
- s->picture_structure= PICT_FRAME;
- }
- if(s->picture_structure==PICT_FRAME){
- h->curr_pic_num= h->frame_num;
- h->max_pic_num= 1<< h->sps.log2_max_frame_num;
- }else{
- h->curr_pic_num= 2*h->frame_num;
- h->max_pic_num= 1<<(h->sps.log2_max_frame_num + 1);
- }
- if(h->nal_unit_type == NAL_IDR_SLICE){
- get_ue_golomb(&s->gb); /* idr_pic_id */
- }
- if(h->sps.poc_type==0){
- h->poc_lsb= get_bits(&s->gb, h->sps.log2_max_poc_lsb);
- if(h->pps.pic_order_present==1 && s->picture_structure==PICT_FRAME){
- h->delta_poc_bottom= get_se_golomb(&s->gb);
- }
- }
- if(h->sps.poc_type==1 && !h->sps.delta_pic_order_always_zero_flag){
- h->delta_poc[0]= get_se_golomb(&s->gb);
- if(h->pps.pic_order_present==1 && s->picture_structure==PICT_FRAME)
- h->delta_poc[1]= get_se_golomb(&s->gb);
- }
- init_poc(h);
- if(h->pps.redundant_pic_cnt_present){
- h->redundant_pic_count= get_ue_golomb(&s->gb);
- }
- //set defaults, might be overriden a few line later
- h->ref_count[0]= h->pps.ref_count[0];
- h->ref_count[1]= h->pps.ref_count[1];
- if(h->slice_type == P_TYPE || h->slice_type == SP_TYPE || h->slice_type == B_TYPE){
- if(h->slice_type == B_TYPE){
- h->direct_spatial_mv_pred= get_bits1(&s->gb);
- }
- num_ref_idx_active_override_flag= get_bits1(&s->gb);
- if(num_ref_idx_active_override_flag){
- h->ref_count[0]= get_ue_golomb(&s->gb) + 1;
- if(h->slice_type==B_TYPE)
- h->ref_count[1]= get_ue_golomb(&s->gb) + 1;
- if(h->ref_count[0] > 32 || h->ref_count[1] > 32){
- av_log(h->s.avctx, AV_LOG_ERROR, "reference overflown");
- return -1;
- }
- }
- }
- if(first_mb_in_slice == 0){
- fill_default_ref_list(h);
- }
- decode_ref_pic_list_reordering(h);
- if( (h->pps.weighted_pred && (h->slice_type == P_TYPE || h->slice_type == SP_TYPE ))
- || (h->pps.weighted_bipred_idc==1 && h->slice_type==B_TYPE ) )
- pred_weight_table(h);
- if(s->current_picture.reference)
- decode_ref_pic_marking(h);
- if( h->slice_type != I_TYPE && h->slice_type != SI_TYPE && h->pps.cabac )
- h->cabac_init_idc = get_ue_golomb(&s->gb);
- h->last_qscale_diff = 0;
- s->qscale = h->pps.init_qp + get_se_golomb(&s->gb);
- if(s->qscale<0 || s->qscale>51){
- av_log(s->avctx, AV_LOG_ERROR, "QP %d out of rangen", s->qscale);
- return -1;
- }
- //FIXME qscale / qp ... stuff
- if(h->slice_type == SP_TYPE){
- get_bits1(&s->gb); /* sp_for_switch_flag */
- }
- if(h->slice_type==SP_TYPE || h->slice_type == SI_TYPE){
- get_se_golomb(&s->gb); /* slice_qs_delta */
- }
- h->deblocking_filter = 1;
- h->slice_alpha_c0_offset = 0;
- h->slice_beta_offset = 0;
- if( h->pps.deblocking_filter_parameters_present ) {
- h->deblocking_filter= get_ue_golomb(&s->gb);
- if(h->deblocking_filter < 2)
- h->deblocking_filter^= 1; // 1<->0
- if( h->deblocking_filter ) {
- h->slice_alpha_c0_offset = get_se_golomb(&s->gb) << 1;
- h->slice_beta_offset = get_se_golomb(&s->gb) << 1;
- }
- }
- #if 0 //FMO
- if( h->pps.num_slice_groups > 1 && h->pps.mb_slice_group_map_type >= 3 && h->pps.mb_slice_group_map_type <= 5)
- slice_group_change_cycle= get_bits(&s->gb, ?);
- #endif
- if(s->avctx->debug&FF_DEBUG_PICT_INFO){
- av_log(h->s.avctx, AV_LOG_DEBUG, "mb:%d %c pps:%d frame:%d poc:%d/%d ref:%d/%d qp:%d loop:%dn",
- first_mb_in_slice,
- av_get_pict_type_char(h->slice_type),
- pps_id, h->frame_num,
- s->current_picture_ptr->field_poc[0], s->current_picture_ptr->field_poc[1],
- h->ref_count[0], h->ref_count[1],
- s->qscale,
- h->deblocking_filter
- );
- }
- return 0;
- }
- /**
- *
- */
- static inline int get_level_prefix(GetBitContext *gb){
- unsigned int buf;
- int log;
- OPEN_READER(re, gb);
- UPDATE_CACHE(re, gb);
- buf=GET_CACHE(re, gb);
- log= 32 - av_log2(buf);
- #ifdef TRACE
- print_bin(buf>>(32-log), log);
- printf("%5d %2d %3d lpr @%5d in %s get_level_prefixn", buf>>(32-log), log, log-1, get_bits_count(gb), __FILE__);
- #endif
- LAST_SKIP_BITS(re, gb, log);
- CLOSE_READER(re, gb);
- return log-1;
- }
- /**
- * decodes a residual block.
- * @param n block index
- * @param scantable scantable
- * @param max_coeff number of coefficients in the block
- * @return <0 if an error occured
- */
- static int decode_residual(H264Context *h, GetBitContext *gb, DCTELEM *block, int n, const uint8_t *scantable, int qp, int max_coeff){
- MpegEncContext * const s = &h->s;
- const uint16_t *qmul= dequant_coeff[qp];
- static const int coeff_token_table_index[17]= {0, 0, 1, 1, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, 3};
- int level[16], run[16];
- int suffix_length, zeros_left, coeff_num, coeff_token, total_coeff, i, trailing_ones;
- //FIXME put trailing_onex into the context
- if(n == CHROMA_DC_BLOCK_INDEX){
- coeff_token= get_vlc2(gb, chroma_dc_coeff_token_vlc.table, CHROMA_DC_COEFF_TOKEN_VLC_BITS, 1);
- total_coeff= coeff_token>>2;
- }else{
- if(n == LUMA_DC_BLOCK_INDEX){
- total_coeff= pred_non_zero_count(h, 0);
- coeff_token= get_vlc2(gb, coeff_token_vlc[ coeff_token_table_index[total_coeff] ].table, COEFF_TOKEN_VLC_BITS, 2);
- total_coeff= coeff_token>>2;
- }else{
- total_coeff= pred_non_zero_count(h, n);
- coeff_token= get_vlc2(gb, coeff_token_vlc[ coeff_token_table_index[total_coeff] ].table, COEFF_TOKEN_VLC_BITS, 2);
- total_coeff= coeff_token>>2;
- h->non_zero_count_cache[ scan8[n] ]= total_coeff;
- }
- }
- //FIXME set last_non_zero?
- if(total_coeff==0)
- return 0;
- trailing_ones= coeff_token&3;
- tprintf("trailing:%d, total:%dn", trailing_ones, total_coeff);
- assert(total_coeff<=16);
- for(i=0; i<trailing_ones; i++){
- level[i]= 1 - 2*get_bits1(gb);
- }
- suffix_length= total_coeff > 10 && trailing_ones < 3;
- for(; i<total_coeff; i++){
- const int prefix= get_level_prefix(gb);
- int level_code, mask;
- if(prefix<14){ //FIXME try to build a large unified VLC table for all this
- if(suffix_length)
- level_code= (prefix<<suffix_length) + get_bits(gb, suffix_length); //part
- else
- level_code= (prefix<<suffix_length); //part
- }else if(prefix==14){
- if(suffix_length)
- level_code= (prefix<<suffix_length) + get_bits(gb, suffix_length); //part
- else
- level_code= prefix + get_bits(gb, 4); //part
- }else if(prefix==15){
- level_code= (prefix<<suffix_length) + get_bits(gb, 12); //part
- if(suffix_length==0) level_code+=15; //FIXME doesnt make (much)sense
- }else{
- av_log(h->s.avctx, AV_LOG_ERROR, "prefix too large at %d %dn", s->mb_x, s->mb_y);
- return -1;
- }
- if(i==trailing_ones && i<3) level_code+= 2; //FIXME split first iteration
- mask= -(level_code&1);
- level[i]= (((2+level_code)>>1) ^ mask) - mask;
- if(suffix_length==0) suffix_length=1; //FIXME split first iteration
- #if 1
- if(ABS(level[i]) > (3<<(suffix_length-1)) && suffix_length<6) suffix_length++;
- #else
- if((2+level_code)>>1) > (3<<(suffix_length-1)) && suffix_length<6) suffix_length++;
- /* ? == prefix > 2 or sth */
- #endif
- tprintf("level: %d suffix_length:%dn", level[i], suffix_length);
- }
- if(total_coeff == max_coeff)
- zeros_left=0;
- else{
- if(n == CHROMA_DC_BLOCK_INDEX)
- zeros_left= get_vlc2(gb, chroma_dc_total_zeros_vlc[ total_coeff-1 ].table, CHROMA_DC_TOTAL_ZEROS_VLC_BITS, 1);
- else
- zeros_left= get_vlc2(gb, total_zeros_vlc[ total_coeff-1 ].table, TOTAL_ZEROS_VLC_BITS, 1);
- }
- for(i=0; i<total_coeff-1; i++){
- if(zeros_left <=0)
- break;
- else if(zeros_left < 7){
- run[i]= get_vlc2(gb, run_vlc[zeros_left-1].table, RUN_VLC_BITS, 1);
- }else{
- run[i]= get_vlc2(gb, run7_vlc.table, RUN7_VLC_BITS, 2);
- }
- zeros_left -= run[i];
- }
- if(zeros_left<0){
- av_log(h->s.avctx, AV_LOG_ERROR, "negative number of zero coeffs at %d %dn", s->mb_x, s->mb_y);
- return -1;
- }
- for(; i<total_coeff-1; i++){
- run[i]= 0;
- }
- run[i]= zeros_left;
- coeff_num=-1;
- if(n > 24){
- for(i=total_coeff-1; i>=0; i--){ //FIXME merge into rundecode?
- int j;
- coeff_num += run[i] + 1; //FIXME add 1 earlier ?
- j= scantable[ coeff_num ];
- block[j]= level[i];
- }
- }else{
- for(i=total_coeff-1; i>=0; i--){ //FIXME merge into rundecode?
- int j;
- coeff_num += run[i] + 1; //FIXME add 1 earlier ?
- j= scantable[ coeff_num ];
- block[j]= level[i] * qmul[j];
- // printf("%d %d ", block[j], qmul[j]);
- }
- }
- return 0;
- }
- /**
- * decodes a macroblock
- * @returns 0 if ok, AC_ERROR / DC_ERROR / MV_ERROR if an error is noticed
- */
- static int decode_mb_cavlc(H264Context *h){
- MpegEncContext * const s = &h->s;
- const int mb_xy= s->mb_x + s->mb_y*s->mb_stride;
- int mb_type, partition_count, cbp;
- s->dsp.clear_blocks(h->mb); //FIXME avoid if allready clear (move after skip handlong?
- tprintf("pic:%d mb:%d/%dn", h->frame_num, s->mb_x, s->mb_y);
- cbp = 0; /* avoid warning. FIXME: find a solution without slowing
- down the code */
- if(h->slice_type != I_TYPE && h->slice_type != SI_TYPE){
- if(s->mb_skip_run==-1)
- s->mb_skip_run= get_ue_golomb(&s->gb);
- if (s->mb_skip_run--) {
- int mx, my;
- /* skip mb */
- //FIXME b frame
- mb_type= MB_TYPE_16x16|MB_TYPE_P0L0|MB_TYPE_P1L0;
- memset(h->non_zero_count[mb_xy], 0, 16);
- memset(h->non_zero_count_cache + 8, 0, 8*5); //FIXME ugly, remove pfui
- if(h->sps.mb_aff && s->mb_skip_run==0 && (s->mb_y&1)==0){
- h->mb_field_decoding_flag= get_bits1(&s->gb);
- }
- if(h->mb_field_decoding_flag)
- mb_type|= MB_TYPE_INTERLACED;
- fill_caches(h, mb_type); //FIXME check what is needed and what not ...
- pred_pskip_motion(h, &mx, &my);
- fill_rectangle(&h->ref_cache[0][scan8[0]], 4, 4, 8, 0, 1);
- fill_rectangle( h->mv_cache[0][scan8[0]], 4, 4, 8, pack16to32(mx,my), 4);
- write_back_motion(h, mb_type);
- s->current_picture.mb_type[mb_xy]= mb_type; //FIXME SKIP type
- s->current_picture.qscale_table[mb_xy]= s->qscale;
- h->slice_table[ mb_xy ]= h->slice_num;
- h->prev_mb_skiped= 1;
- return 0;
- }
- }
- if(h->sps.mb_aff /* && !field pic FIXME needed? */){
- if((s->mb_y&1)==0)
- h->mb_field_decoding_flag = get_bits1(&s->gb);
- }else
- h->mb_field_decoding_flag=0; //FIXME som ed note ?!
- h->prev_mb_skiped= 0;
- mb_type= get_ue_golomb(&s->gb);
- if(h->slice_type == B_TYPE){
- if(mb_type < 23){
- partition_count= b_mb_type_info[mb_type].partition_count;
- mb_type= b_mb_type_info[mb_type].type;
- }else{
- mb_type -= 23;
- goto decode_intra_mb;
- }
- }else if(h->slice_type == P_TYPE /*|| h->slice_type == SP_TYPE */){
- if(mb_type < 5){
- partition_count= p_mb_type_info[mb_type].partition_count;
- mb_type= p_mb_type_info[mb_type].type;
- }else{
- mb_type -= 5;
- goto decode_intra_mb;
- }
- }else{
- assert(h->slice_type == I_TYPE);
- decode_intra_mb:
- if(mb_type > 25){
- av_log(h->s.avctx, AV_LOG_ERROR, "mb_type %d in %c slice to large at %d %dn", mb_type, av_get_pict_type_char(h->slice_type), s->mb_x, s->mb_y);
- return -1;
- }
- partition_count=0;
- cbp= i_mb_type_info[mb_type].cbp;
- h->intra16x16_pred_mode= i_mb_type_info[mb_type].pred_mode;
- mb_type= i_mb_type_info[mb_type].type;
- }
- if(h->mb_field_decoding_flag)
- mb_type |= MB_TYPE_INTERLACED;
- s->current_picture.mb_type[mb_xy]= mb_type;
- h->slice_table[ mb_xy ]= h->slice_num;
- if(IS_INTRA_PCM(mb_type)){
- const uint8_t *ptr;
- int x, y;
- // we assume these blocks are very rare so we dont optimize it
- align_get_bits(&s->gb);
- ptr= s->gb.buffer + get_bits_count(&s->gb);
- for(y=0; y<16; y++){
- const int index= 4*(y&3) + 64*(y>>2);
- for(x=0; x<16; x++){
- h->mb[index + (x&3) + 16*(x>>2)]= *(ptr++);
- }
- }
- for(y=0; y<8; y++){
- const int index= 256 + 4*(y&3) + 32*(y>>2);
- for(x=0; x<8; x++){
- h->mb[index + (x&3) + 16*(x>>2)]= *(ptr++);
- }
- }
- for(y=0; y<8; y++){
- const int index= 256 + 64 + 4*(y&3) + 32*(y>>2);
- for(x=0; x<8; x++){
- h->mb[index + (x&3) + 16*(x>>2)]= *(ptr++);
- }
- }
- skip_bits(&s->gb, 384); //FIXME check /fix the bitstream readers
- //FIXME deblock filter, non_zero_count_cache init ...
- memset(h->non_zero_count[mb_xy], 16, 16);
- s->current_picture.qscale_table[mb_xy]= s->qscale;
- return 0;
- }
- fill_caches(h, mb_type);
- //mb_pred
- if(IS_INTRA(mb_type)){
- // init_top_left_availability(h);
- if(IS_INTRA4x4(mb_type)){
- int i;
- // fill_intra4x4_pred_table(h);
- for(i=0; i<16; i++){
- const int mode_coded= !get_bits1(&s->gb);
- const int predicted_mode= pred_intra_mode(h, i);
- int mode;
- if(mode_coded){
- const int rem_mode= get_bits(&s->gb, 3);
- if(rem_mode<predicted_mode)
- mode= rem_mode;
- else
- mode= rem_mode + 1;
- }else{
- mode= predicted_mode;
- }
- h->intra4x4_pred_mode_cache[ scan8[i] ] = mode;
- }
- write_back_intra_pred_mode(h);
- if( check_intra4x4_pred_mode(h) < 0)
- return -1;
- }else{
- h->intra16x16_pred_mode= check_intra_pred_mode(h, h->intra16x16_pred_mode);
- if(h->intra16x16_pred_mode < 0)
- return -1;
- }
- h->chroma_pred_mode= get_ue_golomb(&s->gb);
- h->chroma_pred_mode= check_intra_pred_mode(h, h->chroma_pred_mode);
- if(h->chroma_pred_mode < 0)
- return -1;
- }else if(partition_count==4){
- int i, j, sub_partition_count[4], list, ref[2][4];
- if(h->slice_type == B_TYPE){
- for(i=0; i<4; i++){
- h->sub_mb_type[i]= get_ue_golomb(&s->gb);
- if(h->sub_mb_type[i] >=13){
- av_log(h->s.avctx, AV_LOG_ERROR, "B sub_mb_type %d out of range at %d %dn", h->sub_mb_type[i], s->mb_x, s->mb_y);
- return -1;
- }
- sub_partition_count[i]= b_sub_mb_type_info[ h->sub_mb_type[i] ].partition_count;
- h->sub_mb_type[i]= b_sub_mb_type_info[ h->sub_mb_type[i] ].type;
- }
- }else{
- assert(h->slice_type == P_TYPE || h->slice_type == SP_TYPE); //FIXME SP correct ?
- for(i=0; i<4; i++){
- h->sub_mb_type[i]= get_ue_golomb(&s->gb);
- if(h->sub_mb_type[i] >=4){
- av_log(h->s.avctx, AV_LOG_ERROR, "P sub_mb_type %d out of range at %d %dn", h->sub_mb_type[i], s->mb_x, s->mb_y);
- return -1;
- }
- sub_partition_count[i]= p_sub_mb_type_info[ h->sub_mb_type[i] ].partition_count;
- h->sub_mb_type[i]= p_sub_mb_type_info[ h->sub_mb_type[i] ].type;
- }
- }
- for(list=0; list<2; list++){
- const int ref_count= IS_REF0(mb_type) ? 1 : h->ref_count[list];
- if(ref_count == 0) continue;
- for(i=0; i<4; i++){
- if(IS_DIR(h->sub_mb_type[i], 0, list) && !IS_DIRECT(h->sub_mb_type[i])){
- ref[list][i] = get_te0_golomb(&s->gb, ref_count); //FIXME init to 0 before and skip?
- }else{
- //FIXME
- ref[list][i] = -1;
- }
- }
- }
- for(list=0; list<2; list++){
- const int ref_count= IS_REF0(mb_type) ? 1 : h->ref_count[list];
- if(ref_count == 0) continue;
- for(i=0; i<4; i++){
- h->ref_cache[list][ scan8[4*i] ]=h->ref_cache[list][ scan8[4*i]+1 ]=
- h->ref_cache[list][ scan8[4*i]+8 ]=h->ref_cache[list][ scan8[4*i]+9 ]= ref[list][i];
- if(IS_DIR(h->sub_mb_type[i], 0, list) && !IS_DIRECT(h->sub_mb_type[i])){
- const int sub_mb_type= h->sub_mb_type[i];
- const int block_width= (sub_mb_type & (MB_TYPE_16x16|MB_TYPE_16x8)) ? 2 : 1;
- for(j=0; j<sub_partition_count[i]; j++){
- int mx, my;
- const int index= 4*i + block_width*j;
- int16_t (* mv_cache)[2]= &h->mv_cache[list][ scan8[index] ];
- pred_motion(h, index, block_width, list, h->ref_cache[list][ scan8[index] ], &mx, &my);
- mx += get_se_golomb(&s->gb);
- my += get_se_golomb(&s->gb);
- tprintf("final mv:%d %dn", mx, my);
- if(IS_SUB_8X8(sub_mb_type)){
- mv_cache[ 0 ][0]= mv_cache[ 1 ][0]=
- mv_cache[ 8 ][0]= mv_cache[ 9 ][0]= mx;
- mv_cache[ 0 ][1]= mv_cache[ 1 ][1]=
- mv_cache[ 8 ][1]= mv_cache[ 9 ][1]= my;
- }else if(IS_SUB_8X4(sub_mb_type)){
- mv_cache[ 0 ][0]= mv_cache[ 1 ][0]= mx;
- mv_cache[ 0 ][1]= mv_cache[ 1 ][1]= my;
- }else if(IS_SUB_4X8(sub_mb_type)){
- mv_cache[ 0 ][0]= mv_cache[ 8 ][0]= mx;
- mv_cache[ 0 ][1]= mv_cache[ 8 ][1]= my;
- }else{
- assert(IS_SUB_4X4(sub_mb_type));
- mv_cache[ 0 ][0]= mx;
- mv_cache[ 0 ][1]= my;
- }
- }
- }else{
- uint32_t *p= (uint32_t *)&h->mv_cache[list][ scan8[4*i] ][0];
- p[0] = p[1]=
- p[8] = p[9]= 0;
- }
- }
- }
- }else if(!IS_DIRECT(mb_type)){
- int list, mx, my, i;
- //FIXME we should set ref_idx_l? to 0 if we use that later ...
- if(IS_16X16(mb_type)){
- for(list=0; list<2; list++){
- if(h->ref_count[0]>0){
- if(IS_DIR(mb_type, 0, list)){
- const int val= get_te0_golomb(&s->gb, h->ref_count[list]);
- fill_rectangle(&h->ref_cache[list][ scan8[0] ], 4, 4, 8, val, 1);
- }
- }
- }
- for(list=0; list<2; list++){
- if(IS_DIR(mb_type, 0, list)){
- pred_motion(h, 0, 4, list, h->ref_cache[list][ scan8[0] ], &mx, &my);
- mx += get_se_golomb(&s->gb);
- my += get_se_golomb(&s->gb);
- tprintf("final mv:%d %dn", mx, my);
- fill_rectangle(h->mv_cache[list][ scan8[0] ], 4, 4, 8, pack16to32(mx,my), 4);
- }
- }
- }
- else if(IS_16X8(mb_type)){
- for(list=0; list<2; list++){
- if(h->ref_count[list]>0){
- for(i=0; i<2; i++){
- if(IS_DIR(mb_type, i, list)){
- const int val= get_te0_golomb(&s->gb, h->ref_count[list]);
- fill_rectangle(&h->ref_cache[list][ scan8[0] + 16*i ], 4, 2, 8, val, 1);
- }
- }
- }
- }
- for(list=0; list<2; list++){
- for(i=0; i<2; i++){
- if(IS_DIR(mb_type, i, list)){
- pred_16x8_motion(h, 8*i, list, h->ref_cache[list][scan8[0] + 16*i], &mx, &my);
- mx += get_se_golomb(&s->gb);
- my += get_se_golomb(&s->gb);
- tprintf("final mv:%d %dn", mx, my);
- fill_rectangle(h->mv_cache[list][ scan8[0] + 16*i ], 4, 2, 8, pack16to32(mx,my), 4);
- }
- }
- }
- }else{
- assert(IS_8X16(mb_type));
- for(list=0; list<2; list++){
- if(h->ref_count[list]>0){
- for(i=0; i<2; i++){
- if(IS_DIR(mb_type, i, list)){ //FIXME optimize
- const int val= get_te0_golomb(&s->gb, h->ref_count[list]);
- fill_rectangle(&h->ref_cache[list][ scan8[0] + 2*i ], 2, 4, 8, val, 1);
- }
- }
- }
- }
- for(list=0; list<2; list++){
- for(i=0; i<2; i++){
- if(IS_DIR(mb_type, i, list)){
- pred_8x16_motion(h, i*4, list, h->ref_cache[list][ scan8[0] + 2*i ], &mx, &my);
- mx += get_se_golomb(&s->gb);
- my += get_se_golomb(&s->gb);
- tprintf("final mv:%d %dn", mx, my);
- fill_rectangle(h->mv_cache[list][ scan8[0] + 2*i ], 2, 4, 8, pack16to32(mx,my), 4);
- }
- }
- }
- }
- }
- if(IS_INTER(mb_type))
- write_back_motion(h, mb_type);
- if(!IS_INTRA16x16(mb_type)){
- cbp= get_ue_golomb(&s->gb);
- if(cbp > 47){
- av_log(h->s.avctx, AV_LOG_ERROR, "cbp too large (%d) at %d %dn", cbp, s->mb_x, s->mb_y);
- return -1;
- }
- if(IS_INTRA4x4(mb_type))
- cbp= golomb_to_intra4x4_cbp[cbp];
- else
- cbp= golomb_to_inter_cbp[cbp];
- }
- if(cbp || IS_INTRA16x16(mb_type)){
- int i8x8, i4x4, chroma_idx;
- int chroma_qp, dquant;
- GetBitContext *gb= IS_INTRA(mb_type) ? h->intra_gb_ptr : h->inter_gb_ptr;
- const uint8_t *scan, *dc_scan;
- // fill_non_zero_count_cache(h);
- if(IS_INTERLACED(mb_type)){
- scan= field_scan;
- dc_scan= luma_dc_field_scan;
- }else{
- scan= zigzag_scan;
- dc_scan= luma_dc_zigzag_scan;
- }
- dquant= get_se_golomb(&s->gb);
- if( dquant > 25 || dquant < -26 ){
- av_log(h->s.avctx, AV_LOG_ERROR, "dquant out of range (%d) at %d %dn", dquant, s->mb_x, s->mb_y);
- return -1;
- }
- s->qscale += dquant;
- if(((unsigned)s->qscale) > 51){
- if(s->qscale<0) s->qscale+= 52;
- else s->qscale-= 52;
- }
- h->chroma_qp= chroma_qp= get_chroma_qp(h, s->qscale);
- if(IS_INTRA16x16(mb_type)){
- if( decode_residual(h, h->intra_gb_ptr, h->mb, LUMA_DC_BLOCK_INDEX, dc_scan, s->qscale, 16) < 0){
- return -1; //FIXME continue if partotioned and other retirn -1 too
- }
- assert((cbp&15) == 0 || (cbp&15) == 15);
- if(cbp&15){
- for(i8x8=0; i8x8<4; i8x8++){
- for(i4x4=0; i4x4<4; i4x4++){
- const int index= i4x4 + 4*i8x8;
- if( decode_residual(h, h->intra_gb_ptr, h->mb + 16*index, index, scan + 1, s->qscale, 15) < 0 ){
- return -1;
- }
- }
- }
- }else{
- fill_rectangle(&h->non_zero_count_cache[scan8[0]], 4, 4, 8, 0, 1);
- }
- }else{
- for(i8x8=0; i8x8<4; i8x8++){
- if(cbp & (1<<i8x8)){
- for(i4x4=0; i4x4<4; i4x4++){
- const int index= i4x4 + 4*i8x8;
- if( decode_residual(h, gb, h->mb + 16*index, index, scan, s->qscale, 16) <0 ){
- return -1;
- }
- }
- }else{
- uint8_t * const nnz= &h->non_zero_count_cache[ scan8[4*i8x8] ];
- nnz[0] = nnz[1] = nnz[8] = nnz[9] = 0;
- }
- }
- }
- if(cbp&0x30){
- for(chroma_idx=0; chroma_idx<2; chroma_idx++)
- if( decode_residual(h, gb, h->mb + 256 + 16*4*chroma_idx, CHROMA_DC_BLOCK_INDEX, chroma_dc_scan, chroma_qp, 4) < 0){
- return -1;
- }
- }
- if(cbp&0x20){
- for(chroma_idx=0; chroma_idx<2; chroma_idx++){
- for(i4x4=0; i4x4<4; i4x4++){
- const int index= 16 + 4*chroma_idx + i4x4;
- if( decode_residual(h, gb, h->mb + 16*index, index, scan + 1, chroma_qp, 15) < 0){
- return -1;
- }
- }
- }
- }else{
- uint8_t * const nnz= &h->non_zero_count_cache[0];
- nnz[ scan8[16]+0 ] = nnz[ scan8[16]+1 ] =nnz[ scan8[16]+8 ] =nnz[ scan8[16]+9 ] =
- nnz[ scan8[20]+0 ] = nnz[ scan8[20]+1 ] =nnz[ scan8[20]+8 ] =nnz[ scan8[20]+9 ] = 0;
- }
- }else{
- uint8_t * const nnz= &h->non_zero_count_cache[0];
- fill_rectangle(&nnz[scan8[0]], 4, 4, 8, 0, 1);
- nnz[ scan8[16]+0 ] = nnz[ scan8[16]+1 ] =nnz[ scan8[16]+8 ] =nnz[ scan8[16]+9 ] =
- nnz[ scan8[20]+0 ] = nnz[ scan8[20]+1 ] =nnz[ scan8[20]+8 ] =nnz[ scan8[20]+9 ] = 0;
- }
- s->current_picture.qscale_table[mb_xy]= s->qscale;
- write_back_non_zero_count(h);
- return 0;
- }
- static int decode_cabac_mb_type( H264Context *h ) {
- MpegEncContext * const s = &h->s;
- if( h->slice_type == I_TYPE ) {
- const int mb_xy= s->mb_x + s->mb_y*s->mb_stride;
- int ctx = 0;
- int mb_type;
- if( s->mb_x > 0 && !IS_INTRA4x4( s->current_picture.mb_type[mb_xy-1] ) )
- ctx++;
- if( s->mb_y > 0 && !IS_INTRA4x4( s->current_picture.mb_type[mb_xy-s->mb_stride] ) )
- ctx++;
- if( get_cabac( &h->cabac, &h->cabac_state[3+ctx] ) == 0 )
- return 0; /* I4x4 */
- if( get_cabac_terminate( &h->cabac ) )
- return 25; /* PCM */
- mb_type = 1; /* I16x16 */
- if( get_cabac( &h->cabac, &h->cabac_state[3+3] ) )
- mb_type += 12; /* cbp_luma != 0 */
- if( get_cabac( &h->cabac, &h->cabac_state[3+4] ) ) {
- if( get_cabac( &h->cabac, &h->cabac_state[3+5] ) )
- mb_type += 4 * 2; /* cbp_chroma == 2 */
- else
- mb_type += 4 * 1; /* cbp_chroma == 1 */
- }
- if( get_cabac( &h->cabac, &h->cabac_state[3+6] ) )
- mb_type += 2;
- if( get_cabac( &h->cabac, &h->cabac_state[3+7] ) )
- mb_type += 1;
- return mb_type;
- } else if( h->slice_type == P_TYPE ) {
- if( get_cabac( &h->cabac, &h->cabac_state[14] ) == 0 ) {
- /* P-type */
- if( get_cabac( &h->cabac, &h->cabac_state[15] ) == 0 ) {
- if( get_cabac( &h->cabac, &h->cabac_state[16] ) == 0 )
- return 0; /* P_L0_D16x16; */
- else
- return 3; /* P_8x8; */
- } else {
- if( get_cabac( &h->cabac, &h->cabac_state[17] ) == 0 )
- return 2; /* P_L0_D8x16; */
- else
- return 1; /* P_L0_D16x8; */
- }
- } else {
- int mb_type;
- /* I-type */
- if( get_cabac( &h->cabac, &h->cabac_state[17] ) == 0 )
- return 5+0; /* I_4x4 */
- if( get_cabac_terminate( &h->cabac ) )
- return 5+25; /*I_PCM */
- mb_type = 5+1; /* I16x16 */
- if( get_cabac( &h->cabac, &h->cabac_state[17+1] ) )
- mb_type += 12; /* cbp_luma != 0 */
- if( get_cabac( &h->cabac, &h->cabac_state[17+2] ) ) {
- if( get_cabac( &h->cabac, &h->cabac_state[17+2] ) )
- mb_type += 4 * 2; /* cbp_chroma == 2 */
- else
- mb_type += 4 * 1; /* cbp_chroma == 1 */
- }
- if( get_cabac( &h->cabac, &h->cabac_state[17+3] ) )
- mb_type += 2;
- if( get_cabac( &h->cabac, &h->cabac_state[17+3] ) )
- mb_type += 1;
- return mb_type;
- }
- } else {
- /* TODO do others frames types */
- return -1;
- }
- }
- static int decode_cabac_mb_skip( H264Context *h) {
- MpegEncContext * const s = &h->s;
- const int mb_xy = s->mb_x + s->mb_y*s->mb_stride;
- const int mba_xy = mb_xy - 1;
- const int mbb_xy = mb_xy - s->mb_stride;
- int ctx = 0;
- if( s->mb_x > 0 && !IS_SKIP( s->current_picture.mb_type[mba_xy] ) )
- ctx++;
- if( s->mb_y > 0 && !IS_SKIP( s->current_picture.mb_type[mbb_xy] ) )
- ctx++;
- if( h->slice_type == P_TYPE || h->slice_type == SP_TYPE)
- return get_cabac( &h->cabac, &h->cabac_state[11+ctx] );
- else /* B-frame */
- return get_cabac( &h->cabac, &h->cabac_state[24+ctx] );
- }
- static int decode_cabac_mb_intra4x4_pred_mode( H264Context *h, int pred_mode ) {
- int mode = 0;
- if( get_cabac( &h->cabac, &h->cabac_state[68] ) )
- return pred_mode;
- if( get_cabac( &h->cabac, &h->cabac_state[69] ) )
- mode += 1;
- if( get_cabac( &h->cabac, &h->cabac_state[69] ) )
- mode += 2;
- if( get_cabac( &h->cabac, &h->cabac_state[69] ) )
- mode += 4;
- if( mode >= pred_mode )
- return mode + 1;
- else
- return mode;
- }
- static int decode_cabac_mb_chroma_pre_mode( H264Context *h) {
- MpegEncContext * const s = &h->s;
- const int mb_xy = s->mb_x + s->mb_y*s->mb_stride;
- const int mba_xy = mb_xy - 1;
- const int mbb_xy = mb_xy - s->mb_stride;
- int ctx = 0;
- /* No need to test for IS_INTRA4x4 and IS_INTRA16x16, as we set chroma_pred_mode_table to 0 */
- if( s->mb_x > 0 && h->chroma_pred_mode_table[mba_xy] != 0 )
- ctx++;
- if( s->mb_y > 0 && h->chroma_pred_mode_table[mbb_xy] != 0 )
- ctx++;
- if( get_cabac( &h->cabac, &h->cabac_state[64+ctx] ) == 0 )
- return 0;
- if( get_cabac( &h->cabac, &h->cabac_state[64+3] ) == 0 )
- return 1;
- if( get_cabac( &h->cabac, &h->cabac_state[64+3] ) == 0 )
- return 2;
- else
- return 3;
- }
- static const uint8_t block_idx_x[16] = {
- 0, 1, 0, 1, 2, 3, 2, 3, 0, 1, 0, 1, 2, 3, 2, 3
- };
- static const uint8_t block_idx_y[16] = {
- 0, 0, 1, 1, 0, 0, 1, 1, 2, 2, 3, 3, 2, 2, 3, 3
- };
- static const uint8_t block_idx_xy[4][4] = {
- { 0, 2, 8, 10},
- { 1, 3, 9, 11},
- { 4, 6, 12, 14},
- { 5, 7, 13, 15}
- };
- static int decode_cabac_mb_cbp_luma( H264Context *h) {
- MpegEncContext * const s = &h->s;
- const int mb_xy = s->mb_x + s->mb_y*s->mb_stride;
- int cbp = 0;
- int i8x8;
- h->cbp_table[mb_xy] = 0; /* FIXME aaahahahah beurk */
- for( i8x8 = 0; i8x8 < 4; i8x8++ ) {
- int mba_xy = -1;
- int mbb_xy = -1;
- int x, y;
- int ctx = 0;
- x = block_idx_x[4*i8x8];
- y = block_idx_y[4*i8x8];
- if( x > 0 )
- mba_xy = mb_xy;
- else if( s->mb_x > 0 )
- mba_xy = mb_xy - 1;
- if( y > 0 )
- mbb_xy = mb_xy;
- else if( s->mb_y > 0 )
- mbb_xy = mb_xy - s->mb_stride;
- /* No need to test for skip as we put 0 for skip block */
- if( mba_xy >= 0 ) {
- int i8x8a = block_idx_xy[(x-1)&0x03][y]/4;
- if( ((h->cbp_table[mba_xy] >> i8x8a)&0x01) == 0 )
- ctx++;
- }
- if( mbb_xy >= 0 ) {
- int i8x8b = block_idx_xy[x][(y-1)&0x03]/4;
- if( ((h->cbp_table[mbb_xy] >> i8x8b)&0x01) == 0 )
- ctx += 2;
- }
- if( get_cabac( &h->cabac, &h->cabac_state[73 + ctx] ) ) {
- cbp |= 1 << i8x8;
- h->cbp_table[mb_xy] = cbp; /* FIXME aaahahahah beurk */
- }
- }
- return cbp;
- }
- static int decode_cabac_mb_cbp_chroma( H264Context *h) {
- MpegEncContext * const s = &h->s;
- const int mb_xy = s->mb_x + s->mb_y*s->mb_stride;
- int ctx;
- int cbp_a, cbp_b;
- /* No need to test for skip */
- if( s->mb_x > 0 )
- cbp_a = (h->cbp_table[mb_xy-1]>>4)&0x03;
- else
- cbp_a = -1;
- if( s->mb_y > 0 )
- cbp_b = (h->cbp_table[mb_xy-s->mb_stride]>>4)&0x03;
- else
- cbp_b = -1;
- ctx = 0;
- if( cbp_a > 0 ) ctx++;
- if( cbp_b > 0 ) ctx += 2;
- if( get_cabac( &h->cabac, &h->cabac_state[77 + ctx] ) == 0 )
- return 0;
- ctx = 4;
- if( cbp_a == 2 ) ctx++;
- if( cbp_b == 2 ) ctx += 2;
- if( get_cabac( &h->cabac, &h->cabac_state[77 + ctx] ) )
- return 2;
- else
- return 1;
- }
- static int decode_cabac_mb_dqp( H264Context *h) {
- MpegEncContext * const s = &h->s;
- int mbn_xy;
- int ctx = 0;
- int val = 0;
- if( s->mb_x > 0 )
- mbn_xy = s->mb_x + s->mb_y*s->mb_stride - 1;
- else
- mbn_xy = s->mb_width - 1 + (s->mb_y-1)*s->mb_stride;
- if( mbn_xy >= 0 && h->last_qscale_diff != 0 && ( IS_INTRA16x16(s->current_picture.mb_type[mbn_xy] ) || (h->cbp_table[mbn_xy]&0x3f) ) )
- ctx++;
- while( get_cabac( &h->cabac, &h->cabac_state[60 + ctx] ) ) {
- if( ctx < 2 )
- ctx = 2;
- else
- ctx = 3;
- val++;
- }
- if( val&0x01 )
- return (val + 1)/2;
- else
- return -(val + 1)/2;
- }
- static int decode_cabac_mb_sub_type( H264Context *h ) {
- if( get_cabac( &h->cabac, &h->cabac_state[21] ) )
- return 0; /* 8x8 */
- if( !get_cabac( &h->cabac, &h->cabac_state[22] ) )
- return 1; /* 8x4 */
- if( get_cabac( &h->cabac, &h->cabac_state[23] ) )
- return 2; /* 4x8 */
- return 3; /* 4x4 */
- }
- static int decode_cabac_mb_ref( H264Context *h, int list, int n ) {
- int refa = h->ref_cache[list][scan8[n] - 1];
- int refb = h->ref_cache[list][scan8[n] - 8];
- int ref = 0;
- int ctx = 0;
- if( refa > 0 )
- ctx++;
- if( refb > 0 )
- ctx += 2;
- while( get_cabac( &h->cabac, &h->cabac_state[54+ctx] ) ) {
- ref++;
- if( ctx < 4 )
- ctx = 4;
- else
- ctx = 5;
- }
- return ref;
- }
- static int decode_cabac_mb_mvd( H264Context *h, int list, int n, int l ) {
- int amvd = abs( h->mvd_cache[list][scan8[n] - 1][l] ) +
- abs( h->mvd_cache[list][scan8[n] - 8][l] );
- int ctxbase = (l == 0) ? 40 : 47;
- int ctx;
- int mvd = 0;
- if( amvd < 3 )
- ctx = 0;
- else if( amvd > 32 )
- ctx = 2;
- else
- ctx = 1;
- while( mvd < 9 && get_cabac( &h->cabac, &h->cabac_state[ctxbase+ctx] ) ) {
- mvd++;
- if( ctx < 3 )
- ctx = 3;
- else if( ctx < 6 )
- ctx++;
- }
- if( mvd >= 9 ) {
- int k = 3;
- while( get_cabac_bypass( &h->cabac ) ) {
- mvd += 1 << k;
- k++;
- }
- while( k-- ) {
- if( get_cabac_bypass( &h->cabac ) )
- mvd += 1 << k;
- }
- }
- if( mvd != 0 && get_cabac_bypass( &h->cabac ) )
- return -mvd;
- return mvd;
- }
- static int get_cabac_cbf_ctx( H264Context *h, int cat, int idx ) {
- MpegEncContext * const s = &h->s;
- const int mb_xy = s->mb_x + s->mb_y*s->mb_stride;
- int mba_xy = -1;
- int mbb_xy = -1;
- int nza = -1;
- int nzb = -1;
- int ctx = 0;
- if( cat == 0 ) {
- if( s->mb_x > 0 ) {
- mba_xy = mb_xy - 1;
- if( IS_INTRA16x16(s->current_picture.mb_type[mba_xy] ) )
- nza = h->cbp_table[mba_xy]&0x100;
- }
- if( s->mb_y > 0 ) {
- mbb_xy = mb_xy - s->mb_stride;
- if( IS_INTRA16x16(s->current_picture.mb_type[mbb_xy] ) )
- nzb = h->cbp_table[mbb_xy]&0x100;
- }
- } else if( cat == 1 || cat == 2 ) {
- int i8x8a, i8x8b;
- int x, y;
- x = block_idx_x[idx];
- y = block_idx_y[idx];
- if( x > 0 )
- mba_xy = mb_xy;
- else if( s->mb_x > 0 )
- mba_xy = mb_xy - 1;
- if( y > 0 )
- mbb_xy = mb_xy;
- else if( s->mb_y > 0 )
- mbb_xy = mb_xy - s->mb_stride;
- /* No need to test for skip */
- if( mba_xy >= 0 ) {
- i8x8a = block_idx_xy[(x-1)&0x03][y]/4;
- if( !IS_INTRA_PCM(s->current_picture.mb_type[mba_xy] ) &&
- ((h->cbp_table[mba_xy]&0x0f)>>i8x8a))
- nza = h->non_zero_count_cache[scan8[idx] - 1];
- }
- if( mbb_xy >= 0 ) {
- i8x8b = block_idx_xy[x][(y-1)&0x03]/4;
- if( !IS_INTRA_PCM(s->current_picture.mb_type[mbb_xy] ) &&
- ((h->cbp_table[mbb_xy]&0x0f)>>i8x8b))
- nzb = h->non_zero_count_cache[scan8[idx] - 8];
- }
- } else if( cat == 3 ) {
- if( s->mb_x > 0 ) {
- mba_xy = mb_xy - 1;
- if( !IS_INTRA_PCM(s->current_picture.mb_type[mba_xy] ) &&
- (h->cbp_table[mba_xy]&0x30) )
- nza = (h->cbp_table[mba_xy]>>(6+idx))&0x01;
- }
- if( s->mb_y > 0 ) {
- mbb_xy = mb_xy - s->mb_stride;
- if( !IS_INTRA_PCM(s->current_picture.mb_type[mbb_xy] ) &&
- (h->cbp_table[mbb_xy]&0x30) )
- nzb = (h->cbp_table[mbb_xy]>>(6+idx))&0x01;
- }
- } else if( cat == 4 ) {
- int idxc = idx % 4 ;
- if( idxc == 1 || idxc == 3 )
- mba_xy = mb_xy;
- else if( s->mb_x > 0 )
- mba_xy = mb_xy -1;
- if( idxc == 2 || idxc == 3 )
- mbb_xy = mb_xy;
- else if( s->mb_y > 0 )
- mbb_xy = mb_xy - s->mb_stride;
- if( mba_xy >= 0 &&
- !IS_INTRA_PCM(s->current_picture.mb_type[mba_xy] ) &&
- (h->cbp_table[mba_xy]&0x30) == 0x20 )
- nza = h->non_zero_count_cache[scan8[16+idx] - 1];
- if( mbb_xy >= 0 &&
- !IS_INTRA_PCM(s->current_picture.mb_type[mbb_xy] ) &&
- (h->cbp_table[mbb_xy]&0x30) == 0x20 )
- nzb = h->non_zero_count_cache[scan8[16+idx] - 8];
- }
- if( ( mba_xy < 0 && IS_INTRA( s->current_picture.mb_type[mb_xy] ) ) ||
- ( mba_xy >= 0 && IS_INTRA_PCM(s->current_picture.mb_type[mba_xy] ) ) ||
- nza > 0 )
- ctx++;
- if( ( mbb_xy < 0 && IS_INTRA( s->current_picture.mb_type[mb_xy] ) ) ||
- ( mbb_xy >= 0 && IS_INTRA_PCM(s->current_picture.mb_type[mbb_xy] ) ) ||
- nzb > 0 )
- ctx += 2;
- return ctx + 4 * cat;
- }
- static int decode_cabac_residual( H264Context *h, DCTELEM *block, int cat, int n, const uint8_t *scantable, int qp, int max_coeff) {
- const int mb_xy = h->s.mb_x + h->s.mb_y*h->s.mb_stride;
- const uint16_t *qmul= dequant_coeff[qp];
- static const int significant_coeff_flag_offset[5] = { 0, 15, 29, 44, 47 };
- static const int last_significant_coeff_flag_offset[5] = { 0, 15, 29, 44, 47 };
- static const int coeff_abs_level_m1_offset[5] = { 0, 10, 20, 30, 39 };
- int coeff[16];
- int last = 0;
- int coeff_count = 0;
- int nz[16] = {0};
- int i;
- int abslevel1 = 0;
- int abslevelgt1 = 0;
- /* cat: 0-> DC 16x16 n = 0
- * 1-> AC 16x16 n = luma4x4idx
- * 2-> Luma4x4 n = luma4x4idx
- * 3-> DC Chroma n = iCbCr
- * 4-> AC Chroma n = 4 * iCbCr + chroma4x4idx
- */
- /* read coded block flag */
- if( get_cabac( &h->cabac, &h->cabac_state[85 + get_cabac_cbf_ctx( h, cat, n ) ] ) == 0 ) {
- if( cat == 1 || cat == 2 )
- h->non_zero_count_cache[scan8[n]] = 0;
- else if( cat == 4 )
- h->non_zero_count_cache[scan8[16+n]] = 0;
- return 0;
- }
- while( last < max_coeff - 1 ) {
- int ctx = FFMIN( last, max_coeff - 2 );
- if( get_cabac( &h->cabac, &h->cabac_state[105+significant_coeff_flag_offset[cat]+ctx] ) == 0 ) {
- nz[last++] = 0;
- }
- else {
- nz[last++] = 1;
- coeff_count++;
- if( get_cabac( &h->cabac, &h->cabac_state[166+last_significant_coeff_flag_offset[cat]+ctx] ) ) {
- while( last < max_coeff ) {
- nz[last++] = 0;
- }
- break;
- }
- }
- }
- if( last == max_coeff -1 ) {
- nz[last++] = 1;
- coeff_count++;
- }
- if( cat == 0 && coeff_count > 0 )
- h->cbp_table[mb_xy] |= 0x100;
- else if( cat == 1 || cat == 2 )
- h->non_zero_count_cache[scan8[n]] = coeff_count;
- else if( cat == 3 && coeff_count > 0 )
- h->cbp_table[mb_xy] |= 0x40 << n;
- else if( cat == 4 )
- h->non_zero_count_cache[scan8[16+n]] = coeff_count;
- for( i = coeff_count - 1; i >= 0; i-- ) {
- int coeff_abs_m1;
- int ctx = (abslevelgt1 != 0 ? 0 : FFMIN( 4, abslevel1 + 1 )) + coeff_abs_level_m1_offset[cat];
- if( get_cabac( &h->cabac, &h->cabac_state[227+ctx] ) == 0 ) {
- coeff_abs_m1 = 0;
- } else {
- coeff_abs_m1 = 1;
- ctx = 5 + FFMIN( 4, abslevelgt1 ) + coeff_abs_level_m1_offset[cat];
- while( coeff_abs_m1 < 14 && get_cabac( &h->cabac, &h->cabac_state[227+ctx] ) ) {
- coeff_abs_m1++;
- }
- }
- if( coeff_abs_m1 >= 14 ) {
- int j = 0;
- while( get_cabac_bypass( &h->cabac ) ) {
- coeff_abs_m1 += 1 << j;
- j++;
- }
- while( j-- ) {
- if( get_cabac_bypass( &h->cabac ) )
- coeff_abs_m1 += 1 << j ;
- }
- }
- if( get_cabac_bypass( &h->cabac ) )
- coeff[i] = -1 *( coeff_abs_m1 + 1 );
- else
- coeff[i] = coeff_abs_m1 + 1;
- if( coeff_abs_m1 == 0 )
- abslevel1++;
- else
- abslevelgt1++;
- }
- if( cat == 0 || cat == 3 ) { /* DC */
- int j;
- for( i = 0, j = 0; j < coeff_count; i++ ) {
- if( nz[i] ) {
- block[scantable[i]] = coeff[j];
- j++;
- }
- }
- } else { /* AC */
- int j;
- for( i = 0, j = 0; j < coeff_count; i++ ) {
- if( nz[i] ) {
- block[scantable[i]] = coeff[j] * qmul[scantable[i]];
- j++;
- }
- }
- }
- return 0;
- }
- /**
- * decodes a macroblock
- * @returns 0 if ok, AC_ERROR / DC_ERROR / MV_ERROR if an error is noticed
- */
- static int decode_mb_cabac(H264Context *h) {
- MpegEncContext * const s = &h->s;
- const int mb_xy= s->mb_x + s->mb_y*s->mb_stride;
- int mb_type, partition_count, cbp = 0;
- s->dsp.clear_blocks(h->mb); //FIXME avoid if allready clear (move after skip handlong?)
- if( h->slice_type == B_TYPE ) {
- av_log( h->s.avctx, AV_LOG_ERROR, "B-frame not supported with CABACn" );
- return -1;
- }
- if( h->sps.mb_aff ) {
- av_log( h->s.avctx, AV_LOG_ERROR, "Fields not supported with CABACn" );
- return -1;
- }
- if( h->slice_type != I_TYPE && h->slice_type != SI_TYPE ) {
- /* read skip flags */
- if( decode_cabac_mb_skip( h ) ) {
- int mx, my;
- /* skip mb */
- mb_type= MB_TYPE_16x16|MB_TYPE_P0L0|MB_TYPE_P1L0|MB_TYPE_SKIP;
- memset(h->non_zero_count[mb_xy], 0, 16);
- memset(h->non_zero_count_cache + 8, 0, 8*5); //FIXME ugly, remove pfui
- #if 0
- if(h->sps.mb_aff && s->mb_skip_run==0 && (s->mb_y&1)==0){
- h->mb_field_decoding_flag= get_bits1(&s->gb);
- }
- if(h->mb_field_decoding_flag)
- mb_type|= MB_TYPE_INTERLACED;
- #endif
- fill_caches(h, mb_type); //FIXME check what is needed and what not ...
- pred_pskip_motion(h, &mx, &my);
- fill_rectangle(&h->ref_cache[0][scan8[0]], 4, 4, 8, 0, 1);
- fill_rectangle( h->mvd_cache[0][scan8[0]], 4, 4, 8, pack16to32(0,0), 4);
- fill_rectangle( h->mv_cache[0][scan8[0]], 4, 4, 8, pack16to32(mx,my), 4);
- write_back_motion(h, mb_type);
- s->current_picture.mb_type[mb_xy]= mb_type; //FIXME SKIP type
- s->current_picture.qscale_table[mb_xy]= s->qscale;
- h->slice_table[ mb_xy ]= h->slice_num;
- h->cbp_table[mb_xy] = 0;
- h->chroma_pred_mode_table[mb_xy] = 0;
- h->last_qscale_diff = 0;
- h->prev_mb_skiped= 1;
- return 0;
- }
- }
- h->prev_mb_skiped = 0;
- if( ( mb_type = decode_cabac_mb_type( h ) ) < 0 ) {
- av_log( h->s.avctx, AV_LOG_ERROR, "decode_cabac_mb_type failedn" );
- return -1;
- }
- if( h->slice_type == P_TYPE ) {
- if( mb_type < 5) {
- partition_count= p_mb_type_info[mb_type].partition_count;
- mb_type= p_mb_type_info[mb_type].type;
- } else {
- mb_type -= 5;
- goto decode_intra_mb;
- }
- } else {
- assert(h->slice_type == I_TYPE);
- decode_intra_mb:
- partition_count = 0;
- cbp= i_mb_type_info[mb_type].cbp;
- h->intra16x16_pred_mode= i_mb_type_info[mb_type].pred_mode;
- mb_type= i_mb_type_info[mb_type].type;
- }
- #if 0
- if(h->mb_field_decoding_flag)
- mb_type |= MB_TYPE_INTERLACED;
- #endif
- s->current_picture.mb_type[mb_xy]= mb_type;
- h->slice_table[ mb_xy ]= h->slice_num;
- if(IS_INTRA_PCM(mb_type)) {
- /* TODO */
- h->cbp_table[mb_xy] = 0xf +4*2;
- h->chroma_pred_mode_table[mb_xy] = 0;
- s->current_picture.qscale_table[mb_xy]= s->qscale;
- return -1;
- }
- fill_caches(h, mb_type);
- if( IS_INTRA( mb_type ) ) {
- if( IS_INTRA4x4( mb_type ) ) {
- int i;
- for( i = 0; i < 16; i++ ) {
- int pred = pred_intra_mode( h, i );
- h->intra4x4_pred_mode_cache[ scan8[i] ] = decode_cabac_mb_intra4x4_pred_mode( h, pred );
- //av_log( s->avctx, AV_LOG_ERROR, "i4x4 pred=%d mode=%dn", pred, h->intra4x4_pred_mode_cache[ scan8[i] ] );
- }
- write_back_intra_pred_mode(h);
- if( check_intra4x4_pred_mode(h) < 0 ) return -1;
- } else {
- h->intra16x16_pred_mode= check_intra_pred_mode( h, h->intra16x16_pred_mode );
- if( h->intra16x16_pred_mode < 0 ) return -1;
- }
- h->chroma_pred_mode_table[mb_xy] =
- h->chroma_pred_mode = decode_cabac_mb_chroma_pre_mode( h );
- h->chroma_pred_mode= check_intra_pred_mode( h, h->chroma_pred_mode );
- if( h->chroma_pred_mode < 0 ) return -1;
- } else if( partition_count == 4 ) {
- int i, j, sub_partition_count[4], list, ref[2][4];
- /* Only P-frame */
- for( i = 0; i < 4; i++ ) {
- h->sub_mb_type[i] = decode_cabac_mb_sub_type( h );
- sub_partition_count[i]= p_sub_mb_type_info[ h->sub_mb_type[i] ].partition_count;
- h->sub_mb_type[i]= p_sub_mb_type_info[ h->sub_mb_type[i] ].type;
- }
- for( list = 0; list < 2; list++ ) {
- if( h->ref_count[list] > 0 ) {
- for( i = 0; i < 4; i++ ) {
- if(IS_DIR(h->sub_mb_type[i], 0, list) && !IS_DIRECT(h->sub_mb_type[i])){
- if( h->ref_count[list] > 1 )
- ref[list][i] = decode_cabac_mb_ref( h, list, 4*i );
- else
- ref[list][i] = 0;
- } else {
- ref[list][i] = -1;
- }
- h->ref_cache[list][ scan8[4*i]+1 ]=
- h->ref_cache[list][ scan8[4*i]+8 ]=h->ref_cache[list][ scan8[4*i]+9 ]= ref[list][i];
- }
- }
- }
- for(list=0; list<2; list++){
- for(i=0; i<4; i++){
- h->ref_cache[list][ scan8[4*i] ]=h->ref_cache[list][ scan8[4*i]+1 ];
- if(IS_DIR(h->sub_mb_type[i], 0, list) && !IS_DIRECT(h->sub_mb_type[i])){
- const int sub_mb_type= h->sub_mb_type[i];
- const int block_width= (sub_mb_type & (MB_TYPE_16x16|MB_TYPE_16x8)) ? 2 : 1;
- for(j=0; j<sub_partition_count[i]; j++){
- int mpx, mpy;
- int mx, my;
- const int index= 4*i + block_width*j;
- int16_t (* mv_cache)[2]= &h->mv_cache[list][ scan8[index] ];
- int16_t (* mvd_cache)[2]= &h->mvd_cache[list][ scan8[index] ];
- pred_motion(h, index, block_width, list, h->ref_cache[list][ scan8[index] ], &mpx, &mpy);
- mx = mpx + decode_cabac_mb_mvd( h, list, index, 0 );
- my = mpy + decode_cabac_mb_mvd( h, list, index, 1 );
- tprintf("final mv:%d %dn", mx, my);
- if(IS_SUB_8X8(sub_mb_type)){
- mv_cache[ 0 ][0]= mv_cache[ 1 ][0]=
- mv_cache[ 8 ][0]= mv_cache[ 9 ][0]= mx;
- mv_cache[ 0 ][1]= mv_cache[ 1 ][1]=
- mv_cache[ 8 ][1]= mv_cache[ 9 ][1]= my;
- mvd_cache[ 0 ][0]= mvd_cache[ 1 ][0]=
- mvd_cache[ 8 ][0]= mvd_cache[ 9 ][0]= mx - mpx;
- mvd_cache[ 0 ][1]= mvd_cache[ 1 ][1]=
- mvd_cache[ 8 ][1]= mvd_cache[ 9 ][1]= my - mpy;
- }else if(IS_SUB_8X4(sub_mb_type)){
- mv_cache[ 0 ][0]= mv_cache[ 1 ][0]= mx;
- mv_cache[ 0 ][1]= mv_cache[ 1 ][1]= my;
- mvd_cache[ 0 ][0]= mvd_cache[ 1 ][0]= mx- mpx;
- mvd_cache[ 0 ][1]= mvd_cache[ 1 ][1]= my - mpy;
- }else if(IS_SUB_4X8(sub_mb_type)){
- mv_cache[ 0 ][0]= mv_cache[ 8 ][0]= mx;
- mv_cache[ 0 ][1]= mv_cache[ 8 ][1]= my;
- mvd_cache[ 0 ][0]= mvd_cache[ 8 ][0]= mx - mpx;
- mvd_cache[ 0 ][1]= mvd_cache[ 8 ][1]= my - mpy;
- }else{
- assert(IS_SUB_4X4(sub_mb_type));
- mv_cache[ 0 ][0]= mx;
- mv_cache[ 0 ][1]= my;
- mvd_cache[ 0 ][0]= mx - mpx;
- mvd_cache[ 0 ][1]= my - mpy;
- }
- }
- }else{
- uint32_t *p= (uint32_t *)&h->mv_cache[list][ scan8[4*i] ][0];
- uint32_t *pd= (uint32_t *)&h->mvd_cache[list][ scan8[4*i] ][0];
- p[0] = p[1] = p[8] = p[9] = 0;
- pd[0]= pd[1]= pd[8]= pd[9]= 0;
- }
- }
- }
- } else if( !IS_DIRECT(mb_type) ) {
- int list, mx, my, i, mpx, mpy;
- if(IS_16X16(mb_type)){
- for(list=0; list<2; list++){
- if(IS_DIR(mb_type, 0, list)){
- if(h->ref_count[list] > 0 ){
- const int ref = h->ref_count[list] > 1 ? decode_cabac_mb_ref( h, list, 0 ) : 0;
- fill_rectangle(&h->ref_cache[list][ scan8[0] ], 4, 4, 8, ref, 1);
- }
- }
- }
- for(list=0; list<2; list++){
- if(IS_DIR(mb_type, 0, list)){
- pred_motion(h, 0, 4, list, h->ref_cache[list][ scan8[0] ], &mpx, &mpy);
- mx = mpx + decode_cabac_mb_mvd( h, list, 0, 0 );
- my = mpy + decode_cabac_mb_mvd( h, list, 0, 1 );
- tprintf("final mv:%d %dn", mx, my);
- fill_rectangle(h->mvd_cache[list][ scan8[0] ], 4, 4, 8, pack16to32(mx-mpx,my-mpy), 4);
- fill_rectangle(h->mv_cache[list][ scan8[0] ], 4, 4, 8, pack16to32(mx,my), 4);
- }
- }
- }
- else if(IS_16X8(mb_type)){
- for(list=0; list<2; list++){
- if(h->ref_count[list]>0){
- for(i=0; i<2; i++){
- if(IS_DIR(mb_type, i, list)){
- const int ref= h->ref_count[list] > 1 ? decode_cabac_mb_ref( h, list, 8*i ) : 0;
- fill_rectangle(&h->ref_cache[list][ scan8[0] + 16*i ], 4, 2, 8, ref, 1);
- }
- }
- }
- }
- for(list=0; list<2; list++){
- for(i=0; i<2; i++){
- if(IS_DIR(mb_type, i, list)){
- pred_16x8_motion(h, 8*i, list, h->ref_cache[list][scan8[0] + 16*i], &mpx, &mpy);
- mx = mpx + decode_cabac_mb_mvd( h, list, 8*i, 0 );
- my = mpy + decode_cabac_mb_mvd( h, list, 8*i, 1 );
- tprintf("final mv:%d %dn", mx, my);
- fill_rectangle(h->mvd_cache[list][ scan8[0] + 16*i ], 4, 2, 8, pack16to32(mx-mpx,my-mpy), 4);
- fill_rectangle(h->mv_cache[list][ scan8[0] + 16*i ], 4, 2, 8, pack16to32(mx,my), 4);
- }
- }
- }
- }else{
- assert(IS_8X16(mb_type));
- for(list=0; list<2; list++){
- if(h->ref_count[list]>0){
- for(i=0; i<2; i++){
- if(IS_DIR(mb_type, i, list)){ //FIXME optimize
- const int ref= h->ref_count[list] > 1 ? decode_cabac_mb_ref( h, list, 4*i ) : 0;
- fill_rectangle(&h->ref_cache[list][ scan8[0] + 2*i ], 2, 4, 8, ref, 1);
- }
- }
- }
- }
- for(list=0; list<2; list++){
- for(i=0; i<2; i++){
- if(IS_DIR(mb_type, i, list)){
- pred_8x16_motion(h, i*4, list, h->ref_cache[list][ scan8[0] + 2*i ], &mpx, &mpy);
- mx = mpx + decode_cabac_mb_mvd( h, list, 4*i, 0 );
- my = mpy + decode_cabac_mb_mvd( h, list, 4*i, 1 );
- tprintf("final mv:%d %dn", mx, my);
- fill_rectangle(h->mvd_cache[list][ scan8[0] + 2*i ], 2, 4, 8, pack16to32(mx-mpx,my-mpy), 4);
- fill_rectangle(h->mv_cache[list][ scan8[0] + 2*i ], 2, 4, 8, pack16to32(mx,my), 4);
- }
- }
- }
- }
- }
- if( IS_INTER( mb_type ) ) {
- h->chroma_pred_mode_table[mb_xy] = 0;
- write_back_motion( h, mb_type );
- }
- if( !IS_INTRA16x16( mb_type ) ) {
- cbp = decode_cabac_mb_cbp_luma( h );
- cbp |= decode_cabac_mb_cbp_chroma( h ) << 4;
- }
- h->cbp_table[mb_xy] = cbp;
- if( cbp || IS_INTRA16x16( mb_type ) ) {
- const uint8_t *scan, *dc_scan;
- int dqp;
- if(IS_INTERLACED(mb_type)){
- scan= field_scan;
- dc_scan= luma_dc_field_scan;
- }else{
- scan= zigzag_scan;
- dc_scan= luma_dc_zigzag_scan;
- }
- h->last_qscale_diff = dqp = decode_cabac_mb_dqp( h );
- s->qscale += dqp;
- if(((unsigned)s->qscale) > 51){
- if(s->qscale<0) s->qscale+= 52;
- else s->qscale-= 52;
- }
- h->chroma_qp = get_chroma_qp(h, s->qscale);
- if( IS_INTRA16x16( mb_type ) ) {
- int i;
- //av_log( s->avctx, AV_LOG_ERROR, "INTRA16x16 DCn" );
- if( decode_cabac_residual( h, h->mb, 0, 0, dc_scan, s->qscale, 16) < 0)
- return -1;
- if( cbp&15 ) {
- for( i = 0; i < 16; i++ ) {
- //av_log( s->avctx, AV_LOG_ERROR, "INTRA16x16 AC:%dn", i );
- if( decode_cabac_residual(h, h->mb + 16*i, 1, i, scan + 1, s->qscale, 15) < 0 )
- return -1;
- }
- } else {
- fill_rectangle(&h->non_zero_count_cache[scan8[0]], 4, 4, 8, 0, 1);
- }
- } else {
- int i8x8, i4x4;
- for( i8x8 = 0; i8x8 < 4; i8x8++ ) {
- if( cbp & (1<<i8x8) ) {
- for( i4x4 = 0; i4x4 < 4; i4x4++ ) {
- const int index = 4*i8x8 + i4x4;
- //av_log( s->avctx, AV_LOG_ERROR, "Luma4x4: %dn", index );
- if( decode_cabac_residual(h, h->mb + 16*index, 2, index, scan, s->qscale, 16) < 0 )
- return -1;
- }
- } else {
- uint8_t * const nnz= &h->non_zero_count_cache[ scan8[4*i8x8] ];
- nnz[0] = nnz[1] = nnz[8] = nnz[9] = 0;
- }
- }
- }
- if( cbp&0x30 ){
- int c;
- for( c = 0; c < 2; c++ ) {
- //av_log( s->avctx, AV_LOG_ERROR, "INTRA C%d-DCn",c );
- if( decode_cabac_residual(h, h->mb + 256 + 16*4*c, 3, c, chroma_dc_scan, h->chroma_qp, 4) < 0)
- return -1;
- }
- }
- if( cbp&0x20 ) {
- int c, i;
- for( c = 0; c < 2; c++ ) {
- for( i = 0; i < 4; i++ ) {
- const int index = 16 + 4 * c + i;
- //av_log( s->avctx, AV_LOG_ERROR, "INTRA C%d-AC %dn",c, index - 16 );
- if( decode_cabac_residual(h, h->mb + 16*index, 4, index - 16, scan + 1, h->chroma_qp, 15) < 0)
- return -1;
- }
- }
- } else {
- uint8_t * const nnz= &h->non_zero_count_cache[0];
- nnz[ scan8[16]+0 ] = nnz[ scan8[16]+1 ] =nnz[ scan8[16]+8 ] =nnz[ scan8[16]+9 ] =
- nnz[ scan8[20]+0 ] = nnz[ scan8[20]+1 ] =nnz[ scan8[20]+8 ] =nnz[ scan8[20]+9 ] = 0;
- }
- } else {
- memset( &h->non_zero_count_cache[8], 0, 8*5 );
- }
- s->current_picture.qscale_table[mb_xy]= s->qscale;
- write_back_non_zero_count(h);
- return 0;
- }
- static void filter_mb_edgev( H264Context *h, uint8_t *pix, int stride, int bS[4], int qp ) {
- int i, d;
- const int index_a = clip( qp + h->slice_alpha_c0_offset, 0, 51 );
- const int alpha = alpha_table[index_a];
- const int beta = beta_table[clip( qp + h->slice_beta_offset, 0, 51 )];
- for( i = 0; i < 4; i++ ) {
- if( bS[i] == 0 ) {
- pix += 4 * stride;
- continue;
- }
- if( bS[i] < 4 ) {
- const int tc0 = tc0_table[index_a][bS[i] - 1];
- /* 4px edge length */
- for( d = 0; d < 4; d++ ) {
- const int p0 = pix[-1];
- const int p1 = pix[-2];
- const int p2 = pix[-3];
- const int q0 = pix[0];
- const int q1 = pix[1];
- const int q2 = pix[2];
- if( ABS( p0 - q0 ) < alpha &&
- ABS( p1 - p0 ) < beta &&
- ABS( q1 - q0 ) < beta ) {
- int tc = tc0;
- int i_delta;
- if( ABS( p2 - p0 ) < beta ) {
- pix[-2] = p1 + clip( ( p2 + ( ( p0 + q0 + 1 ) >> 1 ) - ( p1 << 1 ) ) >> 1, -tc0, tc0 );
- tc++;
- }
- if( ABS( q2 - q0 ) < beta ) {
- pix[1] = q1 + clip( ( q2 + ( ( p0 + q0 + 1 ) >> 1 ) - ( q1 << 1 ) ) >> 1, -tc0, tc0 );
- tc++;
- }
- i_delta = clip( (((q0 - p0 ) << 2) + (p1 - q1) + 4) >> 3, -tc, tc );
- pix[-1] = clip_uint8( p0 + i_delta ); /* p0' */
- pix[0] = clip_uint8( q0 - i_delta ); /* q0' */
- }
- pix += stride;
- }
- }else{
- /* 4px edge length */
- for( d = 0; d < 4; d++ ) {
- const int p0 = pix[-1];
- const int p1 = pix[-2];
- const int p2 = pix[-3];
- const int q0 = pix[0];
- const int q1 = pix[1];
- const int q2 = pix[2];
- if( ABS( p0 - q0 ) < alpha &&
- ABS( p1 - p0 ) < beta &&
- ABS( q1 - q0 ) < beta ) {
- if(ABS( p0 - q0 ) < (( alpha >> 2 ) + 2 )){
- if( ABS( p2 - p0 ) < beta)
- {
- const int p3 = pix[-4];
- /* p0', p1', p2' */
- pix[-1] = ( p2 + 2*p1 + 2*p0 + 2*q0 + q1 + 4 ) >> 3;
- pix[-2] = ( p2 + p1 + p0 + q0 + 2 ) >> 2;
- pix[-3] = ( 2*p3 + 3*p2 + p1 + p0 + q0 + 4 ) >> 3;
- } else {
- /* p0' */
- pix[-1] = ( 2*p1 + p0 + q1 + 2 ) >> 2;
- }
- if( ABS( q2 - q0 ) < beta)
- {
- const int q3 = pix[3];
- /* q0', q1', q2' */
- pix[0] = ( p1 + 2*p0 + 2*q0 + 2*q1 + q2 + 4 ) >> 3;
- pix[1] = ( p0 + q0 + q1 + q2 + 2 ) >> 2;
- pix[2] = ( 2*q3 + 3*q2 + q1 + q0 + p0 + 4 ) >> 3;
- } else {
- /* q0' */
- pix[0] = ( 2*q1 + q0 + p1 + 2 ) >> 2;
- }
- }else{
- /* p0', q0' */
- pix[-1] = ( 2*p1 + p0 + q1 + 2 ) >> 2;
- pix[ 0] = ( 2*q1 + q0 + p1 + 2 ) >> 2;
- }
- }
- pix += stride;
- }
- }
- }
- }
- static void filter_mb_edgecv( H264Context *h, uint8_t *pix, int stride, int bS[4], int qp ) {
- int i, d;
- const int index_a = clip( qp + h->slice_alpha_c0_offset, 0, 51 );
- const int alpha = alpha_table[index_a];
- const int beta = beta_table[clip( qp + h->slice_beta_offset, 0, 51 )];
- for( i = 0; i < 4; i++ ) {
- if( bS[i] == 0 ) {
- pix += 2 * stride;
- continue;
- }
- if( bS[i] < 4 ) {
- const int tc = tc0_table[index_a][bS[i] - 1] + 1;
- /* 2px edge length (because we use same bS than the one for luma) */
- for( d = 0; d < 2; d++ ){
- const int p0 = pix[-1];
- const int p1 = pix[-2];
- const int q0 = pix[0];
- const int q1 = pix[1];
- if( ABS( p0 - q0 ) < alpha &&
- ABS( p1 - p0 ) < beta &&
- ABS( q1 - q0 ) < beta ) {
- const int i_delta = clip( (((q0 - p0 ) << 2) + (p1 - q1) + 4) >> 3, -tc, tc );
- pix[-1] = clip_uint8( p0 + i_delta ); /* p0' */
- pix[0] = clip_uint8( q0 - i_delta ); /* q0' */
- }
- pix += stride;
- }
- }else{
- /* 2px edge length (because we use same bS than the one for luma) */
- for( d = 0; d < 2; d++ ){
- const int p0 = pix[-1];
- const int p1 = pix[-2];
- const int q0 = pix[0];
- const int q1 = pix[1];
- if( ABS( p0 - q0 ) < alpha &&
- ABS( p1 - p0 ) < beta &&
- ABS( q1 - q0 ) < beta ) {
- pix[-1] = ( 2*p1 + p0 + q1 + 2 ) >> 2; /* p0' */
- pix[0] = ( 2*q1 + q0 + p1 + 2 ) >> 2; /* q0' */
- }
- pix += stride;
- }
- }
- }
- }
- static void filter_mb_edgeh( H264Context *h, uint8_t *pix, int stride, int bS[4], int qp ) {
- int i, d;
- const int index_a = clip( qp + h->slice_alpha_c0_offset, 0, 51 );
- const int alpha = alpha_table[index_a];
- const int beta = beta_table[clip( qp + h->slice_beta_offset, 0, 51 )];
- const int pix_next = stride;
- for( i = 0; i < 4; i++ ) {
- if( bS[i] == 0 ) {
- pix += 4;
- continue;
- }
- if( bS[i] < 4 ) {
- const int tc0 = tc0_table[index_a][bS[i] - 1];
- /* 4px edge length */
- for( d = 0; d < 4; d++ ) {
- const int p0 = pix[-1*pix_next];
- const int p1 = pix[-2*pix_next];
- const int p2 = pix[-3*pix_next];
- const int q0 = pix[0];
- const int q1 = pix[1*pix_next];
- const int q2 = pix[2*pix_next];
- if( ABS( p0 - q0 ) < alpha &&
- ABS( p1 - p0 ) < beta &&
- ABS( q1 - q0 ) < beta ) {
- int tc = tc0;
- int i_delta;
- if( ABS( p2 - p0 ) < beta ) {
- pix[-2*pix_next] = p1 + clip( ( p2 + ( ( p0 + q0 + 1 ) >> 1 ) - ( p1 << 1 ) ) >> 1, -tc0, tc0 );
- tc++;
- }
- if( ABS( q2 - q0 ) < beta ) {
- pix[pix_next] = q1 + clip( ( q2 + ( ( p0 + q0 + 1 ) >> 1 ) - ( q1 << 1 ) ) >> 1, -tc0, tc0 );
- tc++;
- }
- i_delta = clip( (((q0 - p0 ) << 2) + (p1 - q1) + 4) >> 3, -tc, tc );
- pix[-pix_next] = clip_uint8( p0 + i_delta ); /* p0' */
- pix[0] = clip_uint8( q0 - i_delta ); /* q0' */
- }
- pix++;
- }
- }else{
- /* 4px edge length */
- for( d = 0; d < 4; d++ ) {
- const int p0 = pix[-1*pix_next];
- const int p1 = pix[-2*pix_next];
- const int p2 = pix[-3*pix_next];
- const int q0 = pix[0];
- const int q1 = pix[1*pix_next];
- const int q2 = pix[2*pix_next];
- if( ABS( p0 - q0 ) < alpha &&
- ABS( p1 - p0 ) < beta &&
- ABS( q1 - q0 ) < beta ) {
- const int p3 = pix[-4*pix_next];
- const int q3 = pix[ 3*pix_next];
- if(ABS( p0 - q0 ) < (( alpha >> 2 ) + 2 )){
- if( ABS( p2 - p0 ) < beta) {
- /* p0', p1', p2' */
- pix[-1*pix_next] = ( p2 + 2*p1 + 2*p0 + 2*q0 + q1 + 4 ) >> 3;
- pix[-2*pix_next] = ( p2 + p1 + p0 + q0 + 2 ) >> 2;
- pix[-3*pix_next] = ( 2*p3 + 3*p2 + p1 + p0 + q0 + 4 ) >> 3;
- } else {
- /* p0' */
- pix[-1*pix_next] = ( 2*p1 + p0 + q1 + 2 ) >> 2;
- }
- if( ABS( q2 - q0 ) < beta) {
- /* q0', q1', q2' */
- pix[0*pix_next] = ( p1 + 2*p0 + 2*q0 + 2*q1 + q2 + 4 ) >> 3;
- pix[1*pix_next] = ( p0 + q0 + q1 + q2 + 2 ) >> 2;
- pix[2*pix_next] = ( 2*q3 + 3*q2 + q1 + q0 + p0 + 4 ) >> 3;
- } else {
- /* q0' */
- pix[0*pix_next] = ( 2*q1 + q0 + p1 + 2 ) >> 2;
- }
- }else{
- /* p0', q0' */
- pix[-1*pix_next] = ( 2*p1 + p0 + q1 + 2 ) >> 2;
- pix[ 0*pix_next] = ( 2*q1 + q0 + p1 + 2 ) >> 2;
- }
- }
- pix++;
- }
- }
- }
- }
- static void filter_mb_edgech( H264Context *h, uint8_t *pix, int stride, int bS[4], int qp ) {
- int i, d;
- const int index_a = clip( qp + h->slice_alpha_c0_offset, 0, 51 );
- const int alpha = alpha_table[index_a];
- const int beta = beta_table[clip( qp + h->slice_beta_offset, 0, 51 )];
- const int pix_next = stride;
- for( i = 0; i < 4; i++ )
- {
- if( bS[i] == 0 ) {
- pix += 2;
- continue;
- }
- if( bS[i] < 4 ) {
- int tc = tc0_table[index_a][bS[i] - 1] + 1;
- /* 2px edge length (see deblocking_filter_edgecv) */
- for( d = 0; d < 2; d++ ) {
- const int p0 = pix[-1*pix_next];
- const int p1 = pix[-2*pix_next];
- const int q0 = pix[0];
- const int q1 = pix[1*pix_next];
- if( ABS( p0 - q0 ) < alpha &&
- ABS( p1 - p0 ) < beta &&
- ABS( q1 - q0 ) < beta ) {
- int i_delta = clip( (((q0 - p0 ) << 2) + (p1 - q1) + 4) >> 3, -tc, tc );
- pix[-pix_next] = clip_uint8( p0 + i_delta ); /* p0' */
- pix[0] = clip_uint8( q0 - i_delta ); /* q0' */
- }
- pix++;
- }
- }else{
- /* 2px edge length (see deblocking_filter_edgecv) */
- for( d = 0; d < 2; d++ ) {
- const int p0 = pix[-1*pix_next];
- const int p1 = pix[-2*pix_next];
- const int q0 = pix[0];
- const int q1 = pix[1*pix_next];
- if( ABS( p0 - q0 ) < alpha &&
- ABS( p1 - p0 ) < beta &&
- ABS( q1 - q0 ) < beta ) {
- pix[-pix_next] = ( 2*p1 + p0 + q1 + 2 ) >> 2; /* p0' */
- pix[0] = ( 2*q1 + q0 + p1 + 2 ) >> 2; /* q0' */
- }
- pix++;
- }
- }
- }
- }
- static void filter_mb( H264Context *h, int mb_x, int mb_y, uint8_t *img_y, uint8_t *img_cb, uint8_t *img_cr) {
- MpegEncContext * const s = &h->s;
- const int mb_xy= mb_x + mb_y*s->mb_stride;
- int linesize, uvlinesize;
- int dir;
- /* FIXME Implement deblocking filter for field MB */
- if( h->sps.mb_aff ) {
- return;
- }
- linesize = s->linesize;
- uvlinesize = s->uvlinesize;
- /* dir : 0 -> vertical edge, 1 -> horizontal edge */
- for( dir = 0; dir < 2; dir++ )
- {
- int start = 0;
- int edge;
- /* test picture boundary */
- if( ( dir == 0 && mb_x == 0 ) || ( dir == 1 && mb_y == 0 ) ) {
- start = 1;
- }
- /* FIXME test slice boundary */
- if( h->deblocking_filter == 2 ) {
- }
- /* Calculate bS */
- for( edge = start; edge < 4; edge++ ) {
- /* mbn_xy: neighbour macroblock (how that works for field ?) */
- int mbn_xy = edge > 0 ? mb_xy : ( dir == 0 ? mb_xy -1 : mb_xy - s->mb_stride );
- int bS[4];
- int qp;
- if( IS_INTRA( s->current_picture.mb_type[mb_xy] ) ||
- IS_INTRA( s->current_picture.mb_type[mbn_xy] ) ) {
- bS[0] = bS[1] = bS[2] = bS[3] = ( edge == 0 ? 4 : 3 );
- } else {
- int i;
- for( i = 0; i < 4; i++ ) {
- int x = dir == 0 ? edge : i;
- int y = dir == 0 ? i : edge;
- int b_idx= 8 + 4 + x + 8*y;
- int bn_idx= b_idx - (dir ? 8:1);
- if( h->non_zero_count_cache[b_idx] != 0 ||
- h->non_zero_count_cache[bn_idx] != 0 ) {
- bS[i] = 2;
- }
- else if( h->slice_type == P_TYPE ) {
- if( h->ref_cache[0][b_idx] != h->ref_cache[0][bn_idx] ||
- ABS( h->mv_cache[0][b_idx][0] - h->mv_cache[0][bn_idx][0] ) >= 4 ||
- ABS( h->mv_cache[0][b_idx][1] - h->mv_cache[0][bn_idx][1] ) >= 4 )
- bS[i] = 1;
- else
- bS[i] = 0;
- }
- else {
- /* FIXME Add support for B frame */
- return;
- }
- }
- if(bS[0]+bS[1]+bS[2]+bS[3] == 0)
- continue;
- }
- /* Filter edge */
- qp = ( s->qscale + s->current_picture.qscale_table[mbn_xy] + 1 ) >> 1;
- if( dir == 0 ) {
- filter_mb_edgev( h, &img_y[4*edge], linesize, bS, qp );
- if( (edge&1) == 0 ) {
- int chroma_qp = ( h->chroma_qp +
- get_chroma_qp( h, s->current_picture.qscale_table[mbn_xy] ) + 1 ) >> 1;
- filter_mb_edgecv( h, &img_cb[2*edge], uvlinesize, bS, chroma_qp );
- filter_mb_edgecv( h, &img_cr[2*edge], uvlinesize, bS, chroma_qp );
- }
- } else {
- filter_mb_edgeh( h, &img_y[4*edge*linesize], linesize, bS, qp );
- if( (edge&1) == 0 ) {
- int chroma_qp = ( h->chroma_qp +
- get_chroma_qp( h, s->current_picture.qscale_table[mbn_xy] ) + 1 ) >> 1;
- filter_mb_edgech( h, &img_cb[2*edge*uvlinesize], uvlinesize, bS, chroma_qp );
- filter_mb_edgech( h, &img_cr[2*edge*uvlinesize], uvlinesize, bS, chroma_qp );
- }
- }
- }
- }
- }
- static int decode_slice(H264Context *h){
- MpegEncContext * const s = &h->s;
- const int part_mask= s->partitioned_frame ? (AC_END|AC_ERROR) : 0x7F;
- s->mb_skip_run= -1;
- if( h->pps.cabac ) {
- int i;
- /* realign */
- align_get_bits( &s->gb );
- /* init cabac */
- ff_init_cabac_states( &h->cabac, ff_h264_lps_range, ff_h264_mps_state, ff_h264_lps_state, 64 );
- ff_init_cabac_decoder( &h->cabac,
- s->gb.buffer + get_bits_count(&s->gb)/8,
- ( s->gb.size_in_bits - get_bits_count(&s->gb) + 7)/8);
- /* calculate pre-state */
- for( i= 0; i < 399; i++ ) {
- int pre;
- if( h->slice_type == I_TYPE )
- pre = clip( ((cabac_context_init_I[i][0] * s->qscale) >>4 ) + cabac_context_init_I[i][1], 1, 126 );
- else
- pre = clip( ((cabac_context_init_PB[h->cabac_init_idc][i][0] * s->qscale) >>4 ) + cabac_context_init_PB[h->cabac_init_idc][i][1], 1, 126 );
- if( pre <= 63 )
- h->cabac_state[i] = 2 * ( 63 - pre ) + 0;
- else
- h->cabac_state[i] = 2 * ( pre - 64 ) + 1;
- }
- for(;;){
- int ret = decode_mb_cabac(h);
- int eos = get_cabac_terminate( &h->cabac ); /* End of Slice flag */
- hl_decode_mb(h);
- /* XXX: useless as decode_mb_cabac it doesn't support that ... */
- if( ret >= 0 && h->sps.mb_aff ) { //FIXME optimal? or let mb_decode decode 16x32 ?
- s->mb_y++;
- ret = decode_mb_cabac(h);
- eos = get_cabac_terminate( &h->cabac );
- hl_decode_mb(h);
- s->mb_y--;
- }
- if( ret < 0 || h->cabac.bytestream > h->cabac.bytestream_end + 1) {
- av_log(h->s.avctx, AV_LOG_ERROR, "error while decoding MB %d %dn", s->mb_x, s->mb_y);
- ff_er_add_slice(s, s->resync_mb_x, s->resync_mb_y, s->mb_x, s->mb_y, (AC_ERROR|DC_ERROR|MV_ERROR)&part_mask);
- return -1;
- }
- if( ++s->mb_x >= s->mb_width ) {
- s->mb_x = 0;
- ff_draw_horiz_band(s, 16*s->mb_y, 16);
- if( ++s->mb_y >= s->mb_height ) {
- tprintf("slice end %d %dn", get_bits_count(&s->gb), s->gb.size_in_bits);
- }
- }
- if( eos || s->mb_y >= s->mb_height ) {
- ff_er_add_slice(s, s->resync_mb_x, s->resync_mb_y, s->mb_x-1, s->mb_y, (AC_END|DC_END|MV_END)&part_mask);
- return 0;
- }
- #if 0
- /* TODO test over-reading in cabac code */
- else if( read too much in h->cabac ) {
- ff_er_add_slice(s, s->resync_mb_x, s->resync_mb_y, s->mb_x, s->mb_y, (AC_ERROR|DC_ERROR|MV_ERROR)&part_mask);
- return -1;
- }
- #endif
- }
- } else {
- for(;;){
- int ret = decode_mb_cavlc(h);
- hl_decode_mb(h);
- if(ret>=0 && h->sps.mb_aff){ //FIXME optimal? or let mb_decode decode 16x32 ?
- s->mb_y++;
- ret = decode_mb_cavlc(h);
- hl_decode_mb(h);
- s->mb_y--;
- }
- if(ret<0){
- av_log(h->s.avctx, AV_LOG_ERROR, "error while decoding MB %d %dn", s->mb_x, s->mb_y);
- ff_er_add_slice(s, s->resync_mb_x, s->resync_mb_y, s->mb_x, s->mb_y, (AC_ERROR|DC_ERROR|MV_ERROR)&part_mask);
- return -1;
- }
- if(++s->mb_x >= s->mb_width){
- s->mb_x=0;
- ff_draw_horiz_band(s, 16*s->mb_y, 16);
- if(++s->mb_y >= s->mb_height){
- tprintf("slice end %d %dn", get_bits_count(&s->gb), s->gb.size_in_bits);
- if(get_bits_count(&s->gb) == s->gb.size_in_bits ) {
- ff_er_add_slice(s, s->resync_mb_x, s->resync_mb_y, s->mb_x-1, s->mb_y, (AC_END|DC_END|MV_END)&part_mask);
- return 0;
- }else{
- ff_er_add_slice(s, s->resync_mb_x, s->resync_mb_y, s->mb_x, s->mb_y, (AC_END|DC_END|MV_END)&part_mask);
- return -1;
- }
- }
- }
- if(get_bits_count(&s->gb) >= s->gb.size_in_bits && s->mb_skip_run<=0){
- if(get_bits_count(&s->gb) == s->gb.size_in_bits ){
- ff_er_add_slice(s, s->resync_mb_x, s->resync_mb_y, s->mb_x-1, s->mb_y, (AC_END|DC_END|MV_END)&part_mask);
- return 0;
- }else{
- ff_er_add_slice(s, s->resync_mb_x, s->resync_mb_y, s->mb_x, s->mb_y, (AC_ERROR|DC_ERROR|MV_ERROR)&part_mask);
- return -1;
- }
- }
- }
- }
- #if 0
- for(;s->mb_y < s->mb_height; s->mb_y++){
- for(;s->mb_x < s->mb_width; s->mb_x++){
- int ret= decode_mb(h);
- hl_decode_mb(h);
- if(ret<0){
- fprintf(stderr, "error while decoding MB %d %dn", s->mb_x, s->mb_y);
- ff_er_add_slice(s, s->resync_mb_x, s->resync_mb_y, s->mb_x, s->mb_y, (AC_ERROR|DC_ERROR|MV_ERROR)&part_mask);
- return -1;
- }
- if(++s->mb_x >= s->mb_width){
- s->mb_x=0;
- if(++s->mb_y >= s->mb_height){
- if(get_bits_count(s->gb) == s->gb.size_in_bits){
- ff_er_add_slice(s, s->resync_mb_x, s->resync_mb_y, s->mb_x-1, s->mb_y, (AC_END|DC_END|MV_END)&part_mask);
- return 0;
- }else{
- ff_er_add_slice(s, s->resync_mb_x, s->resync_mb_y, s->mb_x, s->mb_y, (AC_END|DC_END|MV_END)&part_mask);
- return -1;
- }
- }
- }
- if(get_bits_count(s->?gb) >= s->gb?.size_in_bits){
- if(get_bits_count(s->gb) == s->gb.size_in_bits){
- ff_er_add_slice(s, s->resync_mb_x, s->resync_mb_y, s->mb_x-1, s->mb_y, (AC_END|DC_END|MV_END)&part_mask);
- return 0;
- }else{
- ff_er_add_slice(s, s->resync_mb_x, s->resync_mb_y, s->mb_x, s->mb_y, (AC_ERROR|DC_ERROR|MV_ERROR)&part_mask);
- return -1;
- }
- }
- }
- s->mb_x=0;
- ff_draw_horiz_band(s, 16*s->mb_y, 16);
- }
- #endif
- return -1; //not reached
- }
- static inline int decode_vui_parameters(H264Context *h, SPS *sps){
- MpegEncContext * const s = &h->s;
- int aspect_ratio_info_present_flag, aspect_ratio_idc;
- aspect_ratio_info_present_flag= get_bits1(&s->gb);
- if( aspect_ratio_info_present_flag ) {
- aspect_ratio_idc= get_bits(&s->gb, 8);
- if( aspect_ratio_idc == EXTENDED_SAR ) {
- sps->sar.num= get_bits(&s->gb, 16);
- sps->sar.den= get_bits(&s->gb, 16);
- }else if(aspect_ratio_idc < 16){
- sps->sar= pixel_aspect[aspect_ratio_idc];
- }else{
- av_log(h->s.avctx, AV_LOG_ERROR, "illegal aspect ration");
- return -1;
- }
- }else{
- sps->sar.num=
- sps->sar.den= 0;
- }
- // s->avctx->aspect_ratio= sar_width*s->width / (float)(s->height*sar_height);
- #if 0
- | overscan_info_present_flag |0 |u(1) |
- | if( overscan_info_present_flag ) | | |
- | overscan_appropriate_flag |0 |u(1) |
- | video_signal_type_present_flag |0 |u(1) |
- | if( video_signal_type_present_flag ) { | | |
- | video_format |0 |u(3) |
- | video_full_range_flag |0 |u(1) |
- | colour_description_present_flag |0 |u(1) |
- | if( colour_description_present_flag ) { | | |
- | colour_primaries |0 |u(8) |
- | transfer_characteristics |0 |u(8) |
- | matrix_coefficients |0 |u(8) |
- | } | | |
- | } | | |
- | chroma_location_info_present_flag |0 |u(1) |
- | if ( chroma_location_info_present_flag ) { | | |
- | chroma_sample_location_type_top_field |0 |ue(v) |
- | chroma_sample_location_type_bottom_field |0 |ue(v) |
- | } | | |
- | timing_info_present_flag |0 |u(1) |
- | if( timing_info_present_flag ) { | | |
- | num_units_in_tick |0 |u(32) |
- | time_scale |0 |u(32) |
- | fixed_frame_rate_flag |0 |u(1) |
- | } | | |
- | nal_hrd_parameters_present_flag |0 |u(1) |
- | if( nal_hrd_parameters_present_flag = = 1) | | |
- | hrd_parameters( ) | | |
- | vcl_hrd_parameters_present_flag |0 |u(1) |
- | if( vcl_hrd_parameters_present_flag = = 1) | | |
- | hrd_parameters( ) | | |
- | if( ( nal_hrd_parameters_present_flag = = 1 | || | |
- | | | |
- |( vcl_hrd_parameters_present_flag = = 1 ) ) | | |
- | low_delay_hrd_flag |0 |u(1) |
- | bitstream_restriction_flag |0 |u(1) |
- | if( bitstream_restriction_flag ) { |0 |u(1) |
- | motion_vectors_over_pic_boundaries_flag |0 |u(1) |
- | max_bytes_per_pic_denom |0 |ue(v) |
- | max_bits_per_mb_denom |0 |ue(v) |
- | log2_max_mv_length_horizontal |0 |ue(v) |
- | log2_max_mv_length_vertical |0 |ue(v) |
- | num_reorder_frames |0 |ue(v) |
- | max_dec_frame_buffering |0 |ue(v) |
- | } | | |
- |} | | |
- #endif
- return 0;
- }
- static inline int decode_seq_parameter_set(H264Context *h){
- MpegEncContext * const s = &h->s;
- int profile_idc, level_idc;
- int sps_id, i;
- SPS *sps;
- profile_idc= get_bits(&s->gb, 8);
- get_bits1(&s->gb); //constraint_set0_flag
- get_bits1(&s->gb); //constraint_set1_flag
- get_bits1(&s->gb); //constraint_set2_flag
- get_bits(&s->gb, 5); // reserved
- level_idc= get_bits(&s->gb, 8);
- sps_id= get_ue_golomb(&s->gb);
- sps= &h->sps_buffer[ sps_id ];
- sps->profile_idc= profile_idc;
- sps->level_idc= level_idc;
- sps->log2_max_frame_num= get_ue_golomb(&s->gb) + 4;
- sps->poc_type= get_ue_golomb(&s->gb);
- if(sps->poc_type == 0){ //FIXME #define
- sps->log2_max_poc_lsb= get_ue_golomb(&s->gb) + 4;
- } else if(sps->poc_type == 1){//FIXME #define
- sps->delta_pic_order_always_zero_flag= get_bits1(&s->gb);
- sps->offset_for_non_ref_pic= get_se_golomb(&s->gb);
- sps->offset_for_top_to_bottom_field= get_se_golomb(&s->gb);
- sps->poc_cycle_length= get_ue_golomb(&s->gb);
- for(i=0; i<sps->poc_cycle_length; i++)
- sps->offset_for_ref_frame[i]= get_se_golomb(&s->gb);
- }
- if(sps->poc_type > 2){
- av_log(h->s.avctx, AV_LOG_ERROR, "illegal POC type %dn", sps->poc_type);
- return -1;
- }
- sps->ref_frame_count= get_ue_golomb(&s->gb);
- sps->gaps_in_frame_num_allowed_flag= get_bits1(&s->gb);
- sps->mb_width= get_ue_golomb(&s->gb) + 1;
- sps->mb_height= get_ue_golomb(&s->gb) + 1;
- sps->frame_mbs_only_flag= get_bits1(&s->gb);
- if(!sps->frame_mbs_only_flag)
- sps->mb_aff= get_bits1(&s->gb);
- else
- sps->mb_aff= 0;
- sps->direct_8x8_inference_flag= get_bits1(&s->gb);
- sps->crop= get_bits1(&s->gb);
- if(sps->crop){
- sps->crop_left = get_ue_golomb(&s->gb);
- sps->crop_right = get_ue_golomb(&s->gb);
- sps->crop_top = get_ue_golomb(&s->gb);
- sps->crop_bottom= get_ue_golomb(&s->gb);
- if(sps->crop_left || sps->crop_top){
- av_log(h->s.avctx, AV_LOG_ERROR, "insane cropping not completly supported, this could look slightly wrong ...n");
- }
- }else{
- sps->crop_left =
- sps->crop_right =
- sps->crop_top =
- sps->crop_bottom= 0;
- }
- sps->vui_parameters_present_flag= get_bits1(&s->gb);
- if( sps->vui_parameters_present_flag )
- decode_vui_parameters(h, sps);
- if(s->avctx->debug&FF_DEBUG_PICT_INFO){
- av_log(h->s.avctx, AV_LOG_DEBUG, "sps:%d profile:%d/%d poc:%d ref:%d %dx%d %s %s crop:%d/%d/%d/%d %sn",
- sps_id, sps->profile_idc, sps->level_idc,
- sps->poc_type,
- sps->ref_frame_count,
- sps->mb_width, sps->mb_height,
- sps->frame_mbs_only_flag ? "FRM" : (sps->mb_aff ? "MB-AFF" : "PIC-AFF"),
- sps->direct_8x8_inference_flag ? "8B8" : "",
- sps->crop_left, sps->crop_right,
- sps->crop_top, sps->crop_bottom,
- sps->vui_parameters_present_flag ? "VUI" : ""
- );
- }
- return 0;
- }
- static inline int decode_picture_parameter_set(H264Context *h){
- MpegEncContext * const s = &h->s;
- int pps_id= get_ue_golomb(&s->gb);
- PPS *pps= &h->pps_buffer[pps_id];
- pps->sps_id= get_ue_golomb(&s->gb);
- pps->cabac= get_bits1(&s->gb);
- pps->pic_order_present= get_bits1(&s->gb);
- pps->slice_group_count= get_ue_golomb(&s->gb) + 1;
- if(pps->slice_group_count > 1 ){
- pps->mb_slice_group_map_type= get_ue_golomb(&s->gb);
- av_log(h->s.avctx, AV_LOG_ERROR, "FMO not supportedn");
- switch(pps->mb_slice_group_map_type){
- case 0:
- #if 0
- | for( i = 0; i <= num_slice_groups_minus1; i++ ) | | |
- | run_length[ i ] |1 |ue(v) |
- #endif
- break;
- case 2:
- #if 0
- | for( i = 0; i < num_slice_groups_minus1; i++ ) | | |
- |{ | | |
- | top_left_mb[ i ] |1 |ue(v) |
- | bottom_right_mb[ i ] |1 |ue(v) |
- | } | | |
- #endif
- break;
- case 3:
- case 4:
- case 5:
- #if 0
- | slice_group_change_direction_flag |1 |u(1) |
- | slice_group_change_rate_minus1 |1 |ue(v) |
- #endif
- break;
- case 6:
- #if 0
- | slice_group_id_cnt_minus1 |1 |ue(v) |
- | for( i = 0; i <= slice_group_id_cnt_minus1; i++ | | |
- |) | | |
- | slice_group_id[ i ] |1 |u(v) |
- #endif
- break;
- }
- }
- pps->ref_count[0]= get_ue_golomb(&s->gb) + 1;
- pps->ref_count[1]= get_ue_golomb(&s->gb) + 1;
- if(pps->ref_count[0] > 32 || pps->ref_count[1] > 32){
- av_log(h->s.avctx, AV_LOG_ERROR, "reference overflow (pps)n");
- return -1;
- }
- pps->weighted_pred= get_bits1(&s->gb);
- pps->weighted_bipred_idc= get_bits(&s->gb, 2);
- pps->init_qp= get_se_golomb(&s->gb) + 26;
- pps->init_qs= get_se_golomb(&s->gb) + 26;
- pps->chroma_qp_index_offset= get_se_golomb(&s->gb);
- pps->deblocking_filter_parameters_present= get_bits1(&s->gb);
- pps->constrained_intra_pred= get_bits1(&s->gb);
- pps->redundant_pic_cnt_present = get_bits1(&s->gb);
- if(s->avctx->debug&FF_DEBUG_PICT_INFO){
- av_log(h->s.avctx, AV_LOG_DEBUG, "pps:%d sps:%d %s slice_groups:%d ref:%d/%d %s qp:%d/%d/%d %s %s %sn",
- pps_id, pps->sps_id,
- pps->cabac ? "CABAC" : "CAVLC",
- pps->slice_group_count,
- pps->ref_count[0], pps->ref_count[1],
- pps->weighted_pred ? "weighted" : "",
- pps->init_qp, pps->init_qs, pps->chroma_qp_index_offset,
- pps->deblocking_filter_parameters_present ? "LPAR" : "",
- pps->constrained_intra_pred ? "CONSTR" : "",
- pps->redundant_pic_cnt_present ? "REDU" : ""
- );
- }
- return 0;
- }
- /**
- * finds the end of the current frame in the bitstream.
- * @return the position of the first byte of the next frame, or -1
- */
- static int find_frame_end(ParseContext *pc, const uint8_t *buf, int buf_size){
- int i;
- uint32_t state;
- //printf("first %02X%02X%02X%02Xn", buf[0], buf[1],buf[2],buf[3]);
- // mb_addr= pc->mb_addr - 1;
- state= pc->state;
- //FIXME this will fail with slices
- for(i=0; i<buf_size; i++){
- state= (state<<8) | buf[i];
- if((state&0xFFFFFF1F) == 0x101 || (state&0xFFFFFF1F) == 0x102 || (state&0xFFFFFF1F) == 0x105){
- if(pc->frame_start_found){
- pc->state=-1;
- pc->frame_start_found= 0;
- return i-3;
- }
- pc->frame_start_found= 1;
- }
- }
- pc->state= state;
- return END_NOT_FOUND;
- }
- static int h264_parse(AVCodecParserContext *s,
- AVCodecContext *avctx,
- uint8_t **poutbuf, int *poutbuf_size,
- const uint8_t *buf, int buf_size)
- {
- ParseContext *pc = s->priv_data;
- int next;
- next= find_frame_end(pc, buf, buf_size);
- if (ff_combine_frame(pc, next, (uint8_t **)&buf, &buf_size) < 0) {
- *poutbuf = NULL;
- *poutbuf_size = 0;
- return buf_size;
- }
- *poutbuf = (uint8_t *)buf;
- *poutbuf_size = buf_size;
- return next;
- }
- static int decode_nal_units(H264Context *h, uint8_t *buf, int buf_size){
- MpegEncContext * const s = &h->s;
- AVCodecContext * const avctx= s->avctx;
- int buf_index=0;
- #if 0
- int i;
- for(i=0; i<32; i++){
- printf("%X ", buf[i]);
- }
- #endif
- for(;;){
- int consumed;
- int dst_length;
- int bit_length;
- uint8_t *ptr;
- // start code prefix search
- for(; buf_index + 3 < buf_size; buf_index++){
- // this should allways succeed in the first iteration
- if(buf[buf_index] == 0 && buf[buf_index+1] == 0 && buf[buf_index+2] == 1)
- break;
- }
- if(buf_index+3 >= buf_size) break;
- buf_index+=3;
- ptr= decode_nal(h, buf + buf_index, &dst_length, &consumed, buf_size - buf_index);
- if(ptr[dst_length - 1] == 0) dst_length--;
- bit_length= 8*dst_length - decode_rbsp_trailing(ptr + dst_length - 1);
- if(s->avctx->debug&FF_DEBUG_STARTCODE){
- av_log(h->s.avctx, AV_LOG_DEBUG, "NAL %d at %d length %dn", h->nal_unit_type, buf_index, dst_length);
- }
- buf_index += consumed;
- if( s->hurry_up == 1 && h->nal_ref_idc == 0 )
- continue;
- switch(h->nal_unit_type){
- case NAL_IDR_SLICE:
- idr(h); //FIXME ensure we dont loose some frames if there is reordering
- case NAL_SLICE:
- init_get_bits(&s->gb, ptr, bit_length);
- h->intra_gb_ptr=
- h->inter_gb_ptr= &s->gb;
- s->data_partitioning = 0;
- if(decode_slice_header(h) < 0) return -1;
- if(h->redundant_pic_count==0 && s->hurry_up < 5 )
- decode_slice(h);
- break;
- case NAL_DPA:
- init_get_bits(&s->gb, ptr, bit_length);
- h->intra_gb_ptr=
- h->inter_gb_ptr= NULL;
- s->data_partitioning = 1;
- if(decode_slice_header(h) < 0) return -1;
- break;
- case NAL_DPB:
- init_get_bits(&h->intra_gb, ptr, bit_length);
- h->intra_gb_ptr= &h->intra_gb;
- break;
- case NAL_DPC:
- init_get_bits(&h->inter_gb, ptr, bit_length);
- h->inter_gb_ptr= &h->inter_gb;
- if(h->redundant_pic_count==0 && h->intra_gb_ptr && s->data_partitioning && s->hurry_up < 5 )
- decode_slice(h);
- break;
- case NAL_SEI:
- break;
- case NAL_SPS:
- init_get_bits(&s->gb, ptr, bit_length);
- decode_seq_parameter_set(h);
- if(s->flags& CODEC_FLAG_LOW_DELAY)
- s->low_delay=1;
- avctx->has_b_frames= !s->low_delay;
- break;
- case NAL_PPS:
- init_get_bits(&s->gb, ptr, bit_length);
- decode_picture_parameter_set(h);
- break;
- case NAL_PICTURE_DELIMITER:
- break;
- case NAL_FILTER_DATA:
- break;
- default:
- av_log(avctx, AV_LOG_ERROR, "Unknown NAL code: %dn", h->nal_unit_type);
- }
- //FIXME move after where irt is set
- s->current_picture.pict_type= s->pict_type;
- s->current_picture.key_frame= s->pict_type == I_TYPE;
- }
- if(!s->current_picture_ptr) return buf_index; //no frame
- h->prev_frame_num_offset= h->frame_num_offset;
- h->prev_frame_num= h->frame_num;
- if(s->current_picture_ptr->reference){
- h->prev_poc_msb= h->poc_msb;
- h->prev_poc_lsb= h->poc_lsb;
- }
- if(s->current_picture_ptr->reference)
- execute_ref_pic_marking(h, h->mmco, h->mmco_index);
- else
- assert(h->mmco_index==0);
- ff_er_frame_end(s);
- MPV_frame_end(s);
- return buf_index;
- }
- /**
- * retunrs the number of bytes consumed for building the current frame
- */
- static int get_consumed_bytes(MpegEncContext *s, int pos, int buf_size){
- if(s->flags&CODEC_FLAG_TRUNCATED){
- pos -= s->parse_context.last_index;
- if(pos<0) pos=0; // FIXME remove (uneeded?)
- return pos;
- }else{
- if(pos==0) pos=1; //avoid infinite loops (i doubt thats needed but ...)
- if(pos+10>buf_size) pos=buf_size; // oops ;)
- return pos;
- }
- }
- static int decode_frame(AVCodecContext *avctx,
- void *data, int *data_size,
- uint8_t *buf, int buf_size)
- {
- H264Context *h = avctx->priv_data;
- MpegEncContext *s = &h->s;
- AVFrame *pict = data;
- int buf_index;
- s->flags= avctx->flags;
- s->flags2= avctx->flags2;
- /* no supplementary picture */
- if (buf_size == 0) {
- return 0;
- }
- if(s->flags&CODEC_FLAG_TRUNCATED){
- int next= find_frame_end(&s->parse_context, buf, buf_size);
- if( ff_combine_frame(&s->parse_context, next, &buf, &buf_size) < 0 )
- return buf_size;
- //printf("next:%d buf_size:%d last_index:%dn", next, buf_size, s->parse_context.last_index);
- }
- if(s->avctx->extradata_size && s->picture_number==0){
- if(0 < decode_nal_units(h, s->avctx->extradata, s->avctx->extradata_size) )
- return -1;
- }
- buf_index=decode_nal_units(h, buf, buf_size);
- if(buf_index < 0)
- return -1;
- //FIXME do something with unavailable reference frames
- // if(ret==FRAME_SKIPED) return get_consumed_bytes(s, buf_index, buf_size);
- #if 0
- if(s->pict_type==B_TYPE || s->low_delay){
- *pict= *(AVFrame*)&s->current_picture;
- } else {
- *pict= *(AVFrame*)&s->last_picture;
- }
- #endif
- if(!s->current_picture_ptr){
- av_log(h->s.avctx, AV_LOG_DEBUG, "error, NO framen");
- return -1;
- }
- *pict= *(AVFrame*)&s->current_picture; //FIXME
- // ff_print_debug_info(s, pict); //Del by ty
- assert(pict->data[0]);
- //printf("out %dn", (int)pict->data[0]);
- #if 0 //?
- /* Return the Picture timestamp as the frame number */
- /* we substract 1 because it is added on utils.c */
- avctx->frame_number = s->picture_number - 1;
- #endif
- #if 0
- /* dont output the last pic after seeking */
- if(s->last_picture_ptr || s->low_delay)
- //Note this isnt a issue as a IDR pic should flush teh buffers
- #endif
- *data_size = sizeof(AVFrame);
- return get_consumed_bytes(s, buf_index, buf_size);
- }
- #if 0
- static inline void fill_mb_avail(H264Context *h){
- MpegEncContext * const s = &h->s;
- const int mb_xy= s->mb_x + s->mb_y*s->mb_stride;
- if(s->mb_y){
- h->mb_avail[0]= s->mb_x && h->slice_table[mb_xy - s->mb_stride - 1] == h->slice_num;
- h->mb_avail[1]= h->slice_table[mb_xy - s->mb_stride ] == h->slice_num;
- h->mb_avail[2]= s->mb_x+1 < s->mb_width && h->slice_table[mb_xy - s->mb_stride + 1] == h->slice_num;
- }else{
- h->mb_avail[0]=
- h->mb_avail[1]=
- h->mb_avail[2]= 0;
- }
- h->mb_avail[3]= s->mb_x && h->slice_table[mb_xy - 1] == h->slice_num;
- h->mb_avail[4]= 1; //FIXME move out
- h->mb_avail[5]= 0; //FIXME move out
- }
- #endif
- int FindStartCode (unsigned char *Buf, int zeros_in_startcode)
- {
- int info;
- int i;
- info = 1;
- for (i = 0; i < zeros_in_startcode; i++)
- if(Buf[i] != 0)
- info = 0;
- if(Buf[i] != 1)
- info = 0;
- return info;
- }
- int getNextNal(FILE* inpf, unsigned char* Buf)
- {
- int pos = 0;
- int StartCodeFound = 0;
- int info2 = 0;
- int info3 = 0;
- while(!feof(inpf) && (Buf[pos++]=fgetc(inpf))==0);
- while (!StartCodeFound)
- {
- if (feof (inpf))
- {
- // return -1;
- return pos-1;
- }
- Buf[pos++] = fgetc (inpf);
- info3 = FindStartCode(&Buf[pos-4], 3);
- if(info3 != 1)
- info2 = FindStartCode(&Buf[pos-3], 2);
- StartCodeFound = (info2 == 1 || info3 == 1);
- }
- fseek (inpf, -4, SEEK_CUR);
- return pos - 4;
- }
- #define COUNT 8000
- #define SIZE (COUNT*40)
- int main(){
- FILE * inpf;
- FILE * outf;
- int nWrite;
- int i;
- int nalLen;
- unsigned char* Buf;
- int got_picture, consumed_bytes;
- int cnt=0;
- AVCodec *codec; // Codec
- AVCodecContext *c; // Codec Context
- AVFrame *picture; // Frame
- char outfile[] = "test.pgm";
- inpf = fopen("e:\bitavc\foreman_enc.264", "rb");
- outf = fopen("e:\bitavc\out.yuv", "wb");
- nalLen = 0;
- Buf = (unsigned char*)calloc ( 1000000, sizeof(char));
- avcodec_init();
- avcodec_register_all();
- codec = avcodec_find_decoder(CODEC_ID_H264);
- if (!codec) {
- return 0;
- }
- //allocate codec context
- c = avcodec_alloc_context();
- if(!c){
- return 0;
- }
- //open codec
- if (avcodec_open(c, codec) < 0) {
- return 0;
- }
- //allocate frame buffer
- picture = avcodec_alloc_frame();
- if(!picture){
- return 0;
- }
- while(!feof(inpf))
- {
- nalLen = getNextNal(inpf, Buf);
- //try to decode this frame
- // consumed_bytes= decode_frame(c, picture, &got_picture, Buf, nalLen);
- // if (got_picture > 0)
- // c->frame_number++;
- consumed_bytes= avcodec_decode_video(c, picture, &got_picture, Buf, nalLen);
- //ljz added
- cnt++;
- printf("No:=%4d, length=%4dn",cnt,consumed_bytes);
- if(consumed_bytes > 0)
- {
- for(i=0; i<c->height; i++)
- fwrite(picture->data[0] + i * picture->linesize[0], 1, c->width, outf);
- for(i=0; i<c->height/2; i++)
- fwrite(picture->data[1] + i * picture->linesize[1], 1, c->width/2, outf);
- for(i=0; i<c->height/2; i++)
- fwrite(picture->data[2] + i * picture->linesize[2], 1, c->width/2, outf);
- }
- }
- if(inpf)
- fclose(inpf);
- if(outf)
- fclose(outf);
- if(c) {
- avcodec_close(c);
- av_free(c);
- c = NULL;
- }
- if(picture) {
- av_free(picture);
- picture = NULL;
- }
- if(Buf)
- {
- free(Buf);
- Buf = NULL;
- }
- return 0;
- }
- //#endif
- static int decode_end(AVCodecContext *avctx)
- {
- H264Context *h = avctx->priv_data;
- MpegEncContext *s = &h->s;
- free_tables(h); //FIXME cleanup init stuff perhaps
- MPV_common_end(s);
- // memset(h, 0, sizeof(H264Context));
- return 0;
- }
- AVCodec h264_decoder = {
- "h264",
- CODEC_TYPE_VIDEO,
- CODEC_ID_H264,
- sizeof(H264Context),
- decode_init,
- NULL,
- decode_end,
- decode_frame,
- /*CODEC_CAP_DRAW_HORIZ_BAND |*/ CODEC_CAP_DR1 | CODEC_CAP_TRUNCATED,
- };
- AVCodecParser h264_parser = {
- { CODEC_ID_H264 },
- sizeof(ParseContext),
- NULL,
- h264_parse,
- ff_parse_close,
- };
- //#include "svq3.c"