libbpg/jctvc/TLibEncoder/TEncRateCtrl.h
2015-01-16 13:46:18 +01:00

335 lines
13 KiB
C++

/* The copyright in this software is being made available under the BSD
* License, included below. This software may be subject to other third party
* and contributor rights, including patent rights, and no such rights are
* granted under this license.
*
* Copyright (c) 2010-2014, ITU/ISO/IEC
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* * Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
* * Neither the name of the ITU/ISO/IEC nor the names of its contributors may
* be used to endorse or promote products derived from this software without
* specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
/** \file TEncRateCtrl.h
\brief Rate control manager class
*/
#ifndef __TENCRATECTRL__
#define __TENCRATECTRL__
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000
#include "../TLibCommon/CommonDef.h"
#include "../TLibCommon/TComDataCU.h"
#include <vector>
#include <algorithm>
using namespace std;
//! \ingroup TLibEncoder
//! \{
#include "../TLibEncoder/TEncCfg.h"
#include <list>
#include <cassert>
const Int g_RCInvalidQPValue = -999;
const Int g_RCSmoothWindowSize = 40;
const Int g_RCMaxPicListSize = 32;
const Double g_RCWeightPicTargetBitInGOP = 0.9;
const Double g_RCWeightPicRargetBitInBuffer = 1.0 - g_RCWeightPicTargetBitInGOP;
const Int g_RCIterationNum = 20;
const Double g_RCWeightHistoryLambda = 0.5;
const Double g_RCWeightCurrentLambda = 1.0 - g_RCWeightHistoryLambda;
const Int g_RCLCUSmoothWindowSize = 4;
const Double g_RCAlphaMinValue = 0.05;
const Double g_RCAlphaMaxValue = 500.0;
const Double g_RCBetaMinValue = -3.0;
const Double g_RCBetaMaxValue = -0.1;
#define ALPHA 6.7542;
#define BETA1 1.2517
#define BETA2 1.7860
struct TRCLCU
{
Int m_actualBits;
Int m_QP; // QP of skip mode is set to g_RCInvalidQPValue
Int m_targetBits;
Double m_lambda;
Double m_bitWeight;
Int m_numberOfPixel;
Double m_costIntra;
Int m_targetBitsLeft;
};
struct TRCParameter
{
Double m_alpha;
Double m_beta;
};
class TEncRCSeq
{
public:
TEncRCSeq();
~TEncRCSeq();
public:
Void create( Int totalFrames, Int targetBitrate, Int frameRate, Int GOPSize, Int picWidth, Int picHeight, Int LCUWidth, Int LCUHeight, Int numberOfLevel, Bool useLCUSeparateModel, Int adaptiveBit );
Void destroy();
Void initBitsRatio( Int bitsRatio[] );
Void initGOPID2Level( Int GOPID2Level[] );
Void initPicPara( TRCParameter* picPara = NULL ); // NULL to initial with default value
Void initLCUPara( TRCParameter** LCUPara = NULL ); // NULL to initial with default value
Void updateAfterPic ( Int bits );
Void setAllBitRatio( Double basicLambda, Double* equaCoeffA, Double* equaCoeffB );
public:
Int getTotalFrames() { return m_totalFrames; }
Int getTargetRate() { return m_targetRate; }
Int getFrameRate() { return m_frameRate; }
Int getGOPSize() { return m_GOPSize; }
Int getPicWidth() { return m_picWidth; }
Int getPicHeight() { return m_picHeight; }
Int getLCUWidth() { return m_LCUWidth; }
Int getLCUHeight() { return m_LCUHeight; }
Int getNumberOfLevel() { return m_numberOfLevel; }
Int getAverageBits() { return m_averageBits; }
Int getLeftAverageBits() { assert( m_framesLeft > 0 ); return (Int)(m_bitsLeft / m_framesLeft); }
Bool getUseLCUSeparateModel() { return m_useLCUSeparateModel; }
Int getNumPixel() { return m_numberOfPixel; }
Int64 getTargetBits() { return m_targetBits; }
Int getNumberOfLCU() { return m_numberOfLCU; }
Int* getBitRatio() { return m_bitsRatio; }
Int getBitRatio( Int idx ) { assert( idx<m_GOPSize); return m_bitsRatio[idx]; }
Int* getGOPID2Level() { return m_GOPID2Level; }
Int getGOPID2Level( Int ID ) { assert( ID < m_GOPSize ); return m_GOPID2Level[ID]; }
TRCParameter* getPicPara() { return m_picPara; }
TRCParameter getPicPara( Int level ) { assert( level < m_numberOfLevel ); return m_picPara[level]; }
Void setPicPara( Int level, TRCParameter para ) { assert( level < m_numberOfLevel ); m_picPara[level] = para; }
TRCParameter** getLCUPara() { return m_LCUPara; }
TRCParameter* getLCUPara( Int level ) { assert( level < m_numberOfLevel ); return m_LCUPara[level]; }
TRCParameter getLCUPara( Int level, Int LCUIdx ) { assert( LCUIdx < m_numberOfLCU ); return getLCUPara(level)[LCUIdx]; }
Void setLCUPara( Int level, Int LCUIdx, TRCParameter para ) { assert( level < m_numberOfLevel ); assert( LCUIdx < m_numberOfLCU ); m_LCUPara[level][LCUIdx] = para; }
Int getFramesLeft() { return m_framesLeft; }
Int64 getBitsLeft() { return m_bitsLeft; }
Double getSeqBpp() { return m_seqTargetBpp; }
Double getAlphaUpdate() { return m_alphaUpdate; }
Double getBetaUpdate() { return m_betaUpdate; }
Int getAdaptiveBits() { return m_adaptiveBit; }
Double getLastLambda() { return m_lastLambda; }
Void setLastLambda( Double lamdba ) { m_lastLambda = lamdba; }
private:
Int m_totalFrames;
Int m_targetRate;
Int m_frameRate;
Int m_GOPSize;
Int m_picWidth;
Int m_picHeight;
Int m_LCUWidth;
Int m_LCUHeight;
Int m_numberOfLevel;
Int m_averageBits;
Int m_numberOfPixel;
Int64 m_targetBits;
Int m_numberOfLCU;
Int* m_bitsRatio;
Int* m_GOPID2Level;
TRCParameter* m_picPara;
TRCParameter** m_LCUPara;
Int m_framesLeft;
Int64 m_bitsLeft;
Double m_seqTargetBpp;
Double m_alphaUpdate;
Double m_betaUpdate;
Bool m_useLCUSeparateModel;
Int m_adaptiveBit;
Double m_lastLambda;
};
class TEncRCGOP
{
public:
TEncRCGOP();
~TEncRCGOP();
public:
Void create( TEncRCSeq* encRCSeq, Int numPic );
Void destroy();
Void updateAfterPicture( Int bitsCost );
private:
Int xEstGOPTargetBits( TEncRCSeq* encRCSeq, Int GOPSize );
Void xCalEquaCoeff( TEncRCSeq* encRCSeq, Double* lambdaRatio, Double* equaCoeffA, Double* equaCoeffB, Int GOPSize );
Double xSolveEqua( Double targetBpp, Double* equaCoeffA, Double* equaCoeffB, Int GOPSize );
public:
TEncRCSeq* getEncRCSeq() { return m_encRCSeq; }
Int getNumPic() { return m_numPic;}
Int getTargetBits() { return m_targetBits; }
Int getPicLeft() { return m_picLeft; }
Int getBitsLeft() { return m_bitsLeft; }
Int getTargetBitInGOP( Int i ) { return m_picTargetBitInGOP[i]; }
private:
TEncRCSeq* m_encRCSeq;
Int* m_picTargetBitInGOP;
Int m_numPic;
Int m_targetBits;
Int m_picLeft;
Int m_bitsLeft;
};
class TEncRCPic
{
public:
TEncRCPic();
~TEncRCPic();
public:
Void create( TEncRCSeq* encRCSeq, TEncRCGOP* encRCGOP, Int frameLevel, list<TEncRCPic*>& listPreviousPictures );
Void destroy();
Int estimatePicQP ( Double lambda, list<TEncRCPic*>& listPreviousPictures );
Int getRefineBitsForIntra(Int orgBits);
Double calculateLambdaIntra(Double alpha, Double beta, Double MADPerPixel, Double bitsPerPixel);
Double estimatePicLambda( list<TEncRCPic*>& listPreviousPictures, SliceType eSliceType);
Void updateAlphaBetaIntra(Double *alpha, Double *beta);
Double getLCUTargetBpp(SliceType eSliceType);
Double getLCUEstLambdaAndQP(Double bpp, Int clipPicQP, Int *estQP);
Double getLCUEstLambda( Double bpp );
Int getLCUEstQP( Double lambda, Int clipPicQP );
Void updateAfterCTU( Int LCUIdx, Int bits, Int QP, Double lambda, Bool updateLCUParameter = true );
Void updateAfterPicture( Int actualHeaderBits, Int actualTotalBits, Double averageQP, Double averageLambda, SliceType eSliceType);
Void addToPictureLsit( list<TEncRCPic*>& listPreviousPictures );
Double calAverageQP();
Double calAverageLambda();
private:
Int xEstPicTargetBits( TEncRCSeq* encRCSeq, TEncRCGOP* encRCGOP );
Int xEstPicHeaderBits( list<TEncRCPic*>& listPreviousPictures, Int frameLevel );
public:
TEncRCSeq* getRCSequence() { return m_encRCSeq; }
TEncRCGOP* getRCGOP() { return m_encRCGOP; }
Int getFrameLevel() { return m_frameLevel; }
Int getNumberOfPixel() { return m_numberOfPixel; }
Int getNumberOfLCU() { return m_numberOfLCU; }
Int getTargetBits() { return m_targetBits; }
Int getEstHeaderBits() { return m_estHeaderBits; }
Int getLCULeft() { return m_LCULeft; }
Int getBitsLeft() { return m_bitsLeft; }
Int getPixelsLeft() { return m_pixelsLeft; }
Int getBitsCoded() { return m_targetBits - m_estHeaderBits - m_bitsLeft; }
Int getLCUCoded() { return m_numberOfLCU - m_LCULeft; }
TRCLCU* getLCU() { return m_LCUs; }
TRCLCU& getLCU( Int LCUIdx ) { return m_LCUs[LCUIdx]; }
Int getPicActualHeaderBits() { return m_picActualHeaderBits; }
Void setTargetBits( Int bits ) { m_targetBits = bits; m_bitsLeft = bits;}
Void setTotalIntraCost(Double cost) { m_totalCostIntra = cost; }
Void getLCUInitTargetBits();
Int getPicActualBits() { return m_picActualBits; }
Int getPicActualQP() { return m_picQP; }
Double getPicActualLambda() { return m_picLambda; }
Int getPicEstQP() { return m_estPicQP; }
Void setPicEstQP( Int QP ) { m_estPicQP = QP; }
Double getPicEstLambda() { return m_estPicLambda; }
Void setPicEstLambda( Double lambda ) { m_picLambda = lambda; }
private:
TEncRCSeq* m_encRCSeq;
TEncRCGOP* m_encRCGOP;
Int m_frameLevel;
Int m_numberOfPixel;
Int m_numberOfLCU;
Int m_targetBits;
Int m_estHeaderBits;
Int m_estPicQP;
Double m_estPicLambda;
Int m_LCULeft;
Int m_bitsLeft;
Int m_pixelsLeft;
TRCLCU* m_LCUs;
Int m_picActualHeaderBits; // only SH and potential APS
Double m_totalCostIntra;
Double m_remainingCostIntra;
Int m_picActualBits; // the whole picture, including header
Int m_picQP; // in integer form
Double m_picLambda;
};
class TEncRateCtrl
{
public:
TEncRateCtrl();
~TEncRateCtrl();
public:
Void init( Int totalFrames, Int targetBitrate, Int frameRate, Int GOPSize, Int picWidth, Int picHeight, Int LCUWidth, Int LCUHeight, Int keepHierBits, Bool useLCUSeparateModel, GOPEntry GOPList[MAX_GOP] );
Void destroy();
Void initRCPic( Int frameLevel );
Void initRCGOP( Int numberOfPictures );
Void destroyRCGOP();
public:
Void setRCQP ( Int QP ) { m_RCQP = QP; }
Int getRCQP () { return m_RCQP; }
TEncRCSeq* getRCSeq() { assert ( m_encRCSeq != NULL ); return m_encRCSeq; }
TEncRCGOP* getRCGOP() { assert ( m_encRCGOP != NULL ); return m_encRCGOP; }
TEncRCPic* getRCPic() { assert ( m_encRCPic != NULL ); return m_encRCPic; }
list<TEncRCPic*>& getPicList() { return m_listRCPictures; }
private:
TEncRCSeq* m_encRCSeq;
TEncRCGOP* m_encRCGOP;
TEncRCPic* m_encRCPic;
list<TEncRCPic*> m_listRCPictures;
Int m_RCQP;
};
#endif