📄 slice.c
字号:
/*!
**************************************************************************************
* \file
* slice.c
* \brief
* generate the slice header, setup the bit buffer for slices,
* and generates the slice NALU(s)
* \author
* Main contributors (see contributors.h for copyright, address and affiliation details)
* - Thomas Stockhammer <stockhammer@ei.tum.de>
* - Detlev Marpe <marpe@hhi.de>
* - Stephan Wenger <stewe@cs.tu-berlin.de>
* - Alexis Michael Tourapis <alexismt@ieee.org>
***************************************************************************************
*/
#include "contributors.h"
#include <stdlib.h>
#include <assert.h>
#include "global.h"
#include "header.h"
#include "rtp.h"
#include "fmo.h"
#include "vlc.h"
#include "image.h"
#include "cabac.h"
#include "elements.h"
// Local declarations
static Slice *malloc_slice();
static void free_slice(Slice *slice);
static void init_slice(int start_mb_addr);
static void set_ref_pic_num();
extern ColocatedParams *Co_located;
extern StorablePicture **listX[6];
void poc_ref_pic_reorder(StorablePicture **list, unsigned num_ref_idx_lX_active, int *remapping_of_pic_nums_idc, int *abs_diff_pic_num_minus1, int *long_term_pic_idx, int weighted_prediction, int list_no);
/*!
************************************************************************
* \brief
* init_ref_pic_list_reordering initializations should go here
************************************************************************
*/
void init_ref_pic_list_reordering()
{
Slice* currSlice = img->currentSlice;
currSlice->ref_pic_list_reordering_flag_l0 = 0;
currSlice->ref_pic_list_reordering_flag_l1 = 0;
}
/*!
************************************************************************
* \brief
* This function generates the slice (and partition) header(s)
*
* \return number of bits used for the slice (and partition) header(s)
*
* \par Side effects:
* Adds slice/partition header symbols to the symbol buffer
* increments Picture->no_slices, allocates memory for the
* slice, sets img->currSlice
************************************************************************
*/
int start_slice()
{
EncodingEnvironmentPtr eep;
Slice *currSlice = img->currentSlice;
Bitstream *currStream;
int header_len = 0;
int i;
int NumberOfPartitions = (input->partition_mode == PAR_DP_1?1:3);
//one partition for IDR img
if(img->currentPicture->idr_flag)
{
NumberOfPartitions = 1;
}
RTPUpdateTimestamp (img->tr); // this has no side effects, just leave it for all NALs
for (i=0; i<NumberOfPartitions; i++)
{
currStream = (currSlice->partArr[i]).bitstream;
currStream->write_flag = 0;
if (i==0) // First partition
header_len += SliceHeader (0);
else // Second/Third partition
header_len += Partition_BC_Header(i);
//! Initialize CABAC
if (input->symbol_mode == CABAC)
{
eep = &((currSlice->partArr[i]).ee_cabac);
if (currStream->bits_to_go != 8)
header_len+=currStream->bits_to_go;
writeVlcByteAlign(currStream);
arienco_start_encoding(eep, currStream->streamBuffer, &(currStream->byte_pos)/*, &(currStream->last_startcode)*/,img->type);
cabac_new_slice();
} else
{
// Initialize CA-VLC
CAVLC_init();
}
}
if(input->symbol_mode == CABAC)
{
init_contexts();
}
return header_len;
}
/*!
************************************************************************
* \brief
* This function terminates a slice (but doesn't write it out),
* the old terminate_slice (0)
* \return
* 0 if OK, \n
* 1 in case of error
*
************************************************************************
*/
int terminate_slice()
{
int bytes_written;
Bitstream *currStream;
Slice *currSlice = img->currentSlice;
EncodingEnvironmentPtr eep;
int i;
int byte_pos_before_startcode_emu_prevention;
if (input->symbol_mode == CABAC)
write_terminating_bit (1); // only once, not for all partitions
for (i=0; i<currSlice->max_part_nr; i++)
{
currStream = (currSlice->partArr[i]).bitstream;
if (input->symbol_mode == UVLC)
{
SODBtoRBSP(currStream);
byte_pos_before_startcode_emu_prevention = currStream->byte_pos;
currStream->byte_pos = RBSPtoEBSP(currStream->streamBuffer, 0 , currStream->byte_pos, 0);
*(stats->em_prev_bits) += (currStream->byte_pos - byte_pos_before_startcode_emu_prevention) * 8;
}
else // CABAC
{
eep = &((currSlice->partArr[i]).ee_cabac);
// terminate the arithmetic code
arienco_done_encoding(eep);
currStream->bits_to_go = eep->Ebits_to_go;
currStream->byte_buf = 0;
bytes_written = currStream->byte_pos;
byte_pos_before_startcode_emu_prevention= currStream->byte_pos;
currStream->byte_pos = RBSPtoEBSP(currStream->streamBuffer, 0, currStream->byte_pos, eep->E);
*(stats->em_prev_bits) += (currStream->byte_pos - byte_pos_before_startcode_emu_prevention) * 8;
} // CABAC
} // partition loop
if( input->symbol_mode == CABAC )
{
store_contexts();
}
return 0;
}
/*!
************************************************************************
* \brief
* Encodes one slice
* \par
* returns the number of coded MBs in the SLice
************************************************************************
*/
int encode_one_slice (int SliceGroupId, Picture *pic)
{
Boolean end_of_slice = FALSE;
Boolean recode_macroblock;
int len;
int NumberOfCodedMBs = 0;
int CurrentMbAddr;
double FrameRDCost, FieldRDCost;
img->cod_counter = 0;
CurrentMbAddr = FmoGetFirstMacroblockInSlice (SliceGroupId);
// printf ("\n\nEncode_one_slice: PictureID %d SliceGroupId %d SliceID %d FirstMB %d \n", img->tr, SliceGroupId, img->current_slice_nr, CurrentMbInScanOrder);
init_slice (CurrentMbAddr);
Bytes_After_Header = img->currentSlice->partArr[0].bitstream->byte_pos;
if (input->symbol_mode==CABAC)
{
SetCtxModelNumber ();
}
/*
// Tian Dong: June 7, 2002 JVT-B042
// When the pictures are put into different layers and subseq, not all the reference frames
// in multi-frame buffer are valid for prediction. The acutual number of the valid reference
// frames, fb->num_short_used, will be given by start_slice(sym).
// Save the fb->short_used.
if (input->NumFramesInELSubSeq)
{
short_used = fb->short_used;
img_ref = img->nb_references;
}
*/
len = start_slice ();
// Rate control
img->NumberofHeaderBits +=len;
// basic unit layer rate control
if(img->BasicUnit<img->Frame_Total_Number_MB)
img->NumberofBasicUnitHeaderBits +=len;
// printf("short size, used, num-used: (%d,%d,%d)\n", fb->short_size, fb->short_used, fb->num_short_used);
/*
// Tian Dong: June 7, 2002 JVT-B042
if (input->NumFramesInELSubSeq)
{
fb->short_used = fb->num_short_used;
img->nb_references = fb->short_used + fb->long_used;
}
*/
// Update statistics
stats->bit_slice += len;
stats->bit_use_header[img->type] += len;
// printf ("\n\n");
while (end_of_slice == FALSE) // loop over macroblocks
{
//sw paff
if (!img->MbaffFrameFlag)
{
recode_macroblock = FALSE;
rdopt = &rddata_top_frame_mb; // store data in top frame MB
start_macroblock (CurrentMbAddr, FALSE);
encode_one_macroblock ();
write_one_macroblock (1);
terminate_macroblock (&end_of_slice, &recode_macroblock);
// printf ("encode_one_slice: mb %d, slice %d, bitbuf bytepos %d EOS %d\n",
// img->current_mb_nr, img->current_slice_nr,
// img->currentSlice->partArr[0].bitstream->byte_pos, end_of_slice);
if (recode_macroblock == FALSE) // The final processing of the macroblock has been done
{
CurrentMbAddr = FmoGetNextMBNr (CurrentMbAddr);
if (CurrentMbAddr == -1) // end of slice
{
// printf ("FMO End of Slice Group detected, current MBs %d, force end of slice\n", NumberOfCodedMBs+1);
end_of_slice = TRUE;
}
NumberOfCodedMBs++; // only here we are sure that the coded MB is actually included in the slice
proceed2nextMacroblock (CurrentMbAddr);
}
else
{
//!Go back to the previous MB to recode it
img->current_mb_nr = FmoGetPreviousMBNr(img->current_mb_nr);
if(img->current_mb_nr == -1 ) // The first MB of the slice group is too big,
// which means it's impossible to encode picture using current slice bits restriction
{
snprintf (errortext, ET_SIZE, "Error encoding first MB with spcified parameter, bits of current MB may be too big");
error (errortext, 300);
}
}
}
else // TBD -- Addition of FMO
{
//! This following ugly code breaks slices, at least for a slice mode that accumulates a certain
//! number of bits into one slice.
//! The suggested algorithm is as follows:
//!
//! SaveState (Bitstream, stats, etc. etc.);
//! BitsForThisMBPairInFrameMode = CodeMB (Upper, FRAME_MODE) + CodeMB (Lower, FRAME_MODE);
//! DistortionForThisMBPairInFrameMode = CalculateDistortion(Upper) + CalculateDistortion (Lower);
//! RestoreState();
//! BitsForThisMBPairInFieldMode = CodeMB (Upper, FIELD_MODE) + CodeMB (Lower, FIELD_MODE);
//! DistortionForThisMBPairInFrameMode = CalculateDistortion(Upper) + CalculateDistortion (Lower);
//! FrameFieldMode = Decision (...)
//! RestoreState()
//! if (FrameFieldMode == FRAME) {
//! CodeMB (Upper, FRAME); CodeMB (Lower, FRAME);
//! } else {
//! CodeMB (Upper FIELD); CodeMB (Lower, FIELD);
//! }
//!
//! Open questions/issues:
//! 1. CABAC/CA-VLC state: It seems that the CABAC/CA_VLC states are changed during the
//! dummy encoding processes (for the R-D based selection), but that they are never
//! reset, once the selection is made. I believe that this breaks the MB-adaptive
//! frame/field coding. The necessary code for the state saves is readily available
//! in macroblock.c, start_macroblock() and terminate_macroblock() (this code needs
//! to be double checked that it works with CA-VLC as well
//! 2. would it be an option to allocate Bitstreams with zero data in them (or copy the
//! already generated bitstream) for the "test coding"?
if (input->MbInterlace == ADAPTIVE_CODING)
{
//================ code MB pair as frame MB ================
//----------------------------------------------------------
recode_macroblock = FALSE;
img->field_mode = 0; // MB coded as frame
img->top_field = 0; // Set top field to 0
//Rate control
img->write_macroblock = 0;
img->bot_MB = 0;
start_macroblock (CurrentMbAddr, FALSE);
rdopt = &rddata_top_frame_mb; // store data in top frame MB
encode_one_macroblock (); // code the MB as frame
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -