📄 h.264
字号:
/*!
***********************************************************************
* \mainpage
* This is the H.264/AVC encoder reference software. For detailed documentation
* see the comments in each file.
*
* \author
* The main contributors are listed in contributors.h
*
* \version
* JM 7.3
*
* \note
* tags are used for document system "doxygen"
* available at http://www.doxygen.org
*/
/*!
* \file
* lencod.c
* \brief
* H.264/AVC reference encoder project main()
* \author
* Main contributors (see contributors.h for copyright, address and affiliation details)
* - Inge Lille-Lang鴜 <inge.lille-langoy@telenor.com>
* - Rickard Sjoberg <rickard.sjoberg@era.ericsson.se>
* - Stephan Wenger <stewe@cs.tu-berlin.de>
* - Jani Lainema <jani.lainema@nokia.com>
* - Byeong-Moon Jeon <jeonbm@lge.com>
* - Yoon-Seong Soh <yunsung@lge.com>
* - Thomas Stockhammer <stockhammer@ei.tum.de>
* - Detlev Marpe <marpe@hhi.de>
* - Guido Heising <heising@hhi.de>
* - Karsten Suehring <suehring@hhi.de>
*
***********************************************************************
*/
#include "contributors.h"
#include <string.h>
#include <math.h>
#include <time.h>
#include <sys/timeb.h>
#include <stdlib.h>
#if defined WIN32
#include <conio.h>
#endif
#include <assert.h>
#include "global.h"
#include "configfile.h"
#include "leaky_bucket.h"
#include "memalloc.h"
#include "mbuffer.h"
#include "intrarefresh.h"
#include "fmo.h"
#include "encodeiff.h"
#include "sei.h"
#include "parset.h"
#include "image.h"
#include "output.h"
#define JM "7"
#define VERSION "7.3"
InputParameters inputs, *input = &inputs;
ImageParameters images, *img = &images;
StatParameters stats, *stat = &stats;
SNRParameters snrs, *snr = &snrs;
Decoders decoders, *decs=&decoders;
#ifdef _ADAPT_LAST_GROUP_
int initial_Bframes = 0;
#endif
Boolean In2ndIGOP = FALSE;
int start_frame_no_in_this_IGOP = 0;
int start_tr_in_this_IGOP = 0;
int FirstFrameIn2ndIGOP=0;
int cabac_encoding = 0;
/*!
***********************************************************************
* \brief
* Main function for encoder.
* \param argc
* number of command line arguments
* \param argv
* command line arguments
* \return
* exit code
***********************************************************************
*/
void Init_Motion_Search_Module ();
void Clear_Motion_Search_Module ();
int main(int argc,char **argv)
{
p_dec = p_stat = p_log = p_trace = NULL;
Configure (argc, argv);
init_img();
AllocNalPayloadBuffer();
GenerateParameterSets();
frame_pic = malloc_picture();
if (input->InterlaceCodingOption != FRAME_CODING)
{
top_pic = malloc_picture();
bottom_pic = malloc_picture();
}
init_rdopt ();
init_dpb(input);
init_out_buffer();
enc_picture = enc_frame_picture = enc_top_picture = enc_bottom_picture = NULL;
init_global_buffers();
create_context_memory ();
Init_Motion_Search_Module ();
information_init();
// B pictures
Bframe_ctr=0;
tot_time=0; // time for total encoding session
#ifdef _ADAPT_LAST_GROUP_
if (input->last_frame > 0)
input->no_frames = 1 + (input->last_frame + input->jumpd) / (input->jumpd + 1);
initial_Bframes = input->successive_Bframe;
#endif
PatchInputNoFrames();
init_poc();
// Write sequence header (with parameter sets)
stat->bit_slice = start_sequence();
start_frame_no_in_this_IGOP = 0;
for (img->number=0; img->number < input->no_frames; img->number++)
{
img->nal_reference_idc = 1;
//much of this can go in init_frame() or init_field()?
//poc for this frame or field
img->toppoc = IMG_NUMBER*img->offset_for_ref_frame[0];
if (input->InterlaceCodingOption==FRAME_CODING)
img->bottompoc = img->toppoc; //progressive
else
img->bottompoc = img->toppoc+1; //hard coded
push_poc(img->toppoc, img->bottompoc, REFFRAME); //push poc values into array
img->framepoc = min (img->toppoc, img->bottompoc);
//frame_num for this frame
img->frame_num = IMG_NUMBER % (1 << (LOG2_MAX_FRAME_NUM_MINUS4 + 4));
//the following is sent in the slice header
img->delta_pic_order_cnt[0]=0;
SetImgType();
#ifdef _ADAPT_LAST_GROUP_
if (input->successive_Bframe && input->last_frame && IMG_NUMBER+1 == input->no_frames)
{
int bi = (int)((float)(input->jumpd+1)/(input->successive_Bframe+1.0)+0.499999);
input->successive_Bframe = (input->last_frame-(img->number-1)*(input->jumpd+1))/bi-1;
//about to code the last ref frame, adjust deltapoc
img->delta_pic_order_cnt[0]= -2*(initial_Bframes - input->successive_Bframe);
img->toppoc += img->delta_pic_order_cnt[0];
img->bottompoc += img->delta_pic_order_cnt[0];
toprefpoc[0] = img->toppoc; //fix array values too
bottomrefpoc[0] = img->bottompoc;
}
#endif
img->framepoc = min (img->toppoc, img->bottompoc);
// which layer the image belonged to?
if ( IMG_NUMBER % (input->NumFramesInELSubSeq+1) == 0 )
img->layer = 0;
else
img->layer = 1;
encode_one_frame(); // encode one I- or P-frame
img->nb_references += 1;
img->nb_references = min(img->nb_references, img->buf_cycle); // Tian Dong. PLUS1, +1, June 7, 2002
if ((input->successive_Bframe != 0) && (IMG_NUMBER > 0)) // B-frame(s) to encode
{
img->type = B_SLICE; // set image type to B-frame
img->num_ref_idx_l1_active = 1;
if (input->NumFramesInELSubSeq == 0)
img->layer = 0;
else
img->layer = 1;
img->frame_num++; //increment frame_num once for B-frames
img->frame_num %= (1 << (LOG2_MAX_FRAME_NUM_MINUS4 + 4));
img->nal_reference_idc = (input->StoredBPictures != 0);
for(img->b_frame_to_code=1; img->b_frame_to_code<=input->successive_Bframe; img->b_frame_to_code++)
{
//! somewhere here the disposable flag was set -- B frames are always disposable in this encoder.
//! This happens now in slice.c, terminate_slice, where the nal_reference_idc is set up
//poc for this B frame
img->toppoc = (IMG_NUMBER-1)*img->offset_for_ref_frame[0] + 2* img->b_frame_to_code;
if (input->InterlaceCodingOption==FRAME_CODING)
img->bottompoc = img->toppoc; //progressive
else
img->bottompoc = img->toppoc+1;
push_poc(img->toppoc,img->bottompoc, NONREFFRAME); //push pocs for B frame
img->framepoc = min (img->toppoc, img->bottompoc);
//the following is sent in the slice header
img->delta_pic_order_cnt[0]= 2*(img->b_frame_to_code-1);
img->delta_pic_order_cnt[1]= 0; // POC200301
encode_one_frame(); // encode one B-frame
}
}
process_2nd_IGOP();
}
// terminate sequence
terminate_sequence();
flush_dpb();
fclose(p_in);
if (p_dec)
fclose(p_dec);
if (p_trace)
fclose(p_trace);
//Clear_Motion_Search_Module ();
RandomIntraUninit();
FmoUninit();
// free structure for rd-opt. mode decision
clear_rdopt ();
#ifdef _LEAKYBUCKET_
calc_buffer();
#endif
// report everything
report();
free_picture (frame_pic);
if (top_pic)
free_picture (top_pic);
if (bottom_pic)
free_picture (bottom_pic);
free_dpb();
uninit_out_buffer();
free_global_buffers();
// free image mem
free_img ();
free_context_memory ();
FreeNalPayloadBuffer();
FreeParameterSets();
return 0;
}
/*!
***********************************************************************
* \brief
* Initializes the POC structure with appropriate parameters.
*
***********************************************************************
*/
void init_poc()
{
int i;
if(input->no_frames > (1<<15))error("too many frames",-998);
if(input->num_reference_frames >= MAX_NO_POC_FRAMES)error("NumberReferenceFrames too large",-999);
for(i=0; i<MAX_NO_POC_FRAMES; i++){toprefpoc[i] = bottomrefpoc[i] = 1<<29;} //init with large
//the following should probably go in sequence parameters
// frame poc's increase by 2, field poc's by 1
img->pic_order_cnt_type=input->pic_order_cnt_type; // POC200301
img->num_ref_frames_in_pic_order_cnt_cycle=1;
img->delta_pic_order_always_zero_flag=0;
img->offset_for_non_ref_pic = -2*(input->successive_Bframe);
if (input->InterlaceCodingOption==FRAME_CODING)
img->offset_for_top_to_bottom_field=0;
else
img->offset_for_top_to_bottom_field=1;
img->offset_for_ref_frame[0] = 2*(input->successive_Bframe+1);
//the following should probably go in picture parameters
// img->pic_order_present_flag=0; //img->delta_pic_order_cnt[1] not sent
// POC200301
if (input->InterlaceCodingOption==FRAME_CODING)
{
img->pic_order_present_flag=0;
img->delta_pic_order_cnt_bottom = 0;
}
else
{
img->pic_order_present_flag=1;
img->delta_pic_order_cnt_bottom = 1;
}
}
/*!
***********************************************************************
* \brief
* Pushes values onto the POC ref arrays toprefpoc[] & bottomrefpoc[]
*
***********************************************************************
*/
void push_poc(unsigned int topvalue, unsigned int bottomvalue, unsigned int ref_frame_ind )
{
int i;
static int current_is_ref = 0; //indicates if current top value is for a ref frame
if(current_is_ref){
for(i=MAX_NO_POC_FRAMES-1; i>0; i--){ //move all the data down by one
toprefpoc[i] = toprefpoc[i-1] ;
bottomrefpoc[i] = bottomrefpoc[i-1] ;
}
}
toprefpoc[0] = topvalue; //put new data
bottomrefpoc[0] = bottomvalue;
current_is_ref = ref_frame_ind; //new data type
}
/*!
***********************************************************************
* \brief
* Initializes the img->nz_coeff
* \par Input:
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -