889 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			889 lines
		
	
	
		
			33 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
 | |
| */
 | |
| #include <stdio.h>
 | |
| #include <string.h>
 | |
| #include "ixheaacd_sbr_common.h"
 | |
| #include "ixheaacd_type_def.h"
 | |
| 
 | |
| #include "ixheaacd_constants.h"
 | |
| #include "ixheaacd_basic_ops32.h"
 | |
| #include "ixheaacd_basic_ops16.h"
 | |
| #include "ixheaacd_basic_ops40.h"
 | |
| #include "ixheaacd_basic_ops.h"
 | |
| #include "ixheaacd_bitbuffer.h"
 | |
| 
 | |
| #include "ixheaacd_basic_op.h"
 | |
| #include "ixheaacd_intrinsics.h"
 | |
| 
 | |
| #include "ixheaacd_defines.h"
 | |
| 
 | |
| #include "ixheaacd_aac_rom.h"
 | |
| 
 | |
| #include "ixheaacd_definitions.h"
 | |
| 
 | |
| #include "ixheaacd_error_codes.h"
 | |
| 
 | |
| #include "ixheaacd_pulsedata.h"
 | |
| 
 | |
| #include "ixheaacd_pns.h"
 | |
| #include "ixheaacd_drc_data_struct.h"
 | |
| 
 | |
| #include "ixheaacd_lt_predict.h"
 | |
| 
 | |
| #include "ixheaacd_channelinfo.h"
 | |
| #include "ixheaacd_cnst.h"
 | |
| #include "ixheaacd_drc_dec.h"
 | |
| #include "ixheaacd_sbrdecoder.h"
 | |
| #include "ixheaacd_block.h"
 | |
| #include "ixheaacd_channel.h"
 | |
| 
 | |
| #include "ixheaacd_sbr_payload.h"
 | |
| #include "ixheaacd_common_rom.h"
 | |
| #include "ixheaacd_sbrdecsettings.h"
 | |
| #include "ixheaacd_sbr_scale.h"
 | |
| #include "ixheaacd_env_extr_part.h"
 | |
| #include "ixheaacd_sbr_rom.h"
 | |
| #include "ixheaacd_stereo.h"
 | |
| #include "ixheaacd_lpp_tran.h"
 | |
| #include "ixheaacd_hybrid.h"
 | |
| #include "ixheaacd_ps_dec.h"
 | |
| #include "ixheaacd_env_extr.h"
 | |
| #include "ixheaacd_adts.h"
 | |
| #include "ixheaacd_audioobjtypes.h"
 | |
| #include "ixheaacd_memory_standards.h"
 | |
| #include "ixheaacd_latmdemux.h"
 | |
| #include "ixheaacd_aacdec.h"
 | |
| #include "ixheaacd_mps_polyphase.h"
 | |
| #include "ixheaacd_config.h"
 | |
| #include "ixheaacd_mps_dec.h"
 | |
| #include "ixheaacd_struct_def.h"
 | |
| #include "ixheaacd_headerdecode.h"
 | |
| #include "ixheaacd_multichannel.h"
 | |
| #include "ixheaacd_adts_crc_check.h"
 | |
| 
 | |
| #include "ixheaacd_hcr.h"
 | |
| 
 | |
| #define SIZEOF_INT(x) ((sizeof(x) + sizeof(WORD32) - 1) / sizeof(WORD32))
 | |
| 
 | |
| #define EXT_FILL_DATA 1
 | |
| #define EXT_FIL 0
 | |
| 
 | |
| WORD32 ixheaacd_aacdec_decodeframe(
 | |
|     ia_exhaacplus_dec_api_struct *p_obj_exhaacplus_dec,
 | |
|     ia_aac_dec_scratch_struct *aac_scratch_ptrs, WORD16 *time_data,
 | |
|     FLAG frame_status, WORD *type, WORD *ch_idx, WORD init_flag, WORD channel,
 | |
|     WORD *element_index_order, WORD skip_full_decode, WORD ch_fac,
 | |
|     WORD slot_element, WORD max_channels, WORD32 total_channels,
 | |
|     WORD32 frame_length, WORD32 frame_size, ia_drc_dec_struct *pstr_drc_dec,
 | |
|     WORD32 object_type, WORD32 ch_config,
 | |
|     ia_eld_specific_config_struct eld_specific_config, WORD16 adtsheader,
 | |
|     ia_drc_dec_struct *drc_dummy)
 | |
| 
 | |
| {
 | |
|   WORD ch, ele_type;
 | |
|   ia_aac_dec_state_struct *p_state_enhaacplus_dec;
 | |
|   ia_aac_decoder_struct *aac_dec_handle;
 | |
|   ia_bit_buf_struct *it_bit_buff;
 | |
|   ixheaacd_latm_struct *latm_element;
 | |
| 
 | |
|   WORD error_code = (WORD)frame_status;
 | |
|   WORD previous_element;
 | |
|   WORD prev_data_ele_present = 0;
 | |
|   WORD new_element;
 | |
|   WORD32 num_ch = 0;
 | |
| 
 | |
|   WORD32 crc_reg = 0;
 | |
|   ia_adts_crc_info_struct *ptr_adts_crc_info;
 | |
| 
 | |
|   WORD32 cnt_bits = 0;
 | |
| 
 | |
|   WORD32 eld_sbr_flag = eld_specific_config.ld_sbr_flag_present;
 | |
|   WORD32 ld_sbr_crc_flag = eld_specific_config.ld_sbr_crc_flag;
 | |
|   WORD32 aac_spect_data_resil_flag =
 | |
|       eld_specific_config.aac_spect_data_resil_flag;
 | |
| 
 | |
|   WORD32 ele_ch = 0;
 | |
| 
 | |
|   ia_aac_sfb_code_book_struct *ptr_aac_sfb_code_book_data[CHANNELS];
 | |
|   ia_pns_stereo_data_struct *ptr_pns_stereo_data;
 | |
| 
 | |
|   WORD32 *work_buffer_core = aac_scratch_ptrs->base_scr_8k;
 | |
|   WORD32 *work_buffer_1 = aac_scratch_ptrs->extra_scr_4k[0];
 | |
|   WORD32 *work_buffer_2 = aac_scratch_ptrs->extra_scr_4k[2];
 | |
|   p_state_enhaacplus_dec = p_obj_exhaacplus_dec->p_state_aac;
 | |
| 
 | |
|   aac_dec_handle = p_state_enhaacplus_dec->pstr_aac_dec_info[*ch_idx];
 | |
|   it_bit_buff = p_state_enhaacplus_dec->ptr_bit_stream;
 | |
| 
 | |
|   ptr_adts_crc_info = it_bit_buff->pstr_adts_crc_info;
 | |
| 
 | |
|   latm_element = &p_state_enhaacplus_dec->latm_struct_element;
 | |
| 
 | |
|   ptr_pns_stereo_data =
 | |
|       (ia_pns_stereo_data_struct
 | |
|            *)&work_buffer_1[2 * SIZEOF_INT(ia_aac_dec_channel_info_struct) +
 | |
|                             2 * SIZEOF_INT(ia_aac_sfb_code_book_struct)];
 | |
| 
 | |
|   aac_dec_handle->frame_status = 1;
 | |
| 
 | |
|   for (ch = 0; ch < channel; ch++) {
 | |
|     const ia_aac_dec_imdct_tables_struct *pstr_imdct_tables;
 | |
|     aac_dec_handle->pstr_aac_dec_ch_info[ch] =
 | |
|         (ia_aac_dec_channel_info_struct
 | |
|              *)&work_buffer_1[ch * SIZEOF_INT(ia_aac_dec_channel_info_struct)];
 | |
|     ptr_aac_sfb_code_book_data[ch] =
 | |
|         (ia_aac_sfb_code_book_struct
 | |
|              *)&work_buffer_1[2 * SIZEOF_INT(ia_aac_dec_channel_info_struct) +
 | |
|                               (ch * SIZEOF_INT(ia_aac_sfb_code_book_struct))];
 | |
| 
 | |
|     aac_dec_handle->pstr_aac_dec_ch_info[ch]->ptr_scale_factor =
 | |
|         ptr_aac_sfb_code_book_data[ch]->scale_factor;
 | |
|     aac_dec_handle->pstr_aac_dec_ch_info[ch]->ptr_code_book =
 | |
|         ptr_aac_sfb_code_book_data[ch]->code_book;
 | |
| 
 | |
|     aac_dec_handle->pstr_aac_dec_ch_info[ch]->ptr_spec_coeff =
 | |
|         &work_buffer_core[ch * MAX_BINS_LONG];
 | |
| 
 | |
|     if (object_type == AOT_ER_AAC_ELD) {
 | |
|       aac_dec_handle->pstr_aac_dec_ch_info[ch]->ptr_spec_coeff =
 | |
|           &work_buffer_core[2 * ch * MAX_BINS_LONG];
 | |
|     }
 | |
| 
 | |
|     aac_dec_handle->pstr_aac_dec_ch_info[ch]->pstr_stereo_info =
 | |
|         &ptr_pns_stereo_data->str_stereo_info;
 | |
|     aac_dec_handle->pstr_aac_dec_ch_info[ch]->pstr_pns_corr_info =
 | |
|         &ptr_pns_stereo_data->str_pns_corr_info;
 | |
|     aac_dec_handle->pstr_aac_dec_ch_info[ch]->pstr_pns_rand_vec_data =
 | |
|         aac_dec_handle->pstr_pns_rand_vec_data;
 | |
| 
 | |
|     pstr_imdct_tables = aac_dec_handle->pstr_aac_tables->pstr_imdct_tables;
 | |
| 
 | |
|     aac_dec_handle->pstr_aac_dec_overlap_info[ch]->ptr_long_window[0] =
 | |
|         pstr_imdct_tables->only_long_window_sine;
 | |
|     aac_dec_handle->pstr_aac_dec_overlap_info[ch]->ptr_short_window[0] =
 | |
|         pstr_imdct_tables->only_short_window_sine;
 | |
|     aac_dec_handle->pstr_aac_dec_overlap_info[ch]->ptr_long_window[1] =
 | |
|         pstr_imdct_tables->only_long_window_kbd;
 | |
|     aac_dec_handle->pstr_aac_dec_overlap_info[ch]->ptr_short_window[1] =
 | |
|         pstr_imdct_tables->only_short_window_kbd;
 | |
| 
 | |
|     aac_dec_handle->ptr_aac_dec_static_channel_info[ch]->ptr_long_window[0] =
 | |
|         pstr_imdct_tables->only_long_window_sine;
 | |
|     aac_dec_handle->ptr_aac_dec_static_channel_info[ch]->ptr_short_window[0] =
 | |
|         pstr_imdct_tables->only_short_window_sine;
 | |
|     aac_dec_handle->ptr_aac_dec_static_channel_info[ch]->ptr_long_window[1] =
 | |
|         pstr_imdct_tables->only_long_window_kbd;
 | |
|     aac_dec_handle->ptr_aac_dec_static_channel_info[ch]->ptr_short_window[1] =
 | |
|         pstr_imdct_tables->only_short_window_kbd;
 | |
| 
 | |
|     if (object_type == AOT_ER_AAC_ELD || object_type == AOT_ER_AAC_LD ||
 | |
|         object_type == AOT_AAC_LTP) {
 | |
|       aac_dec_handle->pstr_aac_dec_ch_info[ch]->str_ics_info.frame_length =
 | |
|           frame_length;
 | |
| 
 | |
|       if (512 == aac_dec_handle->samples_per_frame) {
 | |
|         if (object_type != AOT_ER_AAC_ELD) {
 | |
|           aac_dec_handle->pstr_aac_dec_overlap_info[ch]->ptr_long_window[1] =
 | |
|               (WORD16 *)pstr_imdct_tables->low_overlap_win;
 | |
|           aac_dec_handle->pstr_aac_dec_overlap_info[ch]->ptr_long_window[0] =
 | |
|               (WORD16 *)pstr_imdct_tables->window_sine_512;
 | |
| 
 | |
|           aac_dec_handle->ptr_aac_dec_static_channel_info[ch]
 | |
|               ->ptr_long_window[1] =
 | |
|               (WORD16 *)pstr_imdct_tables->low_overlap_win;
 | |
|           aac_dec_handle->ptr_aac_dec_static_channel_info[ch]
 | |
|               ->ptr_long_window[0] =
 | |
|               (WORD16 *)pstr_imdct_tables->window_sine_512;
 | |
|         } else {
 | |
|           aac_dec_handle->pstr_aac_dec_overlap_info[ch]->ptr_long_window[1] =
 | |
|               (WORD16 *)pstr_imdct_tables->window_sine_512_eld;
 | |
|           aac_dec_handle->pstr_aac_dec_overlap_info[ch]->ptr_long_window[0] =
 | |
|               (WORD16 *)pstr_imdct_tables->window_sine_512_eld;
 | |
|         }
 | |
|       } else if (480 == aac_dec_handle->samples_per_frame) {
 | |
|         if (object_type != AOT_ER_AAC_ELD) {
 | |
|           aac_dec_handle->pstr_aac_dec_overlap_info[ch]->ptr_long_window[1] =
 | |
|               (WORD16 *)pstr_imdct_tables->low_overlap_win_480;
 | |
|           aac_dec_handle->pstr_aac_dec_overlap_info[ch]->ptr_long_window[0] =
 | |
|               (WORD16 *)pstr_imdct_tables->window_sine_480;
 | |
| 
 | |
|           aac_dec_handle->ptr_aac_dec_static_channel_info[ch]
 | |
|               ->ptr_long_window[1] =
 | |
|               (WORD16 *)pstr_imdct_tables->low_overlap_win_480;
 | |
|           aac_dec_handle->ptr_aac_dec_static_channel_info[ch]
 | |
|               ->ptr_long_window[0] =
 | |
|               (WORD16 *)pstr_imdct_tables->window_sine_480;
 | |
| 
 | |
|         } else {
 | |
|           aac_dec_handle->pstr_aac_dec_overlap_info[ch]->ptr_long_window[1] =
 | |
|               (WORD16 *)pstr_imdct_tables->window_sine_480_eld;
 | |
|           aac_dec_handle->pstr_aac_dec_overlap_info[ch]->ptr_long_window[0] =
 | |
|               (WORD16 *)pstr_imdct_tables->window_sine_480_eld;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|     if ((object_type == AOT_ER_AAC_LD) || (object_type == AOT_AAC_LTP)) {
 | |
|       if (aac_dec_handle->samples_per_frame <= 512) {
 | |
|         aac_dec_handle->pstr_aac_dec_ch_info[ch]->str_ics_info.ltp.lag =
 | |
|             aac_dec_handle->ptr_aac_dec_static_channel_info[ch]->ltp_lag;
 | |
|       }
 | |
|       aac_dec_handle->pstr_aac_dec_ch_info[ch]->ltp_buf =
 | |
|           aac_dec_handle->ptr_aac_dec_static_channel_info[ch]->ltp_buf;
 | |
|       aac_dec_handle->pstr_aac_dec_ch_info[ch]->ltp_lag =
 | |
|           aac_dec_handle->ptr_aac_dec_static_channel_info[ch]->ltp_lag;
 | |
|     }
 | |
| 
 | |
|     aac_dec_handle->pstr_aac_dec_ch_info[ch]->scratch_buf_ptr = work_buffer_2;
 | |
|     if (object_type == AOT_ER_AAC_ELD) {
 | |
|       aac_dec_handle->pstr_aac_dec_ch_info[ch]->pulse_scratch =
 | |
|           aac_scratch_ptrs->extra_scr_4k[3];
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (channel == 2) {
 | |
|     if (aac_dec_handle->pstr_aac_dec_ch_info[1]->ptr_spec_coeff ==
 | |
|         aac_scratch_ptrs->extra_scr_4k[0]) {
 | |
|       aac_dec_handle->pstr_aac_dec_ch_info[1]->ptr_spec_coeff =
 | |
|           aac_dec_handle->pstr_aac_dec_ch_info[0]->ptr_spec_coeff;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   for (ch = 0; ch < channel; ch++) {
 | |
|     ia_pns_info_struct *ptr_pns_info =
 | |
|         &aac_dec_handle->pstr_aac_dec_ch_info[ch]->str_pns_info;
 | |
|     memset(ptr_pns_info, 0, sizeof(ia_pns_info_struct));
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     ia_pns_correlation_info_struct *ptr_corr_info =
 | |
|         aac_dec_handle->pstr_aac_dec_ch_info[0]->pstr_pns_corr_info;
 | |
|     memset(ptr_corr_info->correlated, 0, sizeof(UWORD8) * PNS_BAND_FLAGS_SIZE);
 | |
|   }
 | |
| 
 | |
|   for (ch = 0; ch < channel; ch++) {
 | |
|     memset(&aac_dec_handle->pstr_aac_dec_ch_info[ch]->str_hcr_info, 0,
 | |
|            sizeof(ia_hcr_info_struct));
 | |
|     ixheaacd_huff_code_reorder_tbl_init(
 | |
|         &aac_dec_handle->pstr_aac_dec_ch_info[ch]->str_hcr_info);
 | |
|   }
 | |
| 
 | |
|   for (ch = 0; ch < channel; ch++) {
 | |
|     aac_dec_handle->pstr_aac_dec_ch_info[ch]->str_ics_info.ltp.data_present = 0;
 | |
|     aac_dec_handle->pstr_aac_dec_ch_info[ch]->str_ics_info.ltp2.data_present =
 | |
|         0;
 | |
|   }
 | |
| 
 | |
|   previous_element = ID_END;
 | |
| 
 | |
|   aac_dec_handle->pstr_sbr_bitstream->no_elements = 0;
 | |
|   new_element = 0;
 | |
|   ele_type = *type;
 | |
| 
 | |
|   cnt_bits = it_bit_buff->cnt_bits;
 | |
| 
 | |
|   if (((object_type != AOT_ER_AAC_ELD) && (object_type != AOT_ER_AAC_LD)) ||
 | |
|       (object_type < ER_OBJECT_START)) {
 | |
|     while (ele_type != ID_END && aac_dec_handle->frame_status) {
 | |
|       ele_type = (WORD)ixheaacd_read_bits_buf(it_bit_buff, 3);
 | |
|       ixheaacd_read_bidirection(it_bit_buff, -3);
 | |
| 
 | |
|       if (it_bit_buff->cnt_bits < 3) {
 | |
|         it_bit_buff->cnt_bits = -1;
 | |
|         error_code = (WORD16)(
 | |
|             (WORD32)IA_ENHAACPLUS_DEC_EXE_NONFATAL_INSUFFICIENT_INPUT_BYTES);
 | |
|         break;
 | |
|       }
 | |
| 
 | |
|       if ((ele_type == ID_FIL) || (ele_type == ID_DSE) || (new_element == 0)) {
 | |
|         ele_type = (WORD)ixheaacd_read_bits_buf(it_bit_buff, 3);
 | |
|         new_element = 1;
 | |
|       } else if ((ele_type != ID_END)) {
 | |
|         ele_type = -1;
 | |
|         break;
 | |
|       } else {
 | |
|         ele_type = (WORD)ixheaacd_read_bits_buf(it_bit_buff, 3);
 | |
|       }
 | |
| 
 | |
|       if (it_bit_buff->cnt_bits < 0) {
 | |
|         aac_dec_handle->frame_status = 0;
 | |
|       }
 | |
| 
 | |
|       switch (ele_type) {
 | |
|         case ID_SCE:
 | |
|         case ID_CPE:
 | |
|         case ID_LFE:
 | |
| 
 | |
|           if (aac_dec_handle->frame_status) {
 | |
|             ia_aac_dec_channel_info_struct *pstr_aac_dec_ch_info =
 | |
|                 aac_dec_handle->pstr_aac_dec_ch_info[LEFT];
 | |
|             ia_ics_info_struct *ptr_ics_info =
 | |
|                 &pstr_aac_dec_ch_info->str_ics_info;
 | |
|             ele_ch = 1;
 | |
|             if (ele_type == ID_CPE) {
 | |
|               ele_ch = 2;
 | |
|             } else {
 | |
|               ele_ch = 1;
 | |
|             }
 | |
| 
 | |
|             prev_data_ele_present = 1;
 | |
| 
 | |
|             if (ptr_adts_crc_info->crc_active == 1 &&
 | |
|                 ptr_adts_crc_info->no_reg < 7) {
 | |
|               crc_reg = ixheaacd_adts_crc_start_reg(
 | |
|                   ptr_adts_crc_info, it_bit_buff, CRC_ADTS_RAW_DATA_BLK_LEN);
 | |
|             }
 | |
| 
 | |
|             pstr_aac_dec_ch_info->element_instance_tag =
 | |
|                 (WORD16)ixheaacd_read_bits_buf(it_bit_buff, 4);
 | |
| 
 | |
|             element_index_order[*ch_idx] =
 | |
|                 pstr_aac_dec_ch_info->element_instance_tag;
 | |
|             pstr_aac_dec_ch_info->common_window = 0;
 | |
| 
 | |
|             ptr_ics_info->num_swb_window = 0;
 | |
|             ptr_ics_info->sampling_rate_index =
 | |
|                 aac_dec_handle->sampling_rate_index;
 | |
|             if ((object_type == AOT_ER_AAC_LD) ||
 | |
|                 (object_type == AOT_AAC_LTP)) {
 | |
|               ptr_ics_info->ltp.data_present = 0;
 | |
|               ptr_ics_info->ltp2.data_present = 0;
 | |
|               ptr_ics_info->predictor_data_present = 0;
 | |
|             }
 | |
| 
 | |
|             if (ele_ch > 1) {
 | |
|               aac_dec_handle->pstr_aac_dec_ch_info[RIGHT]
 | |
|                   ->str_ics_info.num_swb_window = 0;
 | |
|               aac_dec_handle->pstr_aac_dec_ch_info[RIGHT]
 | |
|                   ->str_ics_info.sampling_rate_index =
 | |
|                   aac_dec_handle->sampling_rate_index;
 | |
| 
 | |
|               pstr_aac_dec_ch_info->common_window =
 | |
|                   (WORD16)ixheaacd_read_bits_buf(it_bit_buff, 1);
 | |
| 
 | |
|               if (pstr_aac_dec_ch_info->common_window) {
 | |
|                 error_code = ixheaacd_ics_read(
 | |
|                     it_bit_buff, ptr_ics_info, aac_dec_handle->num_swb_window,
 | |
|                     object_type, pstr_aac_dec_ch_info->common_window,
 | |
|                     aac_dec_handle->samples_per_frame);
 | |
|                 if (error_code) {
 | |
|                   if (it_bit_buff->cnt_bits < 0) {
 | |
|                     error_code = (WORD16)(
 | |
|                         (WORD32)
 | |
|                             IA_ENHAACPLUS_DEC_EXE_NONFATAL_INSUFFICIENT_INPUT_BYTES);
 | |
|                   }
 | |
| 
 | |
|                   goto _ia_handle_error;
 | |
|                 }
 | |
| 
 | |
|                 aac_dec_handle->pstr_aac_dec_ch_info[RIGHT]->str_ics_info =
 | |
|                     pstr_aac_dec_ch_info->str_ics_info;
 | |
| 
 | |
|                 ixheaacd_read_ms_data(it_bit_buff, pstr_aac_dec_ch_info);
 | |
|               }
 | |
|             }
 | |
| 
 | |
|             error_code = ixheaacd_individual_ch_stream(
 | |
|                 it_bit_buff, aac_dec_handle, ele_ch, frame_length,
 | |
|                 total_channels, object_type, eld_specific_config, ele_type);
 | |
|             if (error_code) return error_code;
 | |
| 
 | |
|             if (ptr_adts_crc_info->crc_active == 1) {
 | |
|               ixheaacd_adts_crc_end_reg(ptr_adts_crc_info, it_bit_buff,
 | |
|                                         crc_reg);
 | |
|             }
 | |
| 
 | |
|             if (it_bit_buff->cnt_bits < 0) {
 | |
|               error_code = (WORD16)(
 | |
|                   (WORD32)
 | |
|                       IA_ENHAACPLUS_DEC_EXE_NONFATAL_INSUFFICIENT_INPUT_BYTES);
 | |
|             }
 | |
| 
 | |
|             if (error_code) {
 | |
|               goto _ia_handle_error;
 | |
|             }
 | |
| 
 | |
|           _ia_handle_error:
 | |
|             if (error_code) {
 | |
|               aac_dec_handle->frame_status = 0;
 | |
|               if ((ele_type >= ID_SCE) && (ele_type <= ID_LFE))
 | |
|                 num_ch = num_ch + ele_ch;
 | |
|               break;
 | |
|             } else {
 | |
|               error_code = ixheaacd_channel_pair_process(
 | |
|                   aac_dec_handle->pstr_aac_dec_ch_info, ele_ch,
 | |
|                   aac_dec_handle->pstr_aac_tables, total_channels, object_type,
 | |
|                   aac_spect_data_resil_flag,
 | |
|                   eld_specific_config.aac_sf_data_resil_flag,
 | |
|                   aac_scratch_ptrs->in_data, aac_scratch_ptrs->out_data,
 | |
|                   (void *)aac_dec_handle);
 | |
|               if (error_code) return error_code;
 | |
|               num_ch = num_ch + ele_ch;
 | |
|             }
 | |
|           }
 | |
| 
 | |
|           break;
 | |
|         case ID_CCE:
 | |
|           if (max_channels > 2) {
 | |
|             prev_data_ele_present = 1;
 | |
|             error_code = ixheaacd_dec_coupling_channel_element(
 | |
|                 it_bit_buff, aac_dec_handle,
 | |
|                 aac_dec_handle->sampling_rate_index,
 | |
|                 aac_dec_handle->pstr_aac_tables,
 | |
|                 aac_dec_handle->pstr_common_tables,
 | |
|                 &element_index_order[*ch_idx],
 | |
|                 (ia_enhaacplus_dec_ind_cc *)aac_dec_handle->p_ind_channel_info,
 | |
|                 total_channels, frame_length, object_type, eld_specific_config,
 | |
|                 ele_type);
 | |
| 
 | |
|             num_ch = num_ch + 1;
 | |
| 
 | |
|             if (error_code) {
 | |
|               aac_dec_handle->frame_status = 0;
 | |
|               return error_code;
 | |
|             } else {
 | |
|               error_code = ixheaacd_channel_pair_process(
 | |
|                   aac_dec_handle->pstr_aac_dec_ch_info, 1,
 | |
|                   aac_dec_handle->pstr_aac_tables, total_channels, object_type,
 | |
|                   aac_spect_data_resil_flag,
 | |
|                   eld_specific_config.aac_sf_data_resil_flag,
 | |
|                   aac_scratch_ptrs->in_data, aac_scratch_ptrs->out_data,
 | |
|                   (void *)aac_dec_handle);
 | |
|               if (error_code) return error_code;
 | |
|             }
 | |
|           } else {
 | |
|             error_code =
 | |
|                 (WORD32)((WORD32)IA_ENHAACPLUS_DEC_EXE_FATAL_UNIMPLEMENTED_CCE);
 | |
|           }
 | |
|           if (it_bit_buff->cnt_bits < 0) {
 | |
|             error_code = (WORD16)((
 | |
|                 WORD32)IA_ENHAACPLUS_DEC_EXE_NONFATAL_INSUFFICIENT_INPUT_BYTES);
 | |
|             goto _ia_handle_error;
 | |
|           }
 | |
|           break;
 | |
| 
 | |
|         case ID_DSE:
 | |
|         case ID_PCE:
 | |
|         case ID_FIL:
 | |
| 
 | |
|         {
 | |
|           WORD32 flag = 1;
 | |
| 
 | |
|           if ((ele_type != ID_FIL) && (ptr_adts_crc_info->crc_active == 1) &&
 | |
|               (ptr_adts_crc_info->no_reg < 7)) {
 | |
|             crc_reg =
 | |
|                 ixheaacd_adts_crc_start_reg(ptr_adts_crc_info, it_bit_buff, 0);
 | |
|           }
 | |
|           if (ele_type == ID_DSE) {
 | |
|             ixheaacd_read_data_stream_element(
 | |
|                 it_bit_buff, &aac_dec_handle->byte_align_bits,
 | |
|                 p_obj_exhaacplus_dec->p_state_aac->pstr_drc_dec);
 | |
|           }
 | |
| 
 | |
|           else if (ele_type == ID_PCE) {
 | |
|             error_code = ixheaacd_decode_pce(
 | |
|                 it_bit_buff,
 | |
|                 &p_obj_exhaacplus_dec->aac_config.ui_pce_found_in_hdr,
 | |
|                 &p_obj_exhaacplus_dec->aac_config.str_prog_config);
 | |
|             if (error_code != 0) {
 | |
|               if (it_bit_buff->cnt_bits < 0) {
 | |
|                 error_code = (WORD16)(
 | |
|                     (WORD32)
 | |
|                         IA_ENHAACPLUS_DEC_EXE_NONFATAL_INSUFFICIENT_INPUT_BYTES);
 | |
|                 goto _ia_handle_error;
 | |
|               }
 | |
|               aac_dec_handle->frame_status = 0;
 | |
|               if (error_code > 0) {
 | |
|                 error_code = IA_ENHAACPLUS_DEC_EXE_NONFATAL_DECODE_FRAME_ERROR;
 | |
|                 return error_code;
 | |
|               } else {
 | |
|                 return error_code;
 | |
|               }
 | |
|             }
 | |
|           }
 | |
| 
 | |
|           else if (ele_type == ID_FIL) {
 | |
|             WORD32 bits_decoded = 0;
 | |
|             if (object_type == AOT_ER_AAC_ELD) {
 | |
|               bits_decoded = (it_bit_buff->size - it_bit_buff->cnt_bits);
 | |
|               cnt_bits = (frame_size * 8 - bits_decoded);
 | |
|               if (adtsheader == 1) {
 | |
|                 if (cnt_bits > it_bit_buff->cnt_bits)
 | |
|                   return IA_ENHAACPLUS_DEC_EXE_NONFATAL_INSUFFICIENT_INPUT_BYTES;
 | |
|               }
 | |
|             }
 | |
| 
 | |
|             if (ixheaacd_check_for_sbr_payload(
 | |
|                     it_bit_buff, aac_dec_handle->pstr_sbr_bitstream,
 | |
|                     (WORD16)previous_element, pstr_drc_dec, object_type,
 | |
|                     adtsheader, cnt_bits, ld_sbr_crc_flag, drc_dummy)) {
 | |
|               flag = 0;
 | |
|             }
 | |
|           }
 | |
| 
 | |
|           if (it_bit_buff->cnt_bits < 0) {
 | |
|             error_code = (WORD16)((
 | |
|                 WORD32)IA_ENHAACPLUS_DEC_EXE_NONFATAL_INSUFFICIENT_INPUT_BYTES);
 | |
|             goto _ia_handle_error;
 | |
|           }
 | |
| 
 | |
|           if (flag) {
 | |
|             if (prev_data_ele_present == 0) {
 | |
|               new_element = 0;
 | |
|             }
 | |
|           }
 | |
|           if ((ele_type != ID_FIL) && (ptr_adts_crc_info->crc_active == 1)) {
 | |
|             ixheaacd_adts_crc_end_reg(ptr_adts_crc_info, it_bit_buff, crc_reg);
 | |
|           }
 | |
| 
 | |
|           if (ele_type == ID_PCE) {
 | |
|             if (ptr_adts_crc_info->str_crc_reg_data[crc_reg].bit_buf_cnt) {
 | |
|               ptr_adts_crc_info->str_crc_reg_data[crc_reg].max_bits =
 | |
|                   ptr_adts_crc_info->str_crc_reg_data[crc_reg].bit_buf_cnt;
 | |
|             }
 | |
|           }
 | |
|         }
 | |
| 
 | |
|         break;
 | |
| 
 | |
|         case ID_END:
 | |
|           error_code = 0;
 | |
|           break;
 | |
|       }
 | |
| 
 | |
|       previous_element = ele_type;
 | |
| 
 | |
|       if (init_flag) {
 | |
|         if ((ele_type >= ID_SCE) && (ele_type <= ID_LFE)) {
 | |
|           p_obj_exhaacplus_dec->aac_config.element_type[*ch_idx] = ele_type;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   } else {
 | |
|     {
 | |
|       switch (ch_config) {
 | |
|         default:
 | |
|           if (aac_dec_handle->frame_status) {
 | |
|             ia_aac_dec_channel_info_struct *pstr_aac_dec_ch_info =
 | |
|                 aac_dec_handle->pstr_aac_dec_ch_info[LEFT];
 | |
|             ia_ics_info_struct *ptr_ics_info =
 | |
|                 &pstr_aac_dec_ch_info->str_ics_info;
 | |
| 
 | |
|             if (ch_config == 2)
 | |
|               ele_ch = 2, ele_type = 1;
 | |
|             else
 | |
|               ele_ch = 1, ele_type = 0;
 | |
| 
 | |
|             prev_data_ele_present = 1;
 | |
| 
 | |
|             if ((ptr_adts_crc_info->crc_active == 1) &&
 | |
|                 (ptr_adts_crc_info->no_reg < 7)) {
 | |
|               crc_reg = ixheaacd_adts_crc_start_reg(
 | |
|                   ptr_adts_crc_info, it_bit_buff, CRC_ADTS_RAW_DATA_BLK_LEN);
 | |
|             }
 | |
| 
 | |
|             if (object_type != AOT_ER_AAC_ELD)
 | |
|               pstr_aac_dec_ch_info->element_instance_tag =
 | |
|                   (WORD16)ixheaacd_read_bits_buf(it_bit_buff, 4);
 | |
| 
 | |
|             element_index_order[*ch_idx] =
 | |
|                 pstr_aac_dec_ch_info->element_instance_tag;
 | |
|             pstr_aac_dec_ch_info->common_window = 0;
 | |
| 
 | |
|             ptr_ics_info->num_swb_window = 0;
 | |
|             ptr_ics_info->sampling_rate_index =
 | |
|                 aac_dec_handle->sampling_rate_index;
 | |
| 
 | |
|             if (object_type == AOT_ER_AAC_LD) {
 | |
|               ptr_ics_info->ltp.data_present = 0;
 | |
|               ptr_ics_info->ltp2.data_present = 0;
 | |
|               ptr_ics_info->predictor_data_present = 0;
 | |
|             }
 | |
|             if (ele_ch > 1) {
 | |
|               aac_dec_handle->pstr_aac_dec_ch_info[RIGHT]
 | |
|                   ->str_ics_info.num_swb_window = 0;
 | |
|               aac_dec_handle->pstr_aac_dec_ch_info[RIGHT]
 | |
|                   ->str_ics_info.sampling_rate_index =
 | |
|                   aac_dec_handle->sampling_rate_index;
 | |
| 
 | |
|               if (object_type != 39)
 | |
|                 pstr_aac_dec_ch_info->common_window =
 | |
|                     (WORD16)ixheaacd_read_bits_buf(it_bit_buff, 1);
 | |
|               else
 | |
|                 pstr_aac_dec_ch_info->common_window = 1;
 | |
| 
 | |
|               if (pstr_aac_dec_ch_info->common_window) {
 | |
|                 error_code = ixheaacd_ics_read(
 | |
|                     it_bit_buff, ptr_ics_info, aac_dec_handle->num_swb_window,
 | |
|                     object_type, pstr_aac_dec_ch_info->common_window,
 | |
|                     aac_dec_handle->samples_per_frame);
 | |
|                 if (error_code) {
 | |
|                   if (it_bit_buff->cnt_bits < 0) {
 | |
|                     error_code = (WORD16)(
 | |
|                         (WORD32)
 | |
|                             IA_ENHAACPLUS_DEC_EXE_NONFATAL_INSUFFICIENT_INPUT_BYTES);
 | |
|                   }
 | |
| 
 | |
|                   goto _ia_handle_error1;
 | |
|                 }
 | |
| 
 | |
|                 aac_dec_handle->pstr_aac_dec_ch_info[RIGHT]->str_ics_info =
 | |
|                     pstr_aac_dec_ch_info->str_ics_info;
 | |
| 
 | |
|                 ixheaacd_read_ms_data(it_bit_buff, pstr_aac_dec_ch_info);
 | |
| 
 | |
|                 {
 | |
|                   if (object_type == AOT_ER_AAC_LD) {
 | |
|                     IA_ERRORCODE temp = ixheaacd_ltp_decode(
 | |
|                         it_bit_buff, ptr_ics_info, object_type,
 | |
|                         aac_dec_handle->samples_per_frame, LEFT);
 | |
| 
 | |
|                     if (temp != 0) {
 | |
|                       return temp;
 | |
|                     }
 | |
|                   }
 | |
|                 }
 | |
|               }
 | |
|             }
 | |
| 
 | |
|             error_code = ixheaacd_individual_ch_stream(
 | |
|                 it_bit_buff, aac_dec_handle, ele_ch, frame_length,
 | |
|                 total_channels, object_type, eld_specific_config, ele_type);
 | |
|             if (error_code) return error_code;
 | |
| 
 | |
|             if (ptr_adts_crc_info->crc_active == 1) {
 | |
|               ixheaacd_adts_crc_end_reg(ptr_adts_crc_info, it_bit_buff,
 | |
|                                         crc_reg);
 | |
|             }
 | |
| 
 | |
|             if (it_bit_buff->cnt_bits < 0) {
 | |
|               error_code = (WORD16)(
 | |
|                   (WORD32)
 | |
|                       IA_ENHAACPLUS_DEC_EXE_NONFATAL_INSUFFICIENT_INPUT_BYTES);
 | |
|             }
 | |
| 
 | |
|             if (error_code) {
 | |
|               goto _ia_handle_error1;
 | |
|             }
 | |
| 
 | |
|           _ia_handle_error1:
 | |
|             if (error_code) {
 | |
|               aac_dec_handle->frame_status = 0;
 | |
|               if ((ele_type >= ID_SCE) && (ele_type <= ID_LFE))
 | |
|                 num_ch = num_ch + ele_ch;
 | |
|               break;
 | |
|             } else {
 | |
|               error_code = ixheaacd_channel_pair_process(
 | |
|                   aac_dec_handle->pstr_aac_dec_ch_info, ele_ch,
 | |
|                   aac_dec_handle->pstr_aac_tables, total_channels, object_type,
 | |
|                   aac_spect_data_resil_flag,
 | |
|                   eld_specific_config.aac_sf_data_resil_flag,
 | |
|                   aac_scratch_ptrs->in_data, aac_scratch_ptrs->out_data,
 | |
|                   (void *)aac_dec_handle);
 | |
|               if (error_code) return error_code;
 | |
|               num_ch = num_ch + ele_ch;
 | |
|             }
 | |
|           }
 | |
| 
 | |
|           p_obj_exhaacplus_dec->aac_config.element_type[*ch_idx] = ele_ch - 1;
 | |
|           break;
 | |
|       }
 | |
| 
 | |
|       if ((object_type != AOT_ER_AAC_ELD) || (!eld_sbr_flag)) {
 | |
|         WORD32 bits_decoded, cnt_bits;
 | |
|         bits_decoded = (it_bit_buff->size - it_bit_buff->cnt_bits);
 | |
| 
 | |
|         cnt_bits = (frame_size * 8 - bits_decoded);
 | |
|         if (cnt_bits >= 8) {
 | |
|           error_code = ixheaacd_extension_payload(it_bit_buff, cnt_bits);
 | |
|           if (error_code) return error_code;
 | |
|         }
 | |
| 
 | |
|         if (((object_type == AOT_ER_AAC_ELD) ||
 | |
|              (object_type == AOT_ER_AAC_LD)) &&
 | |
|             (p_obj_exhaacplus_dec->aac_config.ld_decoder != 1)) {
 | |
|           if (it_bit_buff->cnt_bits) {
 | |
|             WORD32 alignment = it_bit_buff->bit_pos & 0x07;
 | |
|             it_bit_buff->cnt_bits = (it_bit_buff->cnt_bits + alignment) & 7;
 | |
|             it_bit_buff->bit_pos = 7;
 | |
|             it_bit_buff->ptr_read_next++;
 | |
|           }
 | |
|         } else {
 | |
|           if (it_bit_buff->bit_pos != 7) {
 | |
|             WORD32 alignment = it_bit_buff->bit_pos & 0x07;
 | |
|             it_bit_buff->cnt_bits -= alignment + 1;
 | |
|             it_bit_buff->bit_pos += 7 - alignment;
 | |
|             it_bit_buff->ptr_read_next++;
 | |
|           }
 | |
|         }
 | |
|       } else {
 | |
|         WORD32 bits_decoded, cnt_bits;
 | |
|         bits_decoded = (it_bit_buff->size - it_bit_buff->cnt_bits);
 | |
|         cnt_bits = (frame_size * 8 - bits_decoded);
 | |
|         if (adtsheader == 1) {
 | |
|           if (cnt_bits > it_bit_buff->cnt_bits)
 | |
|             return IA_ENHAACPLUS_DEC_EXE_NONFATAL_INSUFFICIENT_INPUT_BYTES;
 | |
|         }
 | |
|         ixheaacd_check_for_sbr_payload(
 | |
|             it_bit_buff, aac_dec_handle->pstr_sbr_bitstream,
 | |
|             (WORD16)(ch_config - 1), pstr_drc_dec, object_type, adtsheader,
 | |
|             cnt_bits, ld_sbr_crc_flag, drc_dummy);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (ele_type == ID_END &&
 | |
|       p_state_enhaacplus_dec->bs_format == LOAS_BSFORMAT) {
 | |
|     WORD32 tmp;
 | |
|     tmp = ((WORD32)latm_element->layer_info[0][0].frame_len_bits) -
 | |
|           (it_bit_buff->initial_cnt_bits - it_bit_buff->cnt_bits);
 | |
| 
 | |
|     if (tmp > 0) ixheaacd_read_bidirection(it_bit_buff, tmp);
 | |
| 
 | |
|     if (latm_element->other_data_present) {
 | |
|       WORD32 count_bits = (WORD32)latm_element->other_data_length;
 | |
|       ixheaacd_read_bidirection(it_bit_buff, count_bits);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (object_type == AOT_ER_AAC_LD) {
 | |
|     for (ch = 0; ch < channel; ch++) {
 | |
|       aac_dec_handle->ptr_aac_dec_static_channel_info[ch]->ltp_lag =
 | |
|           aac_dec_handle->pstr_aac_dec_ch_info[ch]->str_ics_info.ltp.lag;
 | |
|     }
 | |
|   }
 | |
|   aac_dec_handle->frame_status = aac_dec_handle->frame_status && frame_status;
 | |
| 
 | |
|   aac_dec_handle->channels = num_ch;
 | |
| 
 | |
|   if (error_code == 0)
 | |
|     if ((skip_full_decode == 0) || ((skip_full_decode == 1) && error_code)) {
 | |
|       ia_ics_info_struct str_ics_info[2];
 | |
|       WORD32 *spec_coef[2];
 | |
|       WORD32 *scratch[2];
 | |
| 
 | |
|       for (ch = 0; ch < channel; ch++) {
 | |
|         str_ics_info[ch] =
 | |
|             aac_dec_handle->pstr_aac_dec_ch_info[ch]->str_ics_info;
 | |
|         spec_coef[ch] =
 | |
|             aac_dec_handle->pstr_aac_dec_ch_info[ch]->ptr_spec_coeff;
 | |
|       }
 | |
| 
 | |
|       scratch[0] = (WORD32 *)aac_scratch_ptrs->extra_scr_4k[2];
 | |
|       scratch[1] = (WORD32 *)aac_scratch_ptrs->extra_scr_4k[1];
 | |
| 
 | |
|       error_code = ixheaacd_drc_map_channels(
 | |
|           pstr_drc_dec, aac_dec_handle->channels,
 | |
|           aac_dec_handle->pstr_aac_dec_ch_info[0]->str_ics_info.frame_length);
 | |
|       if (error_code) return error_code;
 | |
| 
 | |
|       for (ch = 0; ch < aac_dec_handle->channels; ch++) {
 | |
|         WORD32 *overlap1 = aac_dec_handle->ptr_aac_dec_static_channel_info[ch]
 | |
|                                ->overlap_add_data.ptr_overlap_buf;
 | |
|         const WORD16 *ptr_long_window_next =
 | |
|             aac_dec_handle->ptr_aac_dec_static_channel_info[ch]
 | |
|                 ->ptr_long_window[(int)str_ics_info[ch].window_shape];
 | |
|         const WORD16 *ptr_short_window_next =
 | |
|             aac_dec_handle->ptr_aac_dec_static_channel_info[ch]
 | |
|                 ->ptr_short_window[(int)str_ics_info[ch].window_shape];
 | |
|         if (pstr_drc_dec->drc_on) {
 | |
|           ixheaacd_drc_apply(pstr_drc_dec, spec_coef[ch],
 | |
|                              str_ics_info[ch].window_sequence, ch,
 | |
|                              str_ics_info[ch].frame_length);
 | |
|         }
 | |
|         if (skip_full_decode == 0) {
 | |
|           ixheaacd_imdct_process(
 | |
|               aac_dec_handle->pstr_aac_dec_overlap_info[ch], spec_coef[ch],
 | |
|               &str_ics_info[ch], time_data + slot_element, (WORD16)ch_fac,
 | |
|               scratch[ch], aac_dec_handle->pstr_aac_tables, object_type);
 | |
| 
 | |
|           aac_dec_handle->ptr_aac_dec_static_channel_info[ch]
 | |
|               ->overlap_add_data.win_shape = str_ics_info[ch].window_shape;
 | |
|           aac_dec_handle->ptr_aac_dec_static_channel_info[ch]
 | |
|               ->overlap_add_data.win_seq = str_ics_info[ch].window_sequence;
 | |
|           if ((object_type == AOT_ER_AAC_LD) || (object_type == AOT_AAC_LTP)) {
 | |
|             {
 | |
|               if ((str_ics_info[ch].window_sequence == ONLY_LONG_SEQUENCE) ||
 | |
|                   (str_ics_info[ch].window_sequence == LONG_STOP_SEQUENCE)) {
 | |
|                 ixheaacd_lt_update_state(
 | |
|                     aac_dec_handle->ptr_aac_dec_static_channel_info[ch]
 | |
|                         ->ltp_buf,
 | |
|                     time_data + slot_element, overlap1,
 | |
|                     aac_dec_handle->samples_per_frame, object_type,
 | |
|                     (WORD16)ch_fac, str_ics_info[ch].window_sequence,
 | |
|                     (WORD16 *)ptr_long_window_next);
 | |
|               } else {
 | |
|                 ixheaacd_lt_update_state(
 | |
|                     aac_dec_handle->ptr_aac_dec_static_channel_info[ch]
 | |
|                         ->ltp_buf,
 | |
|                     time_data + slot_element, overlap1,
 | |
|                     aac_dec_handle->samples_per_frame, object_type,
 | |
|                     (WORD16)ch_fac, str_ics_info[ch].window_sequence,
 | |
|                     (WORD16 *)ptr_short_window_next);
 | |
|               }
 | |
|             }
 | |
|           }
 | |
|           slot_element++;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|   if (ele_type == ID_END) {
 | |
|     ixheaacd_byte_align(it_bit_buff, &aac_dec_handle->byte_align_bits);
 | |
|     if (p_state_enhaacplus_dec->bs_format == LOAS_BSFORMAT) {
 | |
|       ixheaacd_byte_align(it_bit_buff, &it_bit_buff->audio_mux_align);
 | |
|     }
 | |
|   }
 | |
|   *type = ele_type;
 | |
| 
 | |
|   aac_dec_handle->block_number =
 | |
|       ixheaacd_add32(aac_dec_handle->block_number, 1);
 | |
|   return error_code;
 | |
| }
 | |
| 
 | |
| WORD32 ixheaacd_extension_payload(ia_bit_buf_struct *it_bit_buff, WORD32 cnt) {
 | |
|   WORD16 extension_type, discard;
 | |
|   WORD32 i;
 | |
|   WORD32 fill_nibble;
 | |
| 
 | |
|   WORD32 err = 0;
 | |
|   extension_type = (WORD16)ixheaacd_read_bits_buf(it_bit_buff, 4);
 | |
|   switch (extension_type) {
 | |
|     case EXT_FILL_DATA:
 | |
| 
 | |
|       fill_nibble = ixheaacd_read_bits_buf(it_bit_buff, 4);
 | |
| 
 | |
|       if (fill_nibble == 0) {
 | |
|         for (i = 0; i < (cnt >> 3) - 1; i++) {
 | |
|           ixheaacd_read_bits_buf(it_bit_buff, 8);
 | |
|         }
 | |
| 
 | |
|       } else
 | |
|         err = -1;
 | |
|       break;
 | |
|     case EXT_FIL:
 | |
|     default:
 | |
|       for (i = 0; i < ((cnt)-8) + 4; i++) {
 | |
|         discard = (WORD16)ixheaacd_read_bits_buf(it_bit_buff, 1);
 | |
|       }
 | |
|   }
 | |
| 
 | |
|   return err;
 | |
| }
 |