/****************************************************************************** * * * 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 #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_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_config.h" #include "ixheaacd_mps_polyphase.h" #include "ixheaacd_mps_dec.h" #include "ixheaacd_struct_def.h" #include "ixheaacd_multichannel.h" #include "ixheaacd_headerdecode.h" #include "ixheaacd_error_standards.h" WORD32 ixheaacd_latm_au_chunk_length_info( struct ia_bit_buf_struct *it_bit_buff) { UWORD8 reading_done; WORD32 len = 0; do { UWORD32 tmp = ixheaacd_read_bits_buf(it_bit_buff, 8); reading_done = (tmp < 255); len += tmp; } while (reading_done == 0); len <<= 3; return len; } WORD32 ixheaacd_latm_payload_length_info(struct ia_bit_buf_struct *it_bit_buff, ixheaacd_latm_struct *latm_element) { WORD32 error_code = AAC_DEC_OK; UWORD32 prog, lay; if (latm_element->all_streams_same_time_framing == 1) { for (prog = 0; prog < latm_element->num_program; prog++) { for (lay = 0; lay < latm_element->num_layer; lay++) { ixheaacd_latm_layer_info *layer_info = &latm_element->layer_info[prog][lay]; switch (layer_info->frame_len_type) { case 0: layer_info->frame_len_bits = ixheaacd_latm_au_chunk_length_info(it_bit_buff); if (layer_info->frame_len_bits % 8 != 0) { error_code = IA_ENHAACPLUS_DEC_EXE_FATAL_INVALID_LOAS_HEADER; return error_code; } latm_element->frame_length = layer_info->frame_len_bits >> 3; latm_element->frame_length += (it_bit_buff->size - it_bit_buff->cnt_bits) >> 3; break; default: error_code = IA_ENHAACPLUS_DEC_EXE_FATAL_INVALID_LOAS_HEADER; return error_code; } } } } else { error_code = IA_ENHAACPLUS_DEC_EXE_FATAL_INVALID_LOAS_HEADER; return error_code; } return (error_code); } static UWORD32 ixheaacd_latm_get_value(ia_bit_buf_struct *it_bit_buff) { UWORD32 bytes_read; bytes_read = ixheaacd_read_bits_buf(it_bit_buff, 2) + 1; if (bytes_read <= 3) return ixheaacd_read_bits_buf(it_bit_buff, 8 * bytes_read); else return (ixheaacd_read_bits_buf(it_bit_buff, 24) << 8) + ixheaacd_read_bits_buf(it_bit_buff, 8); } IA_ERRORCODE ixheaacd_latm_stream_mux_config( struct ia_bit_buf_struct *it_bit_buff, ixheaacd_latm_struct *latm_element, ia_aac_dec_state_struct *aac_state_struct, ia_sampling_rate_info_struct *sample_rate_info) { UWORD32 prog; UWORD32 lay; WORD32 bytes_consumed; WORD32 audio_mux_version_a; UWORD32 tara_buf_fullness; IA_ERRORCODE error_code = AAC_DEC_OK; ixheaacd_latm_layer_info *layer_info = 0; latm_element->audio_mux_version = ixheaacd_read_bits_buf(it_bit_buff, 1); if (latm_element->audio_mux_version == 1) audio_mux_version_a = ixheaacd_read_bits_buf(it_bit_buff, 1); else audio_mux_version_a = 0; if (audio_mux_version_a == 0) { if (latm_element->audio_mux_version == 1) { tara_buf_fullness = ixheaacd_latm_get_value(it_bit_buff); } latm_element->all_streams_same_time_framing = ixheaacd_read_bits_buf(it_bit_buff, 1); latm_element->num_sub_frames = ixheaacd_read_bits_buf(it_bit_buff, 6) + 1; if (latm_element->num_sub_frames != 1) return IA_ENHAACPLUS_DEC_EXE_FATAL_INVALID_LOAS_HEADER; latm_element->num_program = ixheaacd_read_bits_buf(it_bit_buff, 4) + 1; if (latm_element->num_program > LATM_MAX_PROG) return IA_FATAL_ERROR; for (prog = 0; prog < latm_element->num_program; prog++) { latm_element->num_layer = ixheaacd_read_bits_buf(it_bit_buff, 3) + 1; for (lay = 0; lay < latm_element->num_layer; lay++) { layer_info = &latm_element->layer_info[prog][lay]; layer_info->frame_len_bits = 0; if ((prog == 0) && (lay == 0)) { WORD32 asc_len, pos; latm_element->use_same_config = 0; asc_len = (latm_element->audio_mux_version == 1) ? ixheaacd_latm_get_value(it_bit_buff) : 0; pos = it_bit_buff->size - it_bit_buff->cnt_bits; if (asc_len > it_bit_buff->size - 106 || asc_len > 2592 || asc_len < 0) { return IA_ENHAACPLUS_DEC_INIT_FATAL_DEC_INIT_FAIL; } if ((error_code = ixheaacd_ga_hdr_dec( aac_state_struct, it_bit_buff->cnt_bits, &bytes_consumed, sample_rate_info, it_bit_buff))) return (error_code); if (asc_len) { asc_len -= (it_bit_buff->size - it_bit_buff->cnt_bits) - pos; ixheaacd_read_bidirection(it_bit_buff, asc_len); } layer_info->asc.aot = aac_state_struct->audio_object_type; layer_info->asc.channel_config = aac_state_struct->ch_config; layer_info->asc.samples_per_frame = aac_state_struct->frame_len_flag; layer_info->asc.sampling_freq = aac_state_struct->sampling_rate; layer_info->asc.samples_per_frame = aac_state_struct->frame_length; } else { latm_element->use_same_config = ixheaacd_read_bits_buf(it_bit_buff, 1); if (latm_element->use_same_config && (lay > 0)) { layer_info->asc = latm_element->layer_info[prog][lay - 1].asc; } else { if ((error_code = ixheaacd_ga_hdr_dec( aac_state_struct, it_bit_buff->cnt_bits, &bytes_consumed, sample_rate_info, it_bit_buff))) return (error_code); } } layer_info->frame_len_type = ixheaacd_read_bits_buf(it_bit_buff, 3); switch (layer_info->frame_len_type) { case 0: layer_info->buffer_fullness = ixheaacd_read_bits_buf(it_bit_buff, 8); if (!latm_element->all_streams_same_time_framing) { if ((lay > 0) && layer_info->asc.aot == AOT_AAC_SCAL) { } } break; default: return IA_ENHAACPLUS_DEC_EXE_FATAL_INVALID_LOAS_HEADER; } } } latm_element->other_data_present = ixheaacd_read_bits_buf(it_bit_buff, 1); if (latm_element->other_data_present) { if (latm_element->audio_mux_version == 1) { latm_element->other_data_length = ixheaacd_latm_get_value(it_bit_buff); } else { UWORD32 other_data_len; latm_element->other_data_length = 0; do { other_data_len = ixheaacd_read_bits_buf(it_bit_buff, 1); latm_element->other_data_length <<= 8; latm_element->other_data_length += ixheaacd_read_bits_buf(it_bit_buff, 8); if (latm_element->other_data_length > (UWORD32)it_bit_buff->cnt_bits) return IA_FATAL_ERROR; } while (other_data_len); } } latm_element->crc_check_present = ixheaacd_read_bits_buf(it_bit_buff, 1); if (latm_element->crc_check_present) { latm_element->crc_check_sum = ixheaacd_read_bits_buf(it_bit_buff, 8); } } else { error_code = IA_ENHAACPLUS_DEC_EXE_FATAL_INVALID_LOAS_HEADER; } return (error_code); } IA_ERRORCODE ixheaacd_latm_audio_mux_element( struct ia_bit_buf_struct *it_bit_buff, ixheaacd_latm_struct *latm_element, ia_aac_dec_state_struct *aac_state_struct, ia_sampling_rate_info_struct *sample_rate_info) { UWORD32 i; IA_ERRORCODE error_code = AAC_DEC_OK; ixheaacd_read_bits_buf(it_bit_buff, 13); latm_element->use_same_stream_mux = ixheaacd_read_bits_buf(it_bit_buff, 1); if (!latm_element->use_same_stream_mux) { if ((error_code = ixheaacd_latm_stream_mux_config( it_bit_buff, latm_element, aac_state_struct, sample_rate_info))) { return (error_code); } } for (i = 0; i < latm_element->num_sub_frames; i++) { if ((error_code = ixheaacd_latm_payload_length_info(it_bit_buff, latm_element))) { if (error_code != 0) return (error_code); } } return (error_code); }