xref: /aosp_15_r20/external/libxaac/encoder/iusace_ms.c (revision 15dc779a375ca8b5125643b829a8aa4b70d7f451)
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 "iusace_cnst.h"
24*15dc779aSAndroid Build Coastguard Worker #include "iusace_bitbuffer.h"
25*15dc779aSAndroid Build Coastguard Worker #include "iusace_tns_usac.h"
26*15dc779aSAndroid Build Coastguard Worker #include "iusace_psy_mod.h"
27*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_basic_ops32.h"
28*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_basic_ops40.h"
29*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_basic_ops.h"
30*15dc779aSAndroid Build Coastguard Worker 
iusace_ms_apply(ia_psy_mod_data_struct * pstr_psy_data,FLOAT64 * ptr_spec_left,FLOAT64 * ptr_spec_right,WORD32 * ms_select,WORD32 ms_used[MAX_SHORT_WINDOWS][MAX_SFB_LONG],const WORD32 sfb_count,const WORD32 sfb_per_group,const WORD32 max_sfb_per_grp,const WORD32 * ptr_sfb_offsets,WORD32 chn,FLOAT64 * ptr_ms_spec)31*15dc779aSAndroid Build Coastguard Worker VOID iusace_ms_apply(ia_psy_mod_data_struct *pstr_psy_data, FLOAT64 *ptr_spec_left,
32*15dc779aSAndroid Build Coastguard Worker                      FLOAT64 *ptr_spec_right, WORD32 *ms_select,
33*15dc779aSAndroid Build Coastguard Worker                      WORD32 ms_used[MAX_SHORT_WINDOWS][MAX_SFB_LONG], const WORD32 sfb_count,
34*15dc779aSAndroid Build Coastguard Worker                      const WORD32 sfb_per_group, const WORD32 max_sfb_per_grp,
35*15dc779aSAndroid Build Coastguard Worker                      const WORD32 *ptr_sfb_offsets, WORD32 chn, FLOAT64 *ptr_ms_spec) {
36*15dc779aSAndroid Build Coastguard Worker   FLOAT32 *ptr_sfb_enegry_left = pstr_psy_data[chn].ptr_sfb_energy_long;
37*15dc779aSAndroid Build Coastguard Worker   FLOAT32 *ptr_sfb_energy_right = pstr_psy_data[chn + 1].ptr_sfb_energy_long;
38*15dc779aSAndroid Build Coastguard Worker   const FLOAT32 *ptr_sfb_energy_mid = pstr_psy_data[chn].ptr_sfb_energy_long_ms;
39*15dc779aSAndroid Build Coastguard Worker   const FLOAT32 *ptr_sfb_energy_side = pstr_psy_data[chn + 1].ptr_sfb_energy_long_ms;
40*15dc779aSAndroid Build Coastguard Worker   FLOAT32 *ptr_sfb_thr_left = pstr_psy_data[chn].ptr_sfb_thr_long;
41*15dc779aSAndroid Build Coastguard Worker   FLOAT32 *ptr_sfb_thr_right = pstr_psy_data[chn + 1].ptr_sfb_thr_long;
42*15dc779aSAndroid Build Coastguard Worker   FLOAT32 *ptr_sfb_spread_energy_left = pstr_psy_data[chn].ptr_sfb_spreaded_energy_long;
43*15dc779aSAndroid Build Coastguard Worker   FLOAT32 *ptr_sfb_spread_energy_right = pstr_psy_data[chn + 1].ptr_sfb_spreaded_energy_long;
44*15dc779aSAndroid Build Coastguard Worker   WORD32 sfb, sfb_offsets, j;
45*15dc779aSAndroid Build Coastguard Worker   WORD32 grp = 0;
46*15dc779aSAndroid Build Coastguard Worker   WORD32 ms_counter = 0;
47*15dc779aSAndroid Build Coastguard Worker   WORD32 lr_counter = 0;
48*15dc779aSAndroid Build Coastguard Worker 
49*15dc779aSAndroid Build Coastguard Worker   *ms_select = 0;
50*15dc779aSAndroid Build Coastguard Worker 
51*15dc779aSAndroid Build Coastguard Worker   for (sfb = 0; sfb < sfb_count; sfb += sfb_per_group, grp++) {
52*15dc779aSAndroid Build Coastguard Worker     for (sfb_offsets = 0; sfb_offsets < max_sfb_per_grp; sfb_offsets++) {
53*15dc779aSAndroid Build Coastguard Worker       FLOAT32 left_right, mid_side, min_thr;
54*15dc779aSAndroid Build Coastguard Worker       WORD32 use_ms;
55*15dc779aSAndroid Build Coastguard Worker       ms_used[grp][sfb_offsets] = 0;
56*15dc779aSAndroid Build Coastguard Worker 
57*15dc779aSAndroid Build Coastguard Worker       min_thr = MIN(ptr_sfb_thr_left[sfb + sfb_offsets], ptr_sfb_thr_right[sfb + sfb_offsets]);
58*15dc779aSAndroid Build Coastguard Worker 
59*15dc779aSAndroid Build Coastguard Worker       left_right =
60*15dc779aSAndroid Build Coastguard Worker           (ptr_sfb_thr_left[sfb + sfb_offsets] /
61*15dc779aSAndroid Build Coastguard Worker            MAX(ptr_sfb_enegry_left[sfb + sfb_offsets], ptr_sfb_thr_left[sfb + sfb_offsets])) *
62*15dc779aSAndroid Build Coastguard Worker           (ptr_sfb_thr_right[sfb + sfb_offsets] /
63*15dc779aSAndroid Build Coastguard Worker            max(ptr_sfb_energy_right[sfb + sfb_offsets], ptr_sfb_thr_right[sfb + sfb_offsets]));
64*15dc779aSAndroid Build Coastguard Worker 
65*15dc779aSAndroid Build Coastguard Worker       mid_side = (min_thr / max(ptr_sfb_energy_mid[sfb + sfb_offsets], min_thr)) *
66*15dc779aSAndroid Build Coastguard Worker                  (min_thr / max(ptr_sfb_energy_side[sfb + sfb_offsets], min_thr));
67*15dc779aSAndroid Build Coastguard Worker 
68*15dc779aSAndroid Build Coastguard Worker       use_ms = (mid_side >= left_right);
69*15dc779aSAndroid Build Coastguard Worker 
70*15dc779aSAndroid Build Coastguard Worker       if (use_ms) {
71*15dc779aSAndroid Build Coastguard Worker         ms_used[grp][sfb_offsets] = 1;
72*15dc779aSAndroid Build Coastguard Worker 
73*15dc779aSAndroid Build Coastguard Worker         for (j = ptr_sfb_offsets[sfb + sfb_offsets]; j < ptr_sfb_offsets[sfb + sfb_offsets + 1];
74*15dc779aSAndroid Build Coastguard Worker              j++) {
75*15dc779aSAndroid Build Coastguard Worker           if (ptr_ms_spec != NULL) {
76*15dc779aSAndroid Build Coastguard Worker             ptr_spec_left[j] = ptr_ms_spec[j];
77*15dc779aSAndroid Build Coastguard Worker             ptr_spec_right[j] = ptr_ms_spec[1024 + j];
78*15dc779aSAndroid Build Coastguard Worker           } else {
79*15dc779aSAndroid Build Coastguard Worker             FLOAT64 tmp = ptr_spec_left[j];
80*15dc779aSAndroid Build Coastguard Worker 
81*15dc779aSAndroid Build Coastguard Worker             ptr_spec_left[j] = 0.5f * (ptr_spec_left[j] + ptr_spec_right[j]);
82*15dc779aSAndroid Build Coastguard Worker 
83*15dc779aSAndroid Build Coastguard Worker             ptr_spec_right[j] = 0.5f * (tmp - ptr_spec_right[j]);
84*15dc779aSAndroid Build Coastguard Worker           }
85*15dc779aSAndroid Build Coastguard Worker         }
86*15dc779aSAndroid Build Coastguard Worker 
87*15dc779aSAndroid Build Coastguard Worker         ptr_sfb_thr_left[sfb + sfb_offsets] = ptr_sfb_thr_right[sfb + sfb_offsets] = min_thr;
88*15dc779aSAndroid Build Coastguard Worker 
89*15dc779aSAndroid Build Coastguard Worker         ptr_sfb_enegry_left[sfb + sfb_offsets] = ptr_sfb_energy_mid[sfb + sfb_offsets];
90*15dc779aSAndroid Build Coastguard Worker         ptr_sfb_energy_right[sfb + sfb_offsets] = ptr_sfb_energy_side[sfb + sfb_offsets];
91*15dc779aSAndroid Build Coastguard Worker 
92*15dc779aSAndroid Build Coastguard Worker         ptr_sfb_spread_energy_left[sfb + sfb_offsets] =
93*15dc779aSAndroid Build Coastguard Worker             ptr_sfb_spread_energy_right[sfb + sfb_offsets] =
94*15dc779aSAndroid Build Coastguard Worker                 min(ptr_sfb_spread_energy_left[sfb + sfb_offsets],
95*15dc779aSAndroid Build Coastguard Worker                     ptr_sfb_spread_energy_right[sfb + sfb_offsets]) *
96*15dc779aSAndroid Build Coastguard Worker                 0.5f;
97*15dc779aSAndroid Build Coastguard Worker 
98*15dc779aSAndroid Build Coastguard Worker         ms_counter++;
99*15dc779aSAndroid Build Coastguard Worker       } else {
100*15dc779aSAndroid Build Coastguard Worker         ms_used[grp][sfb_offsets] = 0;
101*15dc779aSAndroid Build Coastguard Worker         lr_counter++;
102*15dc779aSAndroid Build Coastguard Worker       }
103*15dc779aSAndroid Build Coastguard Worker     }
104*15dc779aSAndroid Build Coastguard Worker   }
105*15dc779aSAndroid Build Coastguard Worker 
106*15dc779aSAndroid Build Coastguard Worker   if (ms_counter == 0) {
107*15dc779aSAndroid Build Coastguard Worker     *ms_select = 0;
108*15dc779aSAndroid Build Coastguard Worker   } else {
109*15dc779aSAndroid Build Coastguard Worker     if (lr_counter != 0) {
110*15dc779aSAndroid Build Coastguard Worker       *ms_select = 1;
111*15dc779aSAndroid Build Coastguard Worker     } else {
112*15dc779aSAndroid Build Coastguard Worker       *ms_select = 2;
113*15dc779aSAndroid Build Coastguard Worker     }
114*15dc779aSAndroid Build Coastguard Worker   }
115*15dc779aSAndroid Build Coastguard Worker   return;
116*15dc779aSAndroid Build Coastguard Worker }
117*15dc779aSAndroid Build Coastguard Worker 
iusace_calc_ms_band_energy(const FLOAT64 * ptr_spec_left,const FLOAT64 * ptr_spec_right,const WORD32 * ptr_band_offset,const WORD32 num_bands,FLOAT32 * ptr_band_energy_mid,FLOAT32 * ptr_band_energy_side)118*15dc779aSAndroid Build Coastguard Worker VOID iusace_calc_ms_band_energy(const FLOAT64 *ptr_spec_left, const FLOAT64 *ptr_spec_right,
119*15dc779aSAndroid Build Coastguard Worker                                 const WORD32 *ptr_band_offset, const WORD32 num_bands,
120*15dc779aSAndroid Build Coastguard Worker                                 FLOAT32 *ptr_band_energy_mid, FLOAT32 *ptr_band_energy_side) {
121*15dc779aSAndroid Build Coastguard Worker   WORD32 i, j;
122*15dc779aSAndroid Build Coastguard Worker 
123*15dc779aSAndroid Build Coastguard Worker   j = 0;
124*15dc779aSAndroid Build Coastguard Worker   for (i = 0; i < num_bands; i++) {
125*15dc779aSAndroid Build Coastguard Worker     ptr_band_energy_mid[i] = 0.0f;
126*15dc779aSAndroid Build Coastguard Worker     ptr_band_energy_side[i] = 0.0f;
127*15dc779aSAndroid Build Coastguard Worker 
128*15dc779aSAndroid Build Coastguard Worker     while (j < ptr_band_offset[i + 1]) {
129*15dc779aSAndroid Build Coastguard Worker       FLOAT32 specm, specs;
130*15dc779aSAndroid Build Coastguard Worker 
131*15dc779aSAndroid Build Coastguard Worker       specm = (FLOAT32)(0.5f * (ptr_spec_left[j] + ptr_spec_right[j]));
132*15dc779aSAndroid Build Coastguard Worker       specs = (FLOAT32)(0.5f * (ptr_spec_left[j] - ptr_spec_right[j]));
133*15dc779aSAndroid Build Coastguard Worker 
134*15dc779aSAndroid Build Coastguard Worker       ptr_band_energy_mid[i] += specm * specm;
135*15dc779aSAndroid Build Coastguard Worker       ptr_band_energy_side[i] += specs * specs;
136*15dc779aSAndroid Build Coastguard Worker 
137*15dc779aSAndroid Build Coastguard Worker       j++;
138*15dc779aSAndroid Build Coastguard Worker     }
139*15dc779aSAndroid Build Coastguard Worker   }
140*15dc779aSAndroid Build Coastguard Worker 
141*15dc779aSAndroid Build Coastguard Worker   return;
142*15dc779aSAndroid Build Coastguard Worker }
143