405 lines
14 KiB
C
405 lines
14 KiB
C
/******************************************************************************
|
|
*
|
|
* Copyright (C) 2018 The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at:
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*
|
|
*****************************************************************************
|
|
* Originally developed and contributed by Ittiam Systems Pvt. Ltd, Bangalore
|
|
*/
|
|
/*!
|
|
******************************************************************************
|
|
* \file vbr_storage_vbv.c
|
|
*
|
|
* \brief
|
|
* This file contain functions related to VBV buffer
|
|
*
|
|
* \date
|
|
*
|
|
* \author
|
|
* ittiam
|
|
*
|
|
******************************************************************************
|
|
*/
|
|
/*****************************************************************************/
|
|
/* File Includes */
|
|
/*****************************************************************************/
|
|
/* System include files */
|
|
#include <stdio.h>
|
|
|
|
/* User include files */
|
|
#include "ittiam_datatypes.h"
|
|
#include "mem_req_and_acq.h"
|
|
#include "rc_common.h"
|
|
#include "rc_cntrl_param.h"
|
|
#include "var_q_operator.h"
|
|
#include "fixed_point_error_bits.h"
|
|
#include "cbr_buffer_control.h"
|
|
#include "rc_rd_model.h"
|
|
#include "est_sad.h"
|
|
#include "cbr_buffer_control.h"
|
|
#include "picture_type.h"
|
|
#include "bit_allocation.h"
|
|
#include "vbr_storage_vbv.h"
|
|
#include "trace_support.h"
|
|
|
|
#define MAX(x, y) ((x) > (y) ? (x) : (y))
|
|
|
|
typedef struct vbr_storage_vbv_t
|
|
{
|
|
WORD32 i4_max_buf_size;
|
|
WORD32 i4_cur_buf_size;
|
|
WORD32 i4_max_bits_inflow_per_frm_period;
|
|
/* Storing input variables */
|
|
WORD32 i4_max_bit_rate;
|
|
WORD32 i4_max_frame_rate;
|
|
/* Error bits calculation module */
|
|
error_bits_handle ps_error_bits;
|
|
} vbr_storage_vbv_t;
|
|
|
|
#if NON_STEADSTATE_CODE
|
|
|
|
WORD32 vbr_vbv_num_fill_use_free_memtab(
|
|
vbr_storage_vbv_t **pps_vbr_storage_vbv, itt_memtab_t *ps_memtab, ITT_FUNC_TYPE_E e_func_type)
|
|
{
|
|
WORD32 i4_mem_tab_idx = 0;
|
|
static vbr_storage_vbv_t s_vbr_storage_vbv_temp;
|
|
|
|
/* Hack for al alloc, during which we dont have any state memory.
|
|
Dereferencing can cause issues */
|
|
if(e_func_type == GET_NUM_MEMTAB || e_func_type == FILL_MEMTAB)
|
|
(*pps_vbr_storage_vbv) = &s_vbr_storage_vbv_temp;
|
|
|
|
/*for src rate control state structure*/
|
|
if(e_func_type != GET_NUM_MEMTAB)
|
|
{
|
|
fill_memtab(
|
|
&ps_memtab[i4_mem_tab_idx],
|
|
sizeof(vbr_storage_vbv_t),
|
|
MEM_TAB_ALIGNMENT,
|
|
PERSISTENT,
|
|
DDR);
|
|
use_or_fill_base(&ps_memtab[0], (void **)pps_vbr_storage_vbv, e_func_type);
|
|
}
|
|
i4_mem_tab_idx++;
|
|
|
|
i4_mem_tab_idx += error_bits_num_fill_use_free_memtab(
|
|
&pps_vbr_storage_vbv[0]->ps_error_bits, &ps_memtab[i4_mem_tab_idx], e_func_type);
|
|
return (i4_mem_tab_idx);
|
|
}
|
|
/******************************************************************************
|
|
Function Name : init_vbr_vbv
|
|
Description :
|
|
Arguments : ps_vbr_storage_vbv
|
|
Return Values : void
|
|
Revision History:
|
|
Creation
|
|
*****************************************************************************/
|
|
void init_vbr_vbv(
|
|
vbr_storage_vbv_t *ps_vbr_storage_vbv,
|
|
WORD32 i4_max_bit_rate,
|
|
WORD32 i4_frm_rate,
|
|
WORD32 i4_max_vbv_buff_size)
|
|
{
|
|
ps_vbr_storage_vbv->i4_max_buf_size = i4_max_vbv_buff_size;
|
|
ps_vbr_storage_vbv->i4_cur_buf_size = i4_max_vbv_buff_size;
|
|
|
|
/* Calculate the max number of bits that flow into the decoder
|
|
in the interval of two frames */
|
|
X_PROD_Y_DIV_Z(
|
|
i4_max_bit_rate, 1000, i4_frm_rate, ps_vbr_storage_vbv->i4_max_bits_inflow_per_frm_period);
|
|
|
|
/* init error bits */
|
|
init_error_bits(ps_vbr_storage_vbv->ps_error_bits, i4_frm_rate, i4_max_bit_rate);
|
|
|
|
/* Storing the input values */
|
|
ps_vbr_storage_vbv->i4_max_bit_rate = i4_max_bit_rate;
|
|
ps_vbr_storage_vbv->i4_max_frame_rate = i4_frm_rate;
|
|
}
|
|
#endif /* #if NON_STEADSTATE_CODE */
|
|
/******************************************************************************
|
|
Function Name : update_vbr_vbv
|
|
Description :
|
|
Arguments : ps_vbr_storage_vbv
|
|
Return Values : void
|
|
Revision History:
|
|
Creation
|
|
*****************************************************************************/
|
|
void update_vbr_vbv(vbr_storage_vbv_t *ps_vbr_storage_vbv, WORD32 i4_total_bits_decoded)
|
|
{
|
|
WORD32 i4_error_bits = get_error_bits(ps_vbr_storage_vbv->ps_error_bits);
|
|
/* In the time interval between two decoded frames the buffer would have been
|
|
filled up by the max_bits_inflow_per_frm_period.*/
|
|
overflow_avoided_summation(
|
|
&ps_vbr_storage_vbv->i4_cur_buf_size,
|
|
(ps_vbr_storage_vbv->i4_max_bits_inflow_per_frm_period + i4_error_bits));
|
|
|
|
if(ps_vbr_storage_vbv->i4_cur_buf_size > ps_vbr_storage_vbv->i4_max_buf_size)
|
|
{
|
|
ps_vbr_storage_vbv->i4_cur_buf_size = ps_vbr_storage_vbv->i4_max_buf_size;
|
|
}
|
|
|
|
ps_vbr_storage_vbv->i4_cur_buf_size -= i4_total_bits_decoded;
|
|
|
|
/* Update the error bits state */
|
|
update_error_bits(ps_vbr_storage_vbv->ps_error_bits);
|
|
|
|
#define PRINT_UNDERFLOW 0
|
|
#if PRINT_UNDERFLOW
|
|
if(ps_vbr_storage_vbv->i4_cur_buf_size < 0)
|
|
printf("The buffer underflows \n");
|
|
#endif
|
|
}
|
|
/******************************************************************************
|
|
Function Name : get_max_target_bits
|
|
Description :
|
|
Arguments : ps_vbr_storage_vbv
|
|
Return Values : void
|
|
Revision History:
|
|
Creation
|
|
*****************************************************************************/
|
|
WORD32 get_max_target_bits(vbr_storage_vbv_t *ps_vbr_storage_vbv)
|
|
{
|
|
WORD32 i4_cur_buf_size = ps_vbr_storage_vbv->i4_cur_buf_size;
|
|
WORD32 i4_error_bits = get_error_bits(ps_vbr_storage_vbv->ps_error_bits);
|
|
|
|
/* The buffer size when the next frame is decoded */
|
|
overflow_avoided_summation(
|
|
&i4_cur_buf_size, (ps_vbr_storage_vbv->i4_max_bits_inflow_per_frm_period + i4_error_bits));
|
|
if(i4_cur_buf_size > ps_vbr_storage_vbv->i4_max_buf_size)
|
|
{
|
|
i4_cur_buf_size = ps_vbr_storage_vbv->i4_max_buf_size;
|
|
}
|
|
|
|
/* Thus for the next frame the maximum number of bits the decoder can consume
|
|
without underflow is i4_cur_buf_size */
|
|
return i4_cur_buf_size;
|
|
}
|
|
|
|
/****************************************************************************
|
|
Function Name : get_buffer_status
|
|
Description : Gets the state of VBV buffer
|
|
Inputs : Rate control API , header and texture bits
|
|
Globals :
|
|
Processing :
|
|
Outputs : 0 = normal, 1 = underflow, 2= overflow
|
|
Returns : vbv_buf_status_e
|
|
Issues :
|
|
Revision History:
|
|
DD MM YYYY Author(s) Changes (Describe the changes made)
|
|
*****************************************************************************/
|
|
vbv_buf_status_e get_vbv_buffer_status(
|
|
vbr_storage_vbv_t *ps_vbr_storage_vbv,
|
|
WORD32 i4_total_frame_bits, /* Total frame bits consumed */
|
|
WORD32 *pi4_num_bits_to_prevent_vbv_underflow) /* The curent buffer status after updation */
|
|
{
|
|
vbv_buf_status_e e_buf_status;
|
|
WORD32 i4_cur_buf;
|
|
WORD32 i4_error_bits = get_error_bits(ps_vbr_storage_vbv->ps_error_bits);
|
|
|
|
/* error bits due to fixed point computation of drain rate*/
|
|
i4_cur_buf = ps_vbr_storage_vbv->i4_cur_buf_size;
|
|
overflow_avoided_summation(
|
|
&i4_cur_buf, (ps_vbr_storage_vbv->i4_max_bits_inflow_per_frm_period + i4_error_bits));
|
|
|
|
if(i4_cur_buf > ps_vbr_storage_vbv->i4_max_buf_size)
|
|
{
|
|
i4_cur_buf = ps_vbr_storage_vbv->i4_max_buf_size;
|
|
}
|
|
|
|
pi4_num_bits_to_prevent_vbv_underflow[0] = i4_cur_buf;
|
|
|
|
i4_cur_buf -= i4_total_frame_bits;
|
|
if(i4_cur_buf < 0)
|
|
{
|
|
e_buf_status = VBV_UNDERFLOW;
|
|
}
|
|
else if(i4_cur_buf > ps_vbr_storage_vbv->i4_max_buf_size)
|
|
{
|
|
e_buf_status = VBV_OVERFLOW;
|
|
}
|
|
else if(i4_cur_buf < (ps_vbr_storage_vbv->i4_max_buf_size >> 2))
|
|
{
|
|
e_buf_status = VBR_CAUTION;
|
|
}
|
|
else
|
|
{
|
|
e_buf_status = VBV_NORMAL;
|
|
}
|
|
|
|
return e_buf_status;
|
|
}
|
|
/******************************************************************************
|
|
Function Name : get_max_vbv_buf_size
|
|
Description :
|
|
Arguments : ps_vbr_storage_vbv
|
|
Return Values : void
|
|
Revision History:
|
|
Creation
|
|
*****************************************************************************/
|
|
WORD32 get_max_vbv_buf_size(vbr_storage_vbv_t *ps_vbr_storage_vbv)
|
|
{
|
|
return (ps_vbr_storage_vbv->i4_max_buf_size);
|
|
}
|
|
/******************************************************************************
|
|
Function Name : get_cur_vbv_buf_size
|
|
Description :
|
|
Arguments : ps_vbr_storage_vbv
|
|
Return Values : void
|
|
Revision History:
|
|
Creation
|
|
*****************************************************************************/
|
|
WORD32 get_cur_vbv_buf_size(vbr_storage_vbv_t *ps_vbr_storage_vbv)
|
|
{
|
|
return (ps_vbr_storage_vbv->i4_cur_buf_size);
|
|
}
|
|
/******************************************************************************
|
|
Function Name : get_max_bits_inflow_per_frm_periode
|
|
Description :
|
|
Arguments : ps_vbr_storage_vbv
|
|
Return Values : void
|
|
Revision History:
|
|
Creation
|
|
*****************************************************************************/
|
|
WORD32 get_max_bits_inflow_per_frm_periode(vbr_storage_vbv_t *ps_vbr_storage_vbv)
|
|
{
|
|
return (ps_vbr_storage_vbv->i4_max_bits_inflow_per_frm_period);
|
|
}
|
|
|
|
/******************************************************************************
|
|
Function Name : get_vbv_buf_fullness
|
|
Description :
|
|
Arguments : ps_vbr_storage_vbv
|
|
Return Values : void
|
|
Revision History:
|
|
Creation
|
|
*****************************************************************************/
|
|
WORD32 get_vbv_buf_fullness(vbr_storage_vbv_t *ps_vbr_storage_vbv, UWORD32 u4_bits)
|
|
{
|
|
WORD32 i4_error_bits = get_error_bits(ps_vbr_storage_vbv->ps_error_bits);
|
|
WORD32 i4_cur_buf_size = ps_vbr_storage_vbv->i4_cur_buf_size;
|
|
|
|
overflow_avoided_summation(
|
|
&i4_cur_buf_size, (ps_vbr_storage_vbv->i4_max_bits_inflow_per_frm_period + i4_error_bits));
|
|
|
|
if(i4_cur_buf_size > ps_vbr_storage_vbv->i4_max_buf_size)
|
|
{
|
|
i4_cur_buf_size = ps_vbr_storage_vbv->i4_max_buf_size;
|
|
}
|
|
|
|
i4_cur_buf_size -= u4_bits;
|
|
|
|
#if PRINT_UNDERFLOW
|
|
if(i4_cur_buf_size < 0)
|
|
printf("The buffer underflows \n");
|
|
#endif
|
|
return (i4_cur_buf_size);
|
|
}
|
|
/******************************************************************************
|
|
Function Name : get_max_tgt_bits_dvd_comp
|
|
Description :
|
|
Arguments : ps_vbr_storage_vbv
|
|
Return Values : void
|
|
Revision History:
|
|
Creation
|
|
*****************************************************************************/
|
|
WORD32 get_max_tgt_bits_dvd_comp(
|
|
vbr_storage_vbv_t *ps_vbr_storage_vbv,
|
|
WORD32 i4_rem_bits_in_gop,
|
|
WORD32 i4_rem_frms_in_gop,
|
|
picture_type_e e_pic_type)
|
|
{
|
|
WORD32 i4_dbf_max, i4_dbf_min, i4_dbf_prev, i4_vbv_size, i4_dbf_desired;
|
|
WORD32 i4_max_tgt_bits;
|
|
|
|
i4_vbv_size = ps_vbr_storage_vbv->i4_max_buf_size;
|
|
i4_dbf_max = 95 * i4_vbv_size / 100;
|
|
i4_dbf_min = 10 * i4_vbv_size / 100;
|
|
i4_dbf_prev = ps_vbr_storage_vbv->i4_cur_buf_size;
|
|
|
|
if(i4_rem_bits_in_gop < 0)
|
|
i4_rem_bits_in_gop = 0;
|
|
if(i4_rem_frms_in_gop <= 0)
|
|
i4_rem_frms_in_gop = 1;
|
|
|
|
if(e_pic_type == I_PIC)
|
|
{
|
|
i4_dbf_desired = i4_dbf_min;
|
|
}
|
|
else
|
|
{
|
|
i4_dbf_desired = (i4_dbf_max - i4_rem_bits_in_gop / i4_rem_frms_in_gop - i4_dbf_prev) /
|
|
i4_rem_frms_in_gop;
|
|
i4_dbf_desired += i4_dbf_prev;
|
|
}
|
|
|
|
i4_dbf_prev += ps_vbr_storage_vbv->i4_max_bits_inflow_per_frm_period;
|
|
if(i4_dbf_prev > ps_vbr_storage_vbv->i4_max_buf_size)
|
|
{
|
|
i4_dbf_prev = ps_vbr_storage_vbv->i4_max_buf_size;
|
|
}
|
|
|
|
i4_max_tgt_bits = MAX(0, (i4_dbf_prev - i4_dbf_desired));
|
|
return (i4_max_tgt_bits);
|
|
}
|
|
|
|
#if NON_STEADSTATE_CODE
|
|
/******************************************************************************
|
|
Function Name : change_vbr_vbv_frame_rate
|
|
Description :
|
|
Arguments : ps_vbr_storage_vbv
|
|
Return Values : void
|
|
Revision History:
|
|
Creation
|
|
*****************************************************************************/
|
|
void change_vbr_vbv_frame_rate(vbr_storage_vbv_t *ps_vbr_storage_vbv, WORD32 i4_frm_rate)
|
|
{
|
|
/* Calculate the max number of bits that flow into the decoder
|
|
in the interval of two frames */
|
|
X_PROD_Y_DIV_Z(
|
|
ps_vbr_storage_vbv->i4_max_bit_rate,
|
|
1000,
|
|
i4_frm_rate,
|
|
ps_vbr_storage_vbv->i4_max_bits_inflow_per_frm_period);
|
|
|
|
/* update the lower modules */
|
|
change_frm_rate_in_error_bits(ps_vbr_storage_vbv->ps_error_bits, i4_frm_rate);
|
|
/* Storing the input values */
|
|
ps_vbr_storage_vbv->i4_max_frame_rate = i4_frm_rate;
|
|
}
|
|
/******************************************************************************
|
|
Function Name : change_vbr_vbv_bit_rate
|
|
Description :
|
|
Arguments : ps_vbr_storage_vbv
|
|
Return Values : void
|
|
Revision History:
|
|
Creation
|
|
*****************************************************************************/
|
|
void change_vbr_vbv_bit_rate(vbr_storage_vbv_t *ps_vbr_storage_vbv, WORD32 i4_max_bit_rate)
|
|
{
|
|
/* Calculate the max number of bits that flow into the decoder
|
|
in the interval of two frames */
|
|
X_PROD_Y_DIV_Z(
|
|
i4_max_bit_rate,
|
|
1000,
|
|
ps_vbr_storage_vbv->i4_max_frame_rate,
|
|
ps_vbr_storage_vbv->i4_max_bits_inflow_per_frm_period);
|
|
|
|
/* update the lower modules */
|
|
change_bitrate_in_error_bits(ps_vbr_storage_vbv->ps_error_bits, i4_max_bit_rate);
|
|
|
|
/* Storing the input values */
|
|
ps_vbr_storage_vbv->i4_max_bit_rate = i4_max_bit_rate;
|
|
}
|
|
#endif /* #if NON_STEADSTATE_CODE */
|