| /****************************************************************************** |
| * |
| * 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 <stdlib.h> |
| #include "impd_type_def.h" |
| #include "impd_drc_extr_delta_coded_info.h" |
| #include "impd_drc_common.h" |
| #include "impd_drc_struct.h" |
| #include "impd_parametric_drc_dec.h" |
| #include "impd_drc_gain_dec.h" |
| #include "impd_drc_filter_bank.h" |
| #include "impd_drc_multi_band.h" |
| |
| WORD32 impd_init_drc_params(WORD32 frame_size, WORD32 sample_rate, |
| WORD32 gain_delay_samples, WORD32 delay_mode, |
| WORD32 sub_band_domain_mode, |
| ia_drc_params_struct* ia_drc_params_struct) { |
| WORD32 k; |
| if (frame_size < 1 || frame_size > AUDIO_CODEC_FRAME_SIZE_MAX) { |
| return -1; |
| } |
| |
| if (sample_rate < 1000) { |
| return -1; |
| } |
| |
| ia_drc_params_struct->drc_frame_size = frame_size; |
| |
| if (ia_drc_params_struct->drc_frame_size < 0.001f * sample_rate) { |
| return -1; |
| } |
| |
| ia_drc_params_struct->sample_rate = sample_rate; |
| |
| ia_drc_params_struct->delta_tmin_default = impd_get_delta_tmin(sample_rate); |
| |
| if (ia_drc_params_struct->delta_tmin_default > |
| ia_drc_params_struct->drc_frame_size) { |
| return -1; |
| } |
| |
| if ((delay_mode != DELAY_MODE_REGULAR_DELAY) && |
| (delay_mode != DELAY_MODE_LOW_DELAY)) { |
| return -1; |
| } |
| |
| ia_drc_params_struct->delay_mode = delay_mode; |
| |
| ia_drc_params_struct->drc_set_counter = 0; |
| ia_drc_params_struct->multiband_sel_drc_idx = -1; |
| |
| for (k = 0; k < SEL_DRC_COUNT; k++) { |
| ia_drc_params_struct->sel_drc_array[k].drc_instructions_index = -1; |
| ia_drc_params_struct->sel_drc_array[k].dwnmix_instructions_index = -1; |
| ia_drc_params_struct->sel_drc_array[k].drc_coeff_idx = -1; |
| } |
| |
| if ((gain_delay_samples > MAX_SIGNAL_DELAY) || (gain_delay_samples < 0)) { |
| return -1; |
| } else { |
| ia_drc_params_struct->gain_delay_samples = gain_delay_samples; |
| } |
| |
| switch (sub_band_domain_mode) { |
| case SUBBAND_DOMAIN_MODE_OFF: |
| case SUBBAND_DOMAIN_MODE_QMF64: |
| case SUBBAND_DOMAIN_MODE_QMF71: |
| case SUBBAND_DOMAIN_MODE_STFT256: |
| ia_drc_params_struct->sub_band_domain_mode = sub_band_domain_mode; |
| break; |
| default: |
| return -1; |
| break; |
| } |
| |
| ia_drc_params_struct->parametric_drc_delay = 0; |
| ia_drc_params_struct->eq_delay = 0; |
| |
| return 0; |
| } |
| |
| WORD32 impd_select_drc_coefficients( |
| ia_drc_config* drc_config, ia_uni_drc_coeffs_struct** drc_coefficients_drc, |
| WORD32* drc_coefficients_selected) { |
| WORD32 i; |
| WORD32 cof1 = -1; |
| WORD32 cof0 = -1; |
| for (i = 0; i < drc_config->drc_coefficients_drc_count; i++) { |
| if (drc_config->str_p_loc_drc_coefficients_uni_drc[i].drc_location == 1) { |
| if (drc_config->str_p_loc_drc_coefficients_uni_drc[i].version == 0) { |
| cof0 = i; |
| *drc_coefficients_selected = cof0; |
| } else { |
| cof1 = i; |
| *drc_coefficients_selected = cof1; |
| } |
| } |
| } |
| |
| if (cof1 >= 0) { |
| *drc_coefficients_drc = |
| &(drc_config->str_p_loc_drc_coefficients_uni_drc[cof1]); |
| } else if (cof0 >= 0) { |
| *drc_coefficients_drc = |
| &(drc_config->str_p_loc_drc_coefficients_uni_drc[cof0]); |
| } else { |
| *drc_coefficients_drc = NULL; |
| } |
| |
| return 0; |
| } |
| |
| WORD32 impd_init_selected_drc_set( |
| ia_drc_config* drc_config, ia_drc_params_struct* ia_drc_params_struct, |
| ia_parametric_drc_params_struct* p_parametric_drc_params, |
| WORD32 audio_num_chan, WORD32 drc_set_id_selected, |
| WORD32 downmix_id_selected, ia_filter_banks_struct* ia_filter_banks_struct, |
| ia_overlap_params_struct* pstr_overlap_params |
| |
| , |
| shape_filter_block* shape_filter_block) { |
| WORD32 g, n, c, err = 0; |
| WORD32 channel_count = 0; |
| WORD32 i; |
| |
| ia_drc_instructions_struct* drc_instructions_uni_drc = NULL; |
| ia_uni_drc_coeffs_struct* drc_coefficients_uni_drc = NULL; |
| WORD32 selected_drc_is_multiband = 0; |
| WORD32 drc_instructions_selected = -1; |
| WORD32 downmix_instructions_selected = -1; |
| WORD32 drc_coefficients_selected = -1; |
| p_parametric_drc_params->parametric_drc_instance_count = 0; |
| |
| if (drc_config->drc_coefficients_drc_count && |
| drc_config->str_p_loc_drc_coefficients_uni_drc->drc_frame_size_present) { |
| if (drc_config->str_p_loc_drc_coefficients_uni_drc->drc_frame_size != |
| ia_drc_params_struct->drc_frame_size) { |
| return -1; |
| } |
| } |
| |
| for (n = 0; n < drc_config->drc_instructions_count_plus; n++) { |
| if (drc_config->str_drc_instruction_str[n].drc_set_id == |
| drc_set_id_selected) |
| break; |
| } |
| if (n == drc_config->drc_instructions_count_plus) { |
| return -1; |
| } |
| drc_instructions_selected = n; |
| drc_instructions_uni_drc = |
| &(drc_config->str_drc_instruction_str[drc_instructions_selected]); |
| |
| if (downmix_id_selected == ID_FOR_BASE_LAYOUT) { |
| channel_count = drc_config->channel_layout.base_channel_count; |
| } else if (downmix_id_selected == ID_FOR_ANY_DOWNMIX) { |
| channel_count = audio_num_chan; |
| } else { |
| for (n = 0; n < drc_config->dwnmix_instructions_count; n++) { |
| if (drc_config->dwnmix_instructions[n].downmix_id == downmix_id_selected) |
| break; |
| } |
| if (n == drc_config->dwnmix_instructions_count) { |
| return (UNEXPECTED_ERROR); |
| } |
| channel_count = drc_config->dwnmix_instructions[n].target_channel_count; |
| |
| downmix_instructions_selected = n; |
| } |
| drc_instructions_uni_drc->audio_num_chan = channel_count; |
| |
| if (drc_instructions_uni_drc->drc_set_id <= 0) { |
| drc_coefficients_selected = 0; |
| } else { |
| err = impd_select_drc_coefficients(drc_config, &drc_coefficients_uni_drc, |
| &drc_coefficients_selected); |
| } |
| |
| ia_drc_params_struct->sel_drc_array[ia_drc_params_struct->drc_set_counter] |
| .drc_instructions_index = drc_instructions_selected; |
| ia_drc_params_struct->sel_drc_array[ia_drc_params_struct->drc_set_counter] |
| .dwnmix_instructions_index = downmix_instructions_selected; |
| ia_drc_params_struct->sel_drc_array[ia_drc_params_struct->drc_set_counter] |
| .drc_coeff_idx = drc_coefficients_selected; |
| |
| if ((drc_instructions_uni_drc->downmix_id[0] == ID_FOR_ANY_DOWNMIX) || |
| (drc_instructions_uni_drc->dwnmix_id_count > 1)) { |
| WORD32 idx = drc_instructions_uni_drc->gain_set_index[0]; |
| for (c = 0; c < drc_instructions_uni_drc->audio_num_chan; c++) { |
| drc_instructions_uni_drc->channel_group_of_ch[c] = (idx >= 0) ? 0 : -1; |
| } |
| } |
| |
| for (g = 0; g < drc_instructions_uni_drc->num_drc_ch_groups; g++) { |
| drc_instructions_uni_drc->num_chan_per_ch_group[g] = 0; |
| for (c = 0; c < drc_instructions_uni_drc->audio_num_chan; c++) { |
| if (drc_instructions_uni_drc->channel_group_of_ch[c] == g) { |
| drc_instructions_uni_drc->num_chan_per_ch_group[g]++; |
| } |
| } |
| } |
| |
| if (drc_instructions_uni_drc->drc_set_effect & |
| (EFFECT_BIT_DUCK_OTHER | EFFECT_BIT_DUCK_SELF)) { |
| drc_instructions_uni_drc->multiband_audio_sig_count = |
| drc_instructions_uni_drc->audio_num_chan; |
| } else { |
| drc_instructions_uni_drc->multiband_audio_sig_count = 0; |
| for (c = 0; c < drc_instructions_uni_drc->audio_num_chan; c++) { |
| g = drc_instructions_uni_drc->channel_group_of_ch[c]; |
| if (g < 0) { |
| drc_instructions_uni_drc->multiband_audio_sig_count++; |
| } else { |
| drc_instructions_uni_drc->multiband_audio_sig_count += |
| drc_instructions_uni_drc->band_count_of_ch_group[g]; |
| } |
| } |
| } |
| |
| for (g = 0; g < drc_instructions_uni_drc->num_drc_ch_groups; g++) { |
| if (g == 0) { |
| drc_instructions_uni_drc->parametric_drc_look_ahead_samples_max = 0; |
| } |
| if (drc_instructions_uni_drc->ch_group_parametric_drc_flag[g] == 0) { |
| if (drc_instructions_uni_drc->band_count_of_ch_group[g] > 1) { |
| if (ia_drc_params_struct->multiband_sel_drc_idx != -1) { |
| return (UNEXPECTED_ERROR); |
| } |
| selected_drc_is_multiband = 1; |
| } |
| } else { |
| WORD32 gain_set_index = |
| drc_instructions_uni_drc->gain_set_idx_of_ch_group_parametric_drc[g]; |
| WORD32 parametric_drc_id = |
| drc_config->str_drc_config_ext.str_drc_coeff_param_drc |
| .str_parametric_drc_gain_set_params[gain_set_index] |
| .parametric_drc_id; |
| WORD32 parametric_drc_look_ahead_samples = 0; |
| ia_parametric_drc_instructions_struct* str_parametric_drc_instructions; |
| |
| for (i = 0; |
| i < drc_config->str_drc_config_ext.parametric_drc_instructions_count; |
| i++) { |
| if (parametric_drc_id == |
| drc_config->str_drc_config_ext.str_parametric_drc_instructions[i] |
| .parametric_drc_id) |
| break; |
| } |
| if (i == |
| drc_config->str_drc_config_ext.parametric_drc_instructions_count) { |
| return (UNEXPECTED_ERROR); |
| } |
| str_parametric_drc_instructions = |
| &drc_config->str_drc_config_ext.str_parametric_drc_instructions[i]; |
| |
| p_parametric_drc_params->parametric_drc_idx |
| [p_parametric_drc_params->parametric_drc_instance_count] = i; |
| p_parametric_drc_params->gain_set_index |
| [p_parametric_drc_params->parametric_drc_instance_count] = |
| gain_set_index; |
| if (drc_instructions_uni_drc->drc_apply_to_dwnmix == 0) { |
| p_parametric_drc_params->dwnmix_id_from_drc_instructions |
| [p_parametric_drc_params->parametric_drc_instance_count] = |
| ID_FOR_BASE_LAYOUT; |
| } else { |
| if (drc_instructions_uni_drc->dwnmix_id_count > 1) { |
| p_parametric_drc_params->dwnmix_id_from_drc_instructions |
| [p_parametric_drc_params->parametric_drc_instance_count] = |
| ID_FOR_ANY_DOWNMIX; |
| } else { |
| p_parametric_drc_params->dwnmix_id_from_drc_instructions |
| [p_parametric_drc_params->parametric_drc_instance_count] = |
| drc_instructions_uni_drc->downmix_id[0]; |
| } |
| } |
| p_parametric_drc_params->audio_num_chan = channel_count; |
| for (i = 0; i < p_parametric_drc_params->audio_num_chan; i++) { |
| if (drc_instructions_uni_drc->channel_group_of_ch[i] == g) { |
| p_parametric_drc_params->channel_map |
| [p_parametric_drc_params->parametric_drc_instance_count][i] = 1; |
| } else { |
| p_parametric_drc_params->channel_map |
| [p_parametric_drc_params->parametric_drc_instance_count][i] = 0; |
| } |
| } |
| drc_instructions_uni_drc->parametric_drc_look_ahead_samples[g] = 0; |
| if (str_parametric_drc_instructions->parametric_drc_look_ahead_flag) { |
| parametric_drc_look_ahead_samples = (WORD32)( |
| (FLOAT32) |
| str_parametric_drc_instructions->parametric_drc_look_ahead * |
| (FLOAT32)p_parametric_drc_params->sampling_rate * 0.001f); |
| } else { |
| if (str_parametric_drc_instructions->parametric_drc_type == |
| PARAM_DRC_TYPE_FF) { |
| parametric_drc_look_ahead_samples = (WORD32)( |
| 10.0f * (FLOAT32)p_parametric_drc_params->sampling_rate * 0.001f); |
| } else if (str_parametric_drc_instructions->parametric_drc_type == |
| PARAM_DRC_TYPE_LIM) { |
| parametric_drc_look_ahead_samples = (WORD32)( |
| 5.0f * (FLOAT32)p_parametric_drc_params->sampling_rate * 0.001f); |
| } else { |
| return (UNEXPECTED_ERROR); |
| } |
| } |
| drc_instructions_uni_drc->parametric_drc_look_ahead_samples[g] = |
| parametric_drc_look_ahead_samples; |
| if (drc_instructions_uni_drc->parametric_drc_look_ahead_samples_max < |
| drc_instructions_uni_drc->parametric_drc_look_ahead_samples[g]) { |
| drc_instructions_uni_drc->parametric_drc_look_ahead_samples_max = |
| drc_instructions_uni_drc->parametric_drc_look_ahead_samples[g]; |
| } |
| p_parametric_drc_params->parametric_drc_instance_count += 1; |
| selected_drc_is_multiband = 0; |
| } |
| } |
| ia_drc_params_struct->parametric_drc_delay += |
| drc_instructions_uni_drc->parametric_drc_look_ahead_samples_max; |
| |
| if (selected_drc_is_multiband == 1) { |
| ia_drc_params_struct->multiband_sel_drc_idx = |
| ia_drc_params_struct->drc_set_counter; |
| err = impd_init_all_filter_banks( |
| drc_coefficients_uni_drc, |
| &(drc_config->str_drc_instruction_str[drc_instructions_selected]), |
| ia_filter_banks_struct); |
| if (err) return (err); |
| |
| err = impd_init_overlap_weight( |
| drc_coefficients_uni_drc, |
| &(drc_config->str_drc_instruction_str[drc_instructions_selected]), |
| ia_drc_params_struct->sub_band_domain_mode, pstr_overlap_params); |
| if (err) return (err); |
| } else { |
| ia_gain_modifiers_struct* gain_modifiers = |
| drc_config->str_drc_instruction_str->str_gain_modifiers_of_ch_group; |
| for (g = 0; g < drc_instructions_uni_drc->num_drc_ch_groups; g++) { |
| if (gain_modifiers[g].shape_filter_flag == 1) { |
| impd_shape_filt_block_init( |
| &drc_coefficients_uni_drc->str_shape_filter_block_params |
| [gain_modifiers[g].shape_filter_idx], |
| &shape_filter_block[g]); |
| } else { |
| shape_filter_block[g].shape_flter_block_flag = 0; |
| } |
| } |
| } |
| |
| ia_drc_params_struct->drc_set_counter++; |
| |
| return (0); |
| } |