368 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			368 lines
		
	
	
		
			11 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 <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_bitbuffer.h"
 | |
| #include "ixheaacd_defines.h"
 | |
| #include "ixheaacd_aac_rom.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_drc_dec.h"
 | |
| #include "ixheaacd_sbrdecoder.h"
 | |
| 
 | |
| #include "ixheaacd_audioobjtypes.h"
 | |
| #include "ixheaacd_sbrdecsettings.h"
 | |
| #include "ixheaacd_memory_standards.h"
 | |
| #include "ixheaacd_error_codes.h"
 | |
| 
 | |
| #include "ixheaacd_defines.h"
 | |
| 
 | |
| #include "ixheaacd_sbr_scale.h"
 | |
| #include "ixheaacd_lpp_tran.h"
 | |
| #include "ixheaacd_env_extr_part.h"
 | |
| #include "ixheaacd_sbr_rom.h"
 | |
| 
 | |
| #include "ixheaacd_hybrid.h"
 | |
| #include "ixheaacd_ps_dec.h"
 | |
| #include "ixheaacd_ps_bitdec.h"
 | |
| 
 | |
| #include "ixheaacd_pulsedata.h"
 | |
| 
 | |
| #include "ixheaacd_pns.h"
 | |
| 
 | |
| #include "ixheaacd_channelinfo.h"
 | |
| 
 | |
| #include "ixheaacd_env_extr.h"
 | |
| #include "ixheaacd_common_rom.h"
 | |
| #include "ixheaacd_block.h"
 | |
| #include "ixheaacd_channel.h"
 | |
| #include "ixheaacd_audioobjtypes.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_basic_op.h"
 | |
| #include "ixheaacd_intrinsics.h"
 | |
| 
 | |
| static PLATFORM_INLINE UWORD32
 | |
| ixheaacd_aac_showbits_7(ia_bit_buf_struct *it_bit_buff) {
 | |
|   UWORD8 *v = it_bit_buff->ptr_read_next;
 | |
|   UWORD32 b = 0;
 | |
|   UWORD32 x;
 | |
|   b = ((WORD32)v[0] << 8);
 | |
|   if (it_bit_buff->bit_pos < 6) {
 | |
|     b |= (WORD32)(v[1]);
 | |
|   }
 | |
|   x = (UWORD32)b << (15 + 8 - it_bit_buff->bit_pos);
 | |
|   x = (UWORD32)x >> (25);
 | |
| 
 | |
|   return x;
 | |
| }
 | |
| 
 | |
