📄 image.c
字号:
/*!
***********************************************************************
* \file image.c
*
* \brief
* Decode a Slice
*
* \author
* Main contributors (see contributors.h for copyright, address and affiliation details)
* - Inge Lille-Langoy <inge.lille-langoy@telenor.com>
* - Rickard Sjoberg <rickard.sjoberg@era.ericsson.se>
* - Jani Lainema <jani.lainema@nokia.com>
* - Sebastian Purreiter <sebastian.purreiter@mch.siemens.de>
* - Byeong-Moon Jeon <jeonbm@lge.com>
* - Thomas Wedi <wedi@tnt.uni-hannover.de>
* - Gabi Blaettermann
* - Ye-Kui Wang <wyk@ieee.org>
* - Antti Hallapuro <antti.hallapuro@nokia.com>
* - Alexis Tourapis <alexismt@ieee.org>
* - Jill Boyce <jill.boyce@thomson.net>
* - Saurav K Bandyopadhyay <saurav@ieee.org>
* - Zhenyu Wu <Zhenyu.Wu@thomson.net
* - Purvin Pandit <Purvin.Pandit@thomson.net>
*
***********************************************************************
*/
#include "contributors.h"
#include <math.h>
#include <limits.h>
#include "global.h"
#include "image.h"
#include "fmo.h"
#include "annexb.h"
#include "nalu.h"
#include "parset.h"
#include "header.h"
#include "sei.h"
#include "output.h"
#include "mb_access.h"
#include "memalloc.h"
#include "macroblock.h"
#include "loopfilter.h"
#include "biaridecod.h"
#include "context_ini.h"
#include "cabac.h"
#include "vlc.h"
#include "quant.h"
#include "errorconcealment.h"
#include "erc_api.h"
extern objectBuffer_t *erc_object_list;
extern ercVariables_t *erc_errorVar;
extern frame erc_recfr;
extern int erc_mvperMB;
extern ImageParameters *erc_img;
extern StorablePicture **listX[6];
extern ColocatedParams *Co_located;
extern StorablePicture *no_reference_picture;
int non_conforming_stream;
StorablePicture *dec_picture;
StorablePicture *dec_picture_JV[MAX_PLANE]; //!< dec_picture to be used during 4:4:4 independent mode decoding
OldSliceParams old_slice;
/*!
************************************************************************
* \brief
* Initializes the parameters for a new picture
************************************************************************
*/
static void init_picture(ImageParameters *img, Slice *currSlice, struct inp_par *inp)
{
int i;
int nplane;
if (dec_picture)
{
// this may only happen on slice loss
exit_picture(&dec_picture);
}
if (img->recovery_point)
img->recovery_frame_num = (img->frame_num + img->recovery_frame_cnt) % img->MaxFrameNum;
if (img->idr_flag)
img->recovery_frame_num = img->frame_num;
if (img->recovery_point == 0 &&
img->frame_num != img->pre_frame_num &&
img->frame_num != (img->pre_frame_num + 1) % img->MaxFrameNum)
{
if (active_sps->gaps_in_frame_num_value_allowed_flag == 0)
{
// picture error concealment
if(inp->conceal_mode !=0)
{
if((img->frame_num) < ((img->pre_frame_num + 1) % img->MaxFrameNum))
{
/* Conceal lost IDR frames and any frames immediately
following the IDR. Use frame copy for these since
lists cannot be formed correctly for motion copy*/
img->conceal_mode = 1;
img->IDR_concealment_flag = 1;
conceal_lost_frames(img);
//reset to original concealment mode for future drops
img->conceal_mode = inp->conceal_mode;
}
else
{
//reset to original concealment mode for future drops
img->conceal_mode = inp->conceal_mode;
img->IDR_concealment_flag = 0;
conceal_lost_frames(img);
}
}
else
{ /* Advanced Error Concealment would be called here to combat unintentional loss of pictures. */
error("An unintentional loss of pictures occurs! Exit\n", 100);
}
}
if(img->conceal_mode == 0)
fill_frame_num_gap(img);
}
if(img->nal_reference_idc)
{
img->pre_frame_num = img->frame_num;
}
img->num_dec_mb = 0;
//calculate POC
decode_poc(img);
if (img->recovery_frame_num == img->frame_num &&
img->recovery_poc == 0x7fffffff)
img->recovery_poc = img->framepoc;
if(img->nal_reference_idc)
img->last_ref_pic_poc = img->framepoc;
// dumppoc (img);
if (img->structure==FRAME ||img->structure==TOP_FIELD)
{
gettime (&(img->start_time)); // start time
}
dec_picture = alloc_storable_picture ((PictureStructure) img->structure, img->width, img->height, img->width_cr, img->height_cr);
dec_picture->top_poc=img->toppoc;
dec_picture->bottom_poc=img->bottompoc;
dec_picture->frame_poc=img->framepoc;
dec_picture->qp = img->qp;
dec_picture->slice_qp_delta = currSlice->slice_qp_delta;
dec_picture->chroma_qp_offset[0] = active_pps->chroma_qp_index_offset;
dec_picture->chroma_qp_offset[1] = active_pps->second_chroma_qp_index_offset;
// reset all variables of the error concealment instance before decoding of every frame.
// here the third parameter should, if perfectly, be equal to the number of slices per frame.
// using little value is ok, the code will allocate more memory if the slice number is larger
ercReset(erc_errorVar, img->PicSizeInMbs, img->PicSizeInMbs, dec_picture->size_x);
erc_mvperMB = 0;
switch (img->structure )
{
case TOP_FIELD:
{
dec_picture->poc=img->toppoc;
img->number *= 2;
break;
}
case BOTTOM_FIELD:
{
dec_picture->poc=img->bottompoc;
img->number = img->number * 2 + 1;
break;
}
case FRAME:
{
dec_picture->poc=img->framepoc;
break;
}
default:
error("img->structure not initialized", 235);
}
img->current_slice_nr=0;
if (img->type > SI_SLICE)
{
set_ec_flag(SE_PTYPE);
img->type = P_SLICE; // concealed element
}
// CAVLC init
if (active_pps->entropy_coding_mode_flag == UVLC)
{
memset(&img->nz_coeff[0][0][0][0], -1, img->PicSizeInMbs * 3 * 4 * 4 *sizeof(byte));
}
if(active_pps->constrained_intra_pred_flag)
{
for (i=0; i<(int)img->PicSizeInMbs; i++)
{
img->intra_block[i] = 1;
}
}
// Set the slice_nr member of each MB to -1, to ensure correct when packet loss occurs
// TO set Macroblock Map (mark all MBs as 'have to be concealed')
if( IS_INDEPENDENT(img) )
{
for( nplane=0; nplane<MAX_PLANE; nplane++ )
{
for(i=0; i<(int)img->PicSizeInMbs; i++)
{
img->mb_data_JV[nplane][i].slice_nr = -1;
img->mb_data_JV[nplane][i].ei_flag = 1;
img->mb_data_JV[nplane][i].dpl_flag = 0;
}
}
}
else
{
for(i=0; i<(int)img->PicSizeInMbs; i++)
{
img->mb_data[i].slice_nr = -1;
img->mb_data[i].ei_flag = 1;
img->mb_data[i].dpl_flag = 0;
}
}
img->mb_y = img->mb_x = 0;
img->block_y_aff = img->block_y = img->pix_y = img->pix_c_y = 0; // define vertical positions
img->block_x = img->pix_x = img->pix_c_x = 0; // define horizontal positions
dec_picture->slice_type = img->type;
dec_picture->used_for_reference = (img->nal_reference_idc != 0);
dec_picture->idr_flag = img->idr_flag;
dec_picture->no_output_of_prior_pics_flag = img->no_output_of_prior_pics_flag;
dec_picture->long_term_reference_flag = img->long_term_reference_flag;
dec_picture->adaptive_ref_pic_buffering_flag = img->adaptive_ref_pic_buffering_flag;
dec_picture->dec_ref_pic_marking_buffer = img->dec_ref_pic_marking_buffer;
img->dec_ref_pic_marking_buffer = NULL;
dec_picture->MbaffFrameFlag = img->MbaffFrameFlag;
dec_picture->PicWidthInMbs = img->PicWidthInMbs;
get_mb_block_pos = dec_picture->MbaffFrameFlag ? get_mb_block_pos_mbaff : get_mb_block_pos_normal;
getNeighbour = dec_picture->MbaffFrameFlag ? getAffNeighbour : getNonAffNeighbour;
dec_picture->pic_num = img->frame_num;
dec_picture->frame_num = img->frame_num;
dec_picture->recovery_frame = (img->frame_num == img->recovery_frame_num);
dec_picture->coded_frame = (img->structure==FRAME);
dec_picture->chroma_format_idc = active_sps->chroma_format_idc;
dec_picture->frame_mbs_only_flag = active_sps->frame_mbs_only_flag;
dec_picture->frame_cropping_flag = active_sps->frame_cropping_flag;
if (dec_picture->frame_cropping_flag)
{
dec_picture->frame_cropping_rect_left_offset = active_sps->frame_cropping_rect_left_offset;
dec_picture->frame_cropping_rect_right_offset = active_sps->frame_cropping_rect_right_offset;
dec_picture->frame_cropping_rect_top_offset = active_sps->frame_cropping_rect_top_offset;
dec_picture->frame_cropping_rect_bottom_offset = active_sps->frame_cropping_rect_bottom_offset;
}
#if (ENABLE_OUTPUT_TONEMAPPING)
// store the necessary tone mapping sei into StorablePicture structure
dec_picture->seiHasTone_mapping = 0;
if (seiToneMapping.seiHasTone_mapping)
{
dec_picture->seiHasTone_mapping = 1;
dec_picture->tone_mapping_model_id = seiToneMapping.model_id;
dec_picture->tonemapped_bit_depth = seiToneMapping.sei_bit_depth;
dec_picture->tone_mapping_lut = malloc(sizeof(int)*(1<<seiToneMapping.coded_data_bit_depth));
if (NULL == dec_picture->tone_mapping_lut)
{
no_mem_exit("init_picture: tone_mapping_lut");
}
memcpy(dec_picture->tone_mapping_lut, seiToneMapping.lut, sizeof(imgpel)*(1<<seiToneMapping.coded_data_bit_depth));
update_tone_mapping_sei();
}
#endif
if( IS_INDEPENDENT(img) )
{
dec_picture_JV[0] = dec_picture;
dec_picture_JV[1] = alloc_storable_picture ((PictureStructure) img->structure, img->width, img->height, img->width_cr, img->height_cr);
copy_dec_picture_JV( dec_picture_JV[1], dec_picture_JV[0] );
dec_picture_JV[2] = alloc_storable_picture ((PictureStructure) img->structure, img->width, img->height, img->width_cr, img->height_cr);
copy_dec_picture_JV( dec_picture_JV[2], dec_picture_JV[0] );
}
}
void MbAffPostProc(void)
{
imgpel temp[32][16];
imgpel ** imgY = dec_picture->imgY;
imgpel ***imgUV = dec_picture->imgUV;
int i, y, x0, y0, uv;
for (i=0; i<(int)dec_picture->PicSizeInMbs; i+=2)
{
if (dec_picture->motion.mb_field[i])
{
get_mb_pos(i, img->mb_size[IS_LUMA], &x0, &y0);
for (y=0; y<(2*MB_BLOCK_SIZE);y++)
memcpy(temp[y], &imgY[y0+y][x0], MB_BLOCK_SIZE * sizeof(imgpel));
for (y=0; y<MB_BLOCK_SIZE;y++)
{
memcpy(&imgY[y0+(2*y )][x0], temp[y ], MB_BLOCK_SIZE * sizeof(imgpel));
memcpy(&imgY[y0+(2*y+1)][x0], temp[y+MB_BLOCK_SIZE], MB_BLOCK_SIZE * sizeof(imgpel));
}
if (dec_picture->chroma_format_idc != YUV400)
{
x0 = x0 / (16/img->mb_cr_size_x);
y0 = y0 / (16/img->mb_cr_size_y);
for (uv=0; uv<2; uv++)
{
for (y=0; y<(2*img->mb_cr_size_y);y++)
memcpy(temp[y], &imgUV[uv][y0+y][x0], img->mb_cr_size_x * sizeof(imgpel));
for (y=0; y<img->mb_cr_size_y;y++)
{
memcpy(&imgUV[uv][y0+(2*y )][x0], temp[y ], img->mb_cr_size_x * sizeof(imgpel));
memcpy(&imgUV[uv][y0+(2*y+1)][x0], temp[y+img->mb_cr_size_y], img->mb_cr_size_x * sizeof(imgpel));
}
}
}
}
}
}
static void fill_wp_params(ImageParameters *img)
{
int i, j, k;
int comp;
int log_weight_denom;
int tb, td;
int bframe = (img->type==B_SLICE);
int tx,DistScaleFactor;
int max_l0_ref = img->num_ref_idx_l0_active;
int max_l1_ref = img->num_ref_idx_l1_active;
if (active_pps->weighted_bipred_idc == 2 && bframe)
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -