1*15dc779aSAndroid Build Coastguard Worker /******************************************************************************
2*15dc779aSAndroid Build Coastguard Worker * *
3*15dc779aSAndroid Build Coastguard Worker * Copyright (C) 2023 The Android Open Source Project
4*15dc779aSAndroid Build Coastguard Worker *
5*15dc779aSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
6*15dc779aSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
7*15dc779aSAndroid Build Coastguard Worker * You may obtain a copy of the License at:
8*15dc779aSAndroid Build Coastguard Worker *
9*15dc779aSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
10*15dc779aSAndroid Build Coastguard Worker *
11*15dc779aSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
12*15dc779aSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
13*15dc779aSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14*15dc779aSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
15*15dc779aSAndroid Build Coastguard Worker * limitations under the License.
16*15dc779aSAndroid Build Coastguard Worker *
17*15dc779aSAndroid Build Coastguard Worker *****************************************************************************
18*15dc779aSAndroid Build Coastguard Worker * Originally developed and contributed by Ittiam Systems Pvt. Ltd, Bangalore
19*15dc779aSAndroid Build Coastguard Worker */
20*15dc779aSAndroid Build Coastguard Worker
21*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_type_def.h"
22*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_constants.h"
23*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_aac_constants.h"
24*15dc779aSAndroid Build Coastguard Worker #include <stdlib.h>
25*15dc779aSAndroid Build Coastguard Worker
26*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_psy_const.h"
27*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_tns.h"
28*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_tns_params.h"
29*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_rom.h"
30*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_common_rom.h"
31*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_bitbuffer.h"
32*15dc779aSAndroid Build Coastguard Worker
33*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_basic_ops32.h"
34*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_basic_ops16.h"
35*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_basic_ops40.h"
36*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_basic_ops.h"
37*15dc779aSAndroid Build Coastguard Worker
38*15dc779aSAndroid Build Coastguard Worker #include <math.h>
39*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_block_switch.h"
40*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_psy_utils_spreading.h"
41*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_psy_utils.h"
42*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_calc_ms_band_energy.h"
43*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_psy_configuration.h"
44*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_psy_data.h"
45*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_ms_stereo.h"
46*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_common_utils.h"
47*15dc779aSAndroid Build Coastguard Worker
iaace_ms_apply(ixheaace_psy_data ** ptr_psy_data,FLOAT32 * ptr_spec_left,FLOAT32 * ptr_spec_right,WORD32 * ptr_ms_select,WORD32 * ptr_ms_used,const WORD32 sfb_count,const WORD32 sfb_per_group,const WORD32 max_sfb_per_grp,const WORD32 * ptr_sfb_offset,FLOAT32 * ptr_weight_ms_lr_pe_ratio)48*15dc779aSAndroid Build Coastguard Worker VOID iaace_ms_apply(ixheaace_psy_data **ptr_psy_data, FLOAT32 *ptr_spec_left,
49*15dc779aSAndroid Build Coastguard Worker FLOAT32 *ptr_spec_right, WORD32 *ptr_ms_select, WORD32 *ptr_ms_used,
50*15dc779aSAndroid Build Coastguard Worker const WORD32 sfb_count, const WORD32 sfb_per_group,
51*15dc779aSAndroid Build Coastguard Worker const WORD32 max_sfb_per_grp, const WORD32 *ptr_sfb_offset,
52*15dc779aSAndroid Build Coastguard Worker FLOAT32 *ptr_weight_ms_lr_pe_ratio) {
53*15dc779aSAndroid Build Coastguard Worker FLOAT32 *ptr_sfb_enegry_left = ptr_psy_data[0]->sfb_energy.long_nrg;
54*15dc779aSAndroid Build Coastguard Worker FLOAT32 *ptr_sfb_energy_right = ptr_psy_data[1]->sfb_energy.long_nrg;
55*15dc779aSAndroid Build Coastguard Worker const FLOAT32 *ptr_sfb_energy_mid = ptr_psy_data[0]->sfb_energy_ms.long_nrg;
56*15dc779aSAndroid Build Coastguard Worker const FLOAT32 *ptr_sfb_energy_side = ptr_psy_data[1]->sfb_energy_ms.long_nrg;
57*15dc779aSAndroid Build Coastguard Worker FLOAT32 *ptr_sfb_thr_left = ptr_psy_data[0]->sfb_threshold.long_nrg;
58*15dc779aSAndroid Build Coastguard Worker FLOAT32 *ptr_sfb_thr_right = ptr_psy_data[1]->sfb_threshold.long_nrg;
59*15dc779aSAndroid Build Coastguard Worker FLOAT32 *ptr_sfb_spread_energy_left = ptr_psy_data[0]->sfb_sreaded_energy.long_nrg;
60*15dc779aSAndroid Build Coastguard Worker FLOAT32 *ptr_sfb_spread_energy_right = ptr_psy_data[1]->sfb_sreaded_energy.long_nrg;
61*15dc779aSAndroid Build Coastguard Worker WORD32 sfb, sfb_offsets, j;
62*15dc779aSAndroid Build Coastguard Worker WORD32 grp = 0;
63*15dc779aSAndroid Build Coastguard Worker WORD32 ms_counter = 0;
64*15dc779aSAndroid Build Coastguard Worker WORD32 lr_counter = 0;
65*15dc779aSAndroid Build Coastguard Worker FLOAT32 sum_ss_sr_pe_ratio = 0;
66*15dc779aSAndroid Build Coastguard Worker WORD32 cnt = 0;
67*15dc779aSAndroid Build Coastguard Worker FLOAT32 atan_val;
68*15dc779aSAndroid Build Coastguard Worker *ptr_ms_select = 0;
69*15dc779aSAndroid Build Coastguard Worker
70*15dc779aSAndroid Build Coastguard Worker for (sfb = 0; sfb < sfb_count; sfb += sfb_per_group, grp++) {
71*15dc779aSAndroid Build Coastguard Worker for (sfb_offsets = 0; sfb_offsets < max_sfb_per_grp; sfb_offsets++) {
72*15dc779aSAndroid Build Coastguard Worker FLOAT32 left_right, mid_side, min_thr;
73*15dc779aSAndroid Build Coastguard Worker WORD32 use_ms;
74*15dc779aSAndroid Build Coastguard Worker ptr_ms_used[sfb + sfb_offsets] = 0;
75*15dc779aSAndroid Build Coastguard Worker
76*15dc779aSAndroid Build Coastguard Worker min_thr = MIN(ptr_sfb_thr_left[sfb + sfb_offsets], ptr_sfb_thr_right[sfb + sfb_offsets]);
77*15dc779aSAndroid Build Coastguard Worker
78*15dc779aSAndroid Build Coastguard Worker left_right =
79*15dc779aSAndroid Build Coastguard Worker (ptr_sfb_thr_left[sfb + sfb_offsets] /
80*15dc779aSAndroid Build Coastguard Worker MAX(ptr_sfb_enegry_left[sfb + sfb_offsets], ptr_sfb_thr_left[sfb + sfb_offsets])) *
81*15dc779aSAndroid Build Coastguard Worker (ptr_sfb_thr_right[sfb + sfb_offsets] /
82*15dc779aSAndroid Build Coastguard Worker MAX(ptr_sfb_energy_right[sfb + sfb_offsets], ptr_sfb_thr_right[sfb + sfb_offsets]));
83*15dc779aSAndroid Build Coastguard Worker
84*15dc779aSAndroid Build Coastguard Worker mid_side = (min_thr / MAX(ptr_sfb_energy_mid[sfb + sfb_offsets], min_thr)) *
85*15dc779aSAndroid Build Coastguard Worker (min_thr / MAX(ptr_sfb_energy_side[sfb + sfb_offsets], min_thr));
86*15dc779aSAndroid Build Coastguard Worker
87*15dc779aSAndroid Build Coastguard Worker sum_ss_sr_pe_ratio += (left_right + 1.0e-9f) / (mid_side + 1.0e-9f);
88*15dc779aSAndroid Build Coastguard Worker cnt++;
89*15dc779aSAndroid Build Coastguard Worker use_ms = (mid_side >= left_right);
90*15dc779aSAndroid Build Coastguard Worker
91*15dc779aSAndroid Build Coastguard Worker if (use_ms) {
92*15dc779aSAndroid Build Coastguard Worker ptr_ms_used[sfb + sfb_offsets] = 1;
93*15dc779aSAndroid Build Coastguard Worker
94*15dc779aSAndroid Build Coastguard Worker for (j = ptr_sfb_offset[sfb + sfb_offsets]; j < ptr_sfb_offset[sfb + sfb_offsets + 1];
95*15dc779aSAndroid Build Coastguard Worker j++) {
96*15dc779aSAndroid Build Coastguard Worker FLOAT32 tmp = ptr_spec_left[j];
97*15dc779aSAndroid Build Coastguard Worker
98*15dc779aSAndroid Build Coastguard Worker ptr_spec_left[j] = 0.5f * (ptr_spec_left[j] + ptr_spec_right[j]);
99*15dc779aSAndroid Build Coastguard Worker
100*15dc779aSAndroid Build Coastguard Worker ptr_spec_right[j] = 0.5f * (tmp - ptr_spec_right[j]);
101*15dc779aSAndroid Build Coastguard Worker }
102*15dc779aSAndroid Build Coastguard Worker
103*15dc779aSAndroid Build Coastguard Worker ptr_sfb_thr_left[sfb + sfb_offsets] = ptr_sfb_thr_right[sfb + sfb_offsets] = min_thr;
104*15dc779aSAndroid Build Coastguard Worker
105*15dc779aSAndroid Build Coastguard Worker ptr_sfb_enegry_left[sfb + sfb_offsets] = ptr_sfb_energy_mid[sfb + sfb_offsets];
106*15dc779aSAndroid Build Coastguard Worker ptr_sfb_energy_right[sfb + sfb_offsets] = ptr_sfb_energy_side[sfb + sfb_offsets];
107*15dc779aSAndroid Build Coastguard Worker
108*15dc779aSAndroid Build Coastguard Worker ptr_sfb_spread_energy_left[sfb + sfb_offsets] =
109*15dc779aSAndroid Build Coastguard Worker ptr_sfb_spread_energy_right[sfb + sfb_offsets] =
110*15dc779aSAndroid Build Coastguard Worker MIN(ptr_sfb_spread_energy_left[sfb + sfb_offsets],
111*15dc779aSAndroid Build Coastguard Worker ptr_sfb_spread_energy_right[sfb + sfb_offsets]) *
112*15dc779aSAndroid Build Coastguard Worker 0.5f;
113*15dc779aSAndroid Build Coastguard Worker
114*15dc779aSAndroid Build Coastguard Worker ms_counter++;
115*15dc779aSAndroid Build Coastguard Worker } else {
116*15dc779aSAndroid Build Coastguard Worker ptr_ms_used[sfb + sfb_offsets] = 0;
117*15dc779aSAndroid Build Coastguard Worker lr_counter++;
118*15dc779aSAndroid Build Coastguard Worker }
119*15dc779aSAndroid Build Coastguard Worker }
120*15dc779aSAndroid Build Coastguard Worker }
121*15dc779aSAndroid Build Coastguard Worker
122*15dc779aSAndroid Build Coastguard Worker if (ms_counter == 0) {
123*15dc779aSAndroid Build Coastguard Worker *ptr_ms_select = 0;
124*15dc779aSAndroid Build Coastguard Worker } else {
125*15dc779aSAndroid Build Coastguard Worker if (lr_counter != 0) {
126*15dc779aSAndroid Build Coastguard Worker *ptr_ms_select = 1;
127*15dc779aSAndroid Build Coastguard Worker } else {
128*15dc779aSAndroid Build Coastguard Worker *ptr_ms_select = 2;
129*15dc779aSAndroid Build Coastguard Worker }
130*15dc779aSAndroid Build Coastguard Worker }
131*15dc779aSAndroid Build Coastguard Worker
132*15dc779aSAndroid Build Coastguard Worker cnt = MAX(1, cnt);
133*15dc779aSAndroid Build Coastguard Worker
134*15dc779aSAndroid Build Coastguard Worker atan_val = iaace_atan_approx((FLOAT32)(0.37f * (sum_ss_sr_pe_ratio / cnt - 6.5f)));
135*15dc779aSAndroid Build Coastguard Worker
136*15dc779aSAndroid Build Coastguard Worker *ptr_weight_ms_lr_pe_ratio = (FLOAT32)((0.28f * atan_val) + 1.25f);
137*15dc779aSAndroid Build Coastguard Worker }
138