xref: /aosp_15_r20/external/libxaac/encoder/ixheaace_tns_hp.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 <math.h>
22*15dc779aSAndroid Build Coastguard Worker #include <stddef.h>
23*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_type_def.h"
24*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_constants.h"
25*15dc779aSAndroid Build Coastguard Worker #include "impd_drc_common_enc.h"
26*15dc779aSAndroid Build Coastguard Worker #include "impd_drc_uni_drc.h"
27*15dc779aSAndroid Build Coastguard Worker #include "impd_drc_tables.h"
28*15dc779aSAndroid Build Coastguard Worker #include "impd_drc_api.h"
29*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_api.h"
30*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_aac_constants.h"
31*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_common_rom.h"
32*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_psy_const.h"
33*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_tns.h"
34*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_tns_params.h"
35*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_rom.h"
36*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_bitbuffer.h"
37*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_psy_configuration.h"
38*15dc779aSAndroid Build Coastguard Worker #include "ixheaace_tns_func.h"
39*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_basic_ops32.h"
40*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_basic_ops16.h"
41*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_basic_ops40.h"
42*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_basic_ops.h"
43*15dc779aSAndroid Build Coastguard Worker 
ia_enhaacplus_enc_calc_weighted_spectrum(FLOAT32 * ptr_spectrum,FLOAT32 * ptr_weighted_spec,FLOAT32 * ptr_sfb_energy,const WORD32 * ptr_sfb_offset,WORD32 lpc_start_line,WORD32 lpc_stop_line,WORD32 lpc_start_band,WORD32 lpc_stop_band,FLOAT32 * ptr_shared_buffer1,WORD32 aot)44*15dc779aSAndroid Build Coastguard Worker VOID ia_enhaacplus_enc_calc_weighted_spectrum(FLOAT32 *ptr_spectrum, FLOAT32 *ptr_weighted_spec,
45*15dc779aSAndroid Build Coastguard Worker                                               FLOAT32 *ptr_sfb_energy,
46*15dc779aSAndroid Build Coastguard Worker                                               const WORD32 *ptr_sfb_offset, WORD32 lpc_start_line,
47*15dc779aSAndroid Build Coastguard Worker                                               WORD32 lpc_stop_line, WORD32 lpc_start_band,
48*15dc779aSAndroid Build Coastguard Worker                                               WORD32 lpc_stop_band, FLOAT32 *ptr_shared_buffer1,
49*15dc779aSAndroid Build Coastguard Worker                                               WORD32 aot) {
50*15dc779aSAndroid Build Coastguard Worker   WORD32 i, sfb, tempcnt;
51*15dc779aSAndroid Build Coastguard Worker   FLOAT32 tmp;
52*15dc779aSAndroid Build Coastguard Worker   FLOAT32 *ptr_tns_sfb_mean = ptr_shared_buffer1;
53*15dc779aSAndroid Build Coastguard Worker   FLOAT32 temp1, temp2;
54*15dc779aSAndroid Build Coastguard Worker   FLOAT32 *ptr_spec;
55*15dc779aSAndroid Build Coastguard Worker   FLOAT32 *ptr_ws1;
56*15dc779aSAndroid Build Coastguard Worker   WORD sfb_width, j;
57*15dc779aSAndroid Build Coastguard Worker 
58*15dc779aSAndroid Build Coastguard Worker   for (sfb = lpc_start_band; sfb < lpc_stop_band; sfb++) {
59*15dc779aSAndroid Build Coastguard Worker     FLOAT32 sfb_nrg_tmp = ptr_sfb_energy[sfb];
60*15dc779aSAndroid Build Coastguard Worker     ptr_tns_sfb_mean[sfb] = 1 / ((FLOAT32)sqrt(sfb_nrg_tmp) + 1e-30f);
61*15dc779aSAndroid Build Coastguard Worker   }
62*15dc779aSAndroid Build Coastguard Worker 
63*15dc779aSAndroid Build Coastguard Worker   sfb = lpc_start_band;
64*15dc779aSAndroid Build Coastguard Worker 
65*15dc779aSAndroid Build Coastguard Worker   tmp = ptr_tns_sfb_mean[sfb];
66*15dc779aSAndroid Build Coastguard Worker 
67*15dc779aSAndroid Build Coastguard Worker   for (i = lpc_start_line; i < lpc_stop_line; i += sfb_width) {
68*15dc779aSAndroid Build Coastguard Worker     ptr_spec = &ptr_weighted_spec[i];
69*15dc779aSAndroid Build Coastguard Worker     WORD start = i, stop = ptr_sfb_offset[sfb + 1];
70*15dc779aSAndroid Build Coastguard Worker 
71*15dc779aSAndroid Build Coastguard Worker     stop = MIN(stop, lpc_stop_line);
72*15dc779aSAndroid Build Coastguard Worker     sfb_width = stop - start;
73*15dc779aSAndroid Build Coastguard Worker 
74*15dc779aSAndroid Build Coastguard Worker     for (j = (sfb_width >> 1) - 1; j >= 0; j--) {
75*15dc779aSAndroid Build Coastguard Worker       *ptr_spec++ = tmp;
76*15dc779aSAndroid Build Coastguard Worker       *ptr_spec++ = tmp;
77*15dc779aSAndroid Build Coastguard Worker     }
78*15dc779aSAndroid Build Coastguard Worker     sfb++;
79*15dc779aSAndroid Build Coastguard Worker 
80*15dc779aSAndroid Build Coastguard Worker     if ((sfb + 1) < lpc_stop_band) {
81*15dc779aSAndroid Build Coastguard Worker       tmp = ptr_tns_sfb_mean[sfb];
82*15dc779aSAndroid Build Coastguard Worker     }
83*15dc779aSAndroid Build Coastguard Worker   }
84*15dc779aSAndroid Build Coastguard Worker 
85*15dc779aSAndroid Build Coastguard Worker   /* Filter down */
86*15dc779aSAndroid Build Coastguard Worker   if (aot == AOT_AAC_LC || aot == AOT_SBR || aot == AOT_PS || aot == AOT_AAC_ELD) {
87*15dc779aSAndroid Build Coastguard Worker     for (i = lpc_stop_line - 2; i >= lpc_start_line; i--) {
88*15dc779aSAndroid Build Coastguard Worker       ptr_weighted_spec[i] = (ptr_weighted_spec[i] + ptr_weighted_spec[i + 1]) * 0.5f;
89*15dc779aSAndroid Build Coastguard Worker     }
90*15dc779aSAndroid Build Coastguard Worker     for (i = lpc_start_line + 1; i < lpc_stop_line; i++) {
91*15dc779aSAndroid Build Coastguard Worker       ptr_weighted_spec[i] = (ptr_weighted_spec[i] + ptr_weighted_spec[i - 1]) * 0.5f;
92*15dc779aSAndroid Build Coastguard Worker     }
93*15dc779aSAndroid Build Coastguard Worker 
94*15dc779aSAndroid Build Coastguard Worker     /* Weight and normalize */
95*15dc779aSAndroid Build Coastguard Worker     for (i = lpc_start_line; i < lpc_stop_line; i++) {
96*15dc779aSAndroid Build Coastguard Worker       ptr_weighted_spec[i] = ptr_weighted_spec[i] * ptr_spectrum[i];
97*15dc779aSAndroid Build Coastguard Worker     }
98*15dc779aSAndroid Build Coastguard Worker   } else if (aot == AOT_AAC_LD || aot == AOT_AAC_ELD) {
99*15dc779aSAndroid Build Coastguard Worker     WORD32 remaining;
100*15dc779aSAndroid Build Coastguard Worker     FLOAT32 multout_temp;
101*15dc779aSAndroid Build Coastguard Worker 
102*15dc779aSAndroid Build Coastguard Worker     ptr_ws1 = &ptr_weighted_spec[lpc_stop_line - 1];
103*15dc779aSAndroid Build Coastguard Worker     tempcnt = (lpc_stop_line - lpc_start_line) >> 2;
104*15dc779aSAndroid Build Coastguard Worker     remaining = lpc_stop_line - lpc_start_line - (tempcnt << 2);
105*15dc779aSAndroid Build Coastguard Worker 
106*15dc779aSAndroid Build Coastguard Worker     temp1 = *ptr_ws1--;
107*15dc779aSAndroid Build Coastguard Worker     temp2 = (*ptr_ws1 + temp1);
108*15dc779aSAndroid Build Coastguard Worker     for (i = tempcnt - 1; i >= 0; i--) {
109*15dc779aSAndroid Build Coastguard Worker       *ptr_ws1-- = temp2;
110*15dc779aSAndroid Build Coastguard Worker       temp1 = (*ptr_ws1 + temp2 * 0.5f);
111*15dc779aSAndroid Build Coastguard Worker       *ptr_ws1-- = temp1;
112*15dc779aSAndroid Build Coastguard Worker       temp2 = (*ptr_ws1 + temp1 * 0.5f);
113*15dc779aSAndroid Build Coastguard Worker       *ptr_ws1-- = temp2;
114*15dc779aSAndroid Build Coastguard Worker       temp1 = (*ptr_ws1 + temp2 * 0.5f);
115*15dc779aSAndroid Build Coastguard Worker       *ptr_ws1-- = temp1;
116*15dc779aSAndroid Build Coastguard Worker       temp2 = (*ptr_ws1 + temp1 * 0.5f);
117*15dc779aSAndroid Build Coastguard Worker     }
118*15dc779aSAndroid Build Coastguard Worker     ptr_ws1++;
119*15dc779aSAndroid Build Coastguard Worker     if (remaining) {
120*15dc779aSAndroid Build Coastguard Worker       for (i = remaining - 1; i >= 0; i--) {
121*15dc779aSAndroid Build Coastguard Worker         temp1 = *ptr_ws1--;
122*15dc779aSAndroid Build Coastguard Worker         *ptr_ws1 = (*ptr_ws1 + temp1 * 0.5f);
123*15dc779aSAndroid Build Coastguard Worker       }
124*15dc779aSAndroid Build Coastguard Worker     }
125*15dc779aSAndroid Build Coastguard Worker 
126*15dc779aSAndroid Build Coastguard Worker     ptr_weighted_spec[lpc_start_line + 1] = (FLOAT32)(
127*15dc779aSAndroid Build Coastguard Worker         ((ptr_weighted_spec[lpc_start_line + 1]) + (ptr_weighted_spec[lpc_start_line])) * 0.5f);
128*15dc779aSAndroid Build Coastguard Worker     multout_temp = (ptr_weighted_spec[lpc_start_line] * ptr_spectrum[lpc_start_line]);
129*15dc779aSAndroid Build Coastguard Worker     ptr_weighted_spec[lpc_start_line] = multout_temp;
130*15dc779aSAndroid Build Coastguard Worker 
131*15dc779aSAndroid Build Coastguard Worker     /* Weight and normalize */
132*15dc779aSAndroid Build Coastguard Worker     ptr_spec = &ptr_spectrum[lpc_start_line + 1];
133*15dc779aSAndroid Build Coastguard Worker     ptr_ws1 = &ptr_weighted_spec[lpc_start_line + 1];
134*15dc779aSAndroid Build Coastguard Worker 
135*15dc779aSAndroid Build Coastguard Worker     tempcnt = (lpc_stop_line - lpc_start_line - 2) >> 2;
136*15dc779aSAndroid Build Coastguard Worker     remaining = (lpc_stop_line - lpc_start_line - 2) - (tempcnt << 2);
137*15dc779aSAndroid Build Coastguard Worker     temp2 = *ptr_ws1;
138*15dc779aSAndroid Build Coastguard Worker 
139*15dc779aSAndroid Build Coastguard Worker     for (i = tempcnt - 1; i >= 0; i--) {
140*15dc779aSAndroid Build Coastguard Worker       temp1 = *(ptr_ws1 + 1);
141*15dc779aSAndroid Build Coastguard Worker       temp1 = (FLOAT32)((temp1 + temp2) * 0.5f);
142*15dc779aSAndroid Build Coastguard Worker       multout_temp = (temp2 * *ptr_spec++);
143*15dc779aSAndroid Build Coastguard Worker       *ptr_ws1++ = multout_temp;
144*15dc779aSAndroid Build Coastguard Worker 
145*15dc779aSAndroid Build Coastguard Worker       temp2 = *(ptr_ws1 + 1);
146*15dc779aSAndroid Build Coastguard Worker       temp2 = (FLOAT32)((temp2 + temp1) * 0.5f);
147*15dc779aSAndroid Build Coastguard Worker       multout_temp = (temp1 * *ptr_spec++);
148*15dc779aSAndroid Build Coastguard Worker       *ptr_ws1++ = multout_temp;
149*15dc779aSAndroid Build Coastguard Worker 
150*15dc779aSAndroid Build Coastguard Worker       temp1 = *(ptr_ws1 + 1);
151*15dc779aSAndroid Build Coastguard Worker       temp1 = (FLOAT32)((temp2 + temp1) * 0.5f);
152*15dc779aSAndroid Build Coastguard Worker       multout_temp = (temp2 * *ptr_spec++);
153*15dc779aSAndroid Build Coastguard Worker       *ptr_ws1++ = multout_temp;
154*15dc779aSAndroid Build Coastguard Worker 
155*15dc779aSAndroid Build Coastguard Worker       temp2 = *(ptr_ws1 + 1);
156*15dc779aSAndroid Build Coastguard Worker       temp2 = (FLOAT32)((temp2 + temp1) * 0.5f);
157*15dc779aSAndroid Build Coastguard Worker       multout_temp = (temp1 * *ptr_spec++);
158*15dc779aSAndroid Build Coastguard Worker       *ptr_ws1++ = multout_temp;
159*15dc779aSAndroid Build Coastguard Worker     }
160*15dc779aSAndroid Build Coastguard Worker 
161*15dc779aSAndroid Build Coastguard Worker     if (remaining) {
162*15dc779aSAndroid Build Coastguard Worker       for (i = remaining - 1; i >= 0; i--) {
163*15dc779aSAndroid Build Coastguard Worker         temp1 = *(ptr_ws1 + 1);
164*15dc779aSAndroid Build Coastguard Worker 
165*15dc779aSAndroid Build Coastguard Worker         multout_temp = (temp2 * *ptr_spec++);
166*15dc779aSAndroid Build Coastguard Worker         *ptr_ws1++ = multout_temp;
167*15dc779aSAndroid Build Coastguard Worker         temp2 = (FLOAT32)((temp1 + temp2) * 0.5f);
168*15dc779aSAndroid Build Coastguard Worker       }
169*15dc779aSAndroid Build Coastguard Worker     }
170*15dc779aSAndroid Build Coastguard Worker 
171*15dc779aSAndroid Build Coastguard Worker     multout_temp = (temp2 + ptr_spectrum[lpc_stop_line - 1]);
172*15dc779aSAndroid Build Coastguard Worker 
173*15dc779aSAndroid Build Coastguard Worker     ptr_weighted_spec[lpc_stop_line - 1] = multout_temp;
174*15dc779aSAndroid Build Coastguard Worker   }
175*15dc779aSAndroid Build Coastguard Worker }
176*15dc779aSAndroid Build Coastguard Worker 
ia_enhaacplus_enc_auto_correlation(const FLOAT32 * ptr_input,FLOAT32 * ptr_corr,WORD32 samples,WORD32 corr_coeff)177*15dc779aSAndroid Build Coastguard Worker VOID ia_enhaacplus_enc_auto_correlation(const FLOAT32 *ptr_input, FLOAT32 *ptr_corr,
178*15dc779aSAndroid Build Coastguard Worker                                         WORD32 samples, WORD32 corr_coeff) {
179*15dc779aSAndroid Build Coastguard Worker   WORD32 i, j;
180*15dc779aSAndroid Build Coastguard Worker   FLOAT32 tmp_var;
181*15dc779aSAndroid Build Coastguard Worker   WORD32 remaining;
182*15dc779aSAndroid Build Coastguard Worker   remaining = corr_coeff - ((corr_coeff >> 1) << 1);
183*15dc779aSAndroid Build Coastguard Worker 
184*15dc779aSAndroid Build Coastguard Worker   for (i = 0; i < samples; i += 2) {
185*15dc779aSAndroid Build Coastguard Worker     const FLOAT32 *ptr_input1 = &ptr_input[i];
186*15dc779aSAndroid Build Coastguard Worker     FLOAT32 temp1 = *ptr_input1;
187*15dc779aSAndroid Build Coastguard Worker     FLOAT32 temp2 = *(ptr_input1 + 1);
188*15dc779aSAndroid Build Coastguard Worker     FLOAT32 inp_tmp1 = *ptr_input1++;
189*15dc779aSAndroid Build Coastguard Worker     for (j = 0; j < (corr_coeff >> 1) << 1; j++) {
190*15dc779aSAndroid Build Coastguard Worker       FLOAT32 inp_tmp2;
191*15dc779aSAndroid Build Coastguard Worker       tmp_var = (temp1 * inp_tmp1);
192*15dc779aSAndroid Build Coastguard Worker       inp_tmp2 = *ptr_input1++;
193*15dc779aSAndroid Build Coastguard Worker       tmp_var += (temp2 * inp_tmp2);
194*15dc779aSAndroid Build Coastguard Worker       ptr_corr[j] += tmp_var;
195*15dc779aSAndroid Build Coastguard Worker       j++;
196*15dc779aSAndroid Build Coastguard Worker       tmp_var = (temp1 * inp_tmp2);
197*15dc779aSAndroid Build Coastguard Worker       inp_tmp1 = *ptr_input1++;
198*15dc779aSAndroid Build Coastguard Worker       tmp_var += (temp2 * inp_tmp1);
199*15dc779aSAndroid Build Coastguard Worker       ptr_corr[j] += (tmp_var);
200*15dc779aSAndroid Build Coastguard Worker     }
201*15dc779aSAndroid Build Coastguard Worker     if (remaining) {
202*15dc779aSAndroid Build Coastguard Worker       tmp_var = (temp1 * inp_tmp1);
203*15dc779aSAndroid Build Coastguard Worker       tmp_var += (temp2 * *ptr_input1);
204*15dc779aSAndroid Build Coastguard Worker       ptr_corr[j] += (tmp_var);
205*15dc779aSAndroid Build Coastguard Worker     }
206*15dc779aSAndroid Build Coastguard Worker   }
207*15dc779aSAndroid Build Coastguard Worker }
208*15dc779aSAndroid Build Coastguard Worker 
ia_enhaacplus_enc_analysis_filter_lattice(const FLOAT32 * ptr_signal,WORD32 num_lines,const FLOAT32 * ptr_par_coeff,WORD32 order,FLOAT32 * ptr_output)209*15dc779aSAndroid Build Coastguard Worker VOID ia_enhaacplus_enc_analysis_filter_lattice(const FLOAT32 *ptr_signal, WORD32 num_lines,
210*15dc779aSAndroid Build Coastguard Worker                                                const FLOAT32 *ptr_par_coeff, WORD32 order,
211*15dc779aSAndroid Build Coastguard Worker                                                FLOAT32 *ptr_output) {
212*15dc779aSAndroid Build Coastguard Worker   FLOAT32 state_par[TEMPORAL_NOISE_SHAPING_MAX_ORDER] = {0};
213*15dc779aSAndroid Build Coastguard Worker   WORD32 j;
214*15dc779aSAndroid Build Coastguard Worker 
215*15dc779aSAndroid Build Coastguard Worker   if (order <= 0) {
216*15dc779aSAndroid Build Coastguard Worker     return;
217*15dc779aSAndroid Build Coastguard Worker   }
218*15dc779aSAndroid Build Coastguard Worker 
219*15dc779aSAndroid Build Coastguard Worker   for (j = 0; j < num_lines; j++) {
220*15dc779aSAndroid Build Coastguard Worker     WORD32 i;
221*15dc779aSAndroid Build Coastguard Worker     FLOAT32 x = ptr_signal[j];
222*15dc779aSAndroid Build Coastguard Worker     FLOAT32 accu, tmp, tmp_save;
223*15dc779aSAndroid Build Coastguard Worker 
224*15dc779aSAndroid Build Coastguard Worker     tmp_save = x;
225*15dc779aSAndroid Build Coastguard Worker     accu = x;
226*15dc779aSAndroid Build Coastguard Worker 
227*15dc779aSAndroid Build Coastguard Worker     for (i = 0; i < order - 1; i++) {
228*15dc779aSAndroid Build Coastguard Worker       tmp = (accu * ptr_par_coeff[i]);
229*15dc779aSAndroid Build Coastguard Worker 
230*15dc779aSAndroid Build Coastguard Worker       tmp += state_par[i];
231*15dc779aSAndroid Build Coastguard Worker 
232*15dc779aSAndroid Build Coastguard Worker       accu += (state_par[i] * ptr_par_coeff[i]);
233*15dc779aSAndroid Build Coastguard Worker 
234*15dc779aSAndroid Build Coastguard Worker       state_par[i] = tmp_save;
235*15dc779aSAndroid Build Coastguard Worker       tmp_save = tmp;
236*15dc779aSAndroid Build Coastguard Worker     }
237*15dc779aSAndroid Build Coastguard Worker 
238*15dc779aSAndroid Build Coastguard Worker     /* last stage: only need half operations */
239*15dc779aSAndroid Build Coastguard Worker     accu += (state_par[order - 1] * ptr_par_coeff[order - 1]);
240*15dc779aSAndroid Build Coastguard Worker 
241*15dc779aSAndroid Build Coastguard Worker     state_par[order - 1] = tmp_save;
242*15dc779aSAndroid Build Coastguard Worker 
243*15dc779aSAndroid Build Coastguard Worker     ptr_output[j] = accu;
244*15dc779aSAndroid Build Coastguard Worker   }
245*15dc779aSAndroid Build Coastguard Worker }
246