| WORD ixheaacd_get_channel_mask(
 | |
|     ia_exhaacplus_dec_api_struct *p_obj_enhaacplus_dec) {
 | |
|   WORD ixheaacd_drc_offset = 0, channel_mask = 0;
 | |
|   WORD flag1 = 0, flag2 = 0;
 | |
|   WORD ch_idx;
 | |
|   WORD *ptr_slot_element = p_obj_enhaacplus_dec->aac_config.slot_element;
 | |
|   WORD *ptr_element_type = p_obj_enhaacplus_dec->aac_config.element_type;
 | |
| 
 | |
|   memset(ptr_slot_element, 0, sizeof(WORD) * MAX_BS_ELEMENT);
 | |
| 
 | |
|   for (ch_idx = 0; ch_idx < MAX_BS_ELEMENT; ch_idx++) {
 | |
|     if (ptr_element_type[ch_idx] == 1) {
 | |
|       channel_mask += 0x3;
 | |
|       ptr_slot_element[ch_idx] = ixheaacd_drc_offset;
 | |
|       ixheaacd_drc_offset += 2;
 | |
|       flag1 = ch_idx + 1;
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   for (ch_idx = 0; ch_idx < MAX_BS_ELEMENT; ch_idx++) {
 | |
|     if (ptr_element_type[ch_idx] == 0) {
 | |
|       channel_mask += 0x4;
 | |
|       ptr_slot_element[ch_idx] = ixheaacd_drc_offset;
 | |
|       ixheaacd_drc_offset += 1;
 | |
|       flag2 = ch_idx + 1;
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
|   for (ch_idx = 0; ch_idx < MAX_BS_ELEMENT; ch_idx++) {
 | |
|     if (ptr_element_type[ch_idx] == 3) {
 | |
|       channel_mask += 0x8;
 | |
|       ptr_slot_element[ch_idx] = ixheaacd_drc_offset;
 | |
|       ixheaacd_drc_offset += 1;
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
|   for (ch_idx = flag1; ch_idx < MAX_BS_ELEMENT; ch_idx++) {
 | |
|     if (ptr_element_type[ch_idx] == 1) {
 | |
|       channel_mask += 0x30;
 | |
|       ptr_slot_element[ch_idx] = ixheaacd_drc_offset;
 | |
|       ixheaacd_drc_offset += 2;
 | |
|       flag1 = ch_idx + 1;
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
|   for (ch_idx = flag2; ch_idx < MAX_BS_ELEMENT; ch_idx++) {
 | |
|     if (ptr_element_type[ch_idx] == 0) {
 | |
|       channel_mask += 0x100;
 | |
|       ptr_slot_element[ch_idx] = ixheaacd_drc_offset;
 | |
|       ixheaacd_drc_offset += 1;
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
|   for (ch_idx = flag1; ch_idx < MAX_BS_ELEMENT; ch_idx++) {
 | |
|     if (ptr_element_type[ch_idx] == 1) {
 | |
|       {
 | |
|         channel_mask += (0x40 + 0x80);
 | |
|         ptr_slot_element[ch_idx] = ixheaacd_drc_offset;
 | |
|         ixheaacd_drc_offset += 2;
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return channel_mask;
 | |
| }
 | |
| 
 | |
| VOID ixheaacd_read_data_stream_element(ia_bit_buf_struct *it_bit_buff,
 | |
|                                        WORD32 *byte_align_bits,
 | |
|                                        ia_drc_dec_struct *drc_handle) {
 | |
|   ia_bit_buf_struct temp_bs = {0};
 | |
|   WORD32 count = ixheaacd_read_bits_buf(it_bit_buff, 13);
 | |
|   WORD32 cnt = (count & 0xff);
 | |
|   WORD32 start_pos = 0;
 | |
| 
 | |
|   if (cnt == 255) {
 | |
|     cnt += ixheaacd_read_bits_buf(it_bit_buff, 8);
 | |
|   }
 | |
| 
 | |
|   if ((count & 0x0100) >> 8) {
 | |
|     ixheaacd_byte_align(it_bit_buff, byte_align_bits);
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     memcpy(&temp_bs, it_bit_buff, sizeof(ia_bit_buf_struct));
 | |
|     start_pos = temp_bs.cnt_bits;
 | |
| 
 | |
|     if (ixheaacd_read_bits_buf(&temp_bs, 8) == DVB_ANC_DATA_SYNC_BYTE) {
 | |
|       int dmx_level_present, compression_present;
 | |
|       int coarse_gain_present, fine_grain_present;
 | |
| 
 | |
|       ixheaacd_read_bits_buf(&temp_bs, 8);
 | |
| 
 | |
|       ixheaacd_read_bits_buf(&temp_bs, 3);
 | |
|       dmx_level_present = ixheaacd_read_bits_buf(&temp_bs, 1);
 | |
|       ixheaacd_read_bits_buf(&temp_bs, 1);
 | |
|       compression_present = ixheaacd_read_bits_buf(&temp_bs, 1);
 | |
|       coarse_gain_present = ixheaacd_read_bits_buf(&temp_bs, 1);
 | |
|       fine_grain_present = ixheaacd_read_bits_buf(&temp_bs, 1);
 | |
| 
 | |
|       if (dmx_level_present) ixheaacd_read_bits_buf(&temp_bs, 8);
 | |
| 
 | |
|       if (compression_present) ixheaacd_read_bits_buf(&temp_bs, 16);
 | |
| 
 | |
|       if (coarse_gain_present) ixheaacd_read_bits_buf(&temp_bs, 16);
 | |
| 
 | |
|       if (fine_grain_present) ixheaacd_read_bits_buf(&temp_bs, 16);
 | |
| 
 | |
|       if (!drc_handle->dvb_anc_data_present && temp_bs.cnt_bits >= 0) {
 | |
|         drc_handle->dvb_anc_data_pos = start_pos;
 | |
|         drc_handle->dvb_anc_data_present = 1;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (it_bit_buff->cnt_bits < (cnt << 3)) {
 | |
|     longjmp(*(it_bit_buff->xaac_jmp_buf),
 | |
|             IA_ENHAACPLUS_DEC_EXE_NONFATAL_INSUFFICIENT_INPUT_BYTES);
 | |
|   }
 | |
|   it_bit_buff->ptr_read_next += cnt;
 | |
|   it_bit_buff->cnt_bits -= ((cnt) << 3);
 | |
| }
 | |
| 
 | |
| VOID ixheaacd_read_fill_element(ia_bit_buf_struct *it_bit_buff,
 | |
|                                 ia_drc_dec_struct *drc_dummy,
 | |
|                                 ia_drc_dec_struct *ptr_drc_dec) {
 | |
|   WORD32 count;
 | |
|   count = ixheaacd_read_bits_buf(it_bit_buff, 4);
 | |
| 
 | |
|   if ((count - 15) == 0) {
 | |
|     count = ixheaacd_read_bits_buf(it_bit_buff, 8);
 | |
|     count = (count + 14);
 | |
|   }
 | |
| 
 | |
|   if (count > 0) {
 | |
|     WORD32 extension_type;
 | |
| 
 | |
|     extension_type = ixheaacd_read_bits_buf(it_bit_buff, 4);
 | |
| 
 | |
|     if (extension_type == EXT_DYNAMIC_RANGE) {
 | |
|       ptr_drc_dec->drc_element_found = 1;
 | |
|       count -=
 | |
|           ixheaacd_dec_drc_read_element(ptr_drc_dec, drc_dummy, it_bit_buff);
 | |
| 
 | |
|     } else {
 | |
|       ixheaacd_read_bits_buf(it_bit_buff, 4);
 | |
| 
 | |
|       if (it_bit_buff->cnt_bits < ((count - 1) << 3)) {
 | |
|         longjmp(*(it_bit_buff->xaac_jmp_buf),
 | |
|                 IA_ENHAACPLUS_DEC_EXE_NONFATAL_INSUFFICIENT_INPUT_BYTES);
 | |
|       }
 | |
|       it_bit_buff->ptr_read_next += count - 1;
 | |
|       it_bit_buff->cnt_bits -= ((count - 1) << 3);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| WORD32 ixheaacd_get_element_index_tag(
 | |
|     ia_exhaacplus_dec_api_struct *p_obj_enhaacplus_dec, WORD ch_idx1,
 | |
|     WORD *ch_idx, WORD *channel, WORD *ele_idx_order, WORD total_elements,
 | |
|     WORD8 *element_used, WORD total_channels, ia_drc_dec_struct *pstr_drc_dec,
 | |
|     ia_drc_dec_struct *drc_dummy) {
 | |
|   WORD element_tag, j;
 | |
|   ia_aac_dec_state_struct *p_state_enhaacplus_dec =
 | |
|       p_obj_enhaacplus_dec->p_state_aac;
 | |
| 
 | |
|   ia_bit_buf_struct *it_bit_buff = p_state_enhaacplus_dec->ptr_bit_stream;
 | |
|   WORD element_idx;
 | |
|   WORD element_type;
 | |
| 
 | |
|   ia_aac_decoder_struct *aac_dec_handle =
 | |
|       p_state_enhaacplus_dec->pstr_aac_dec_info[ch_idx1];
 | |
| 
 | |
|   *ch_idx = ch_idx1;
 | |
| 
 | |
|   if (p_state_enhaacplus_dec->bs_format != LOAS_BSFORMAT) {
 | |
|     if (ch_idx1 == 0) {
 | |
|       ixheaacd_byte_align(it_bit_buff, &aac_dec_handle->byte_align_bits);
 | |
|     }
 | |
|   }
 | |
|   {
 | |
|     if (ch_idx1 == 0) {
 | |
|       aac_dec_handle->byte_align_bits = it_bit_buff->cnt_bits;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (it_bit_buff->cnt_bits < 3) {
 | |
|     it_bit_buff->cnt_bits = -1;
 | |
|     return (WORD16)(
 | |
|         (WORD32)IA_ENHAACPLUS_DEC_EXE_NONFATAL_INSUFFICIENT_INPUT_BYTES);
 | |
|   }
 | |
| 
 | |
|   element_tag = (WORD)ixheaacd_read_bits_buf(it_bit_buff, 7);
 | |
|   ixheaacd_read_bidirection(it_bit_buff, -7);
 | |
| 
 | |
|   element_idx = (element_tag & 0xF);
 | |
|   element_type = (element_tag >> 4) & 0x7;
 | |
| 
 | |
|   p_obj_enhaacplus_dec->aac_config.str_prog_config.alignment_bits =
 | |
|       it_bit_buff->bit_pos;
 | |
| 
 | |
|   while (element_type == 4 || element_type == 5 || element_type == 6) {
 | |
|     WORD type = (WORD)ixheaacd_read_bits_buf(it_bit_buff, 3);
 | |
| 
 | |
|     if (it_bit_buff->cnt_bits < 3) {
 | |
|       it_bit_buff->cnt_bits = -1;
 | |
|       return (WORD16)(
 | |
|           (WORD32)IA_ENHAACPLUS_DEC_EXE_NONFATAL_INSUFFICIENT_INPUT_BYTES);
 | |
|     }
 | |
| 
 | |
|     if (type == 4) {
 | |
|       ixheaacd_read_data_stream_element(
 | |
|           it_bit_buff, &aac_dec_handle->byte_align_bits, pstr_drc_dec);
 | |
|     }
 | |
|     if (type == 5) {
 | |
|       WORD32 error_code = 0;
 | |
|       error_code = ixheaacd_decode_pce(
 | |
|           it_bit_buff, &p_obj_enhaacplus_dec->aac_config.ui_pce_found_in_hdr,
 | |
|           &p_obj_enhaacplus_dec->aac_config.str_prog_config);
 | |
|       if (error_code != 0) {
 | |
|         if (error_code < 0) return error_code;
 | |
|         return IA_ENHAACPLUS_DEC_EXE_NONFATAL_DECODE_FRAME_ERROR;
 | |
|       }
 | |
|     }
 | |
|     if (type == 6) {
 | |
|       ixheaacd_read_fill_element(it_bit_buff, drc_dummy, pstr_drc_dec);
 | |
|     }
 | |
| 
 | |
|     if (it_bit_buff->cnt_bits < 7) {
 | |
|       it_bit_buff->cnt_bits = -1;
 | |
|       return (WORD16)(
 | |
|           (WORD32)IA_ENHAACPLUS_DEC_EXE_NONFATAL_INSUFFICIENT_INPUT_BYTES);
 | |
|     }
 | |
| 
 | |
|     element_tag = (WORD)ixheaacd_aac_showbits_7(it_bit_buff);
 | |
|     element_idx = (element_tag & 0xF);
 | |
|     element_type = (element_tag >> 4) & 0x7;
 | |
|   }
 | |
| 
 | |
|   if (total_elements == 2 && total_channels == 2 &&
 | |
|       p_state_enhaacplus_dec->p_config->ui_pce_found_in_hdr == 2 &&
 | |
|       ch_idx1 == 0) {
 | |
|     ixheaacd_fill_prog_config_slots(p_state_enhaacplus_dec);
 | |
|   }
 | |
| 
 | |
|   *channel = 1;
 | |
|   if (element_type == 1) {
 | |
|     *channel = 2;
 | |
|   }
 | |
| 
 | |
|   for (j = 0; j < total_elements; j++) {
 | |
|     if (p_obj_enhaacplus_dec->aac_config.element_type[j] == element_type &&
 | |
|         (element_idx == ele_idx_order[j]) && (element_used[j] == 0)) {
 | |
|       *ch_idx = j;
 | |
|       element_used[j] = 1;
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (j == total_elements) {
 | |
|     if (it_bit_buff->cnt_bits < 0) {
 | |
|       return (WORD16)(
 | |
|           (WORD32)IA_ENHAACPLUS_DEC_EXE_NONFATAL_INSUFFICIENT_INPUT_BYTES);
 | |
|     }
 | |
| 
 | |
|     ixheaacd_read_bidirection(
 | |
|         it_bit_buff, (WORD16)(it_bit_buff->cnt_bits - it_bit_buff->size));
 | |
|     return IA_ENHAACPLUS_DEC_EXE_NONFATAL_ELE_INSTANCE_TAG_NOT_FOUND;
 | |
|   } else
 | |
|     return 0;
 | |
| }
 |