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 #include "ixheaac_type_def.h"
21*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_constants.h"
22*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_basic_ops32.h"
23*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_basic_ops16.h"
24*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_basic_ops40.h"
25*15dc779aSAndroid Build Coastguard Worker #include "ixheaac_basic_ops.h"
26*15dc779aSAndroid Build Coastguard Worker #include "ixheaacd_defines.h"
27*15dc779aSAndroid Build Coastguard Worker
ixheaacd_res_tns_parcor_2_lpc_32x16(WORD16 * parcor,WORD16 * lpc,WORD16 * scale,WORD order)28*15dc779aSAndroid Build Coastguard Worker VOID ixheaacd_res_tns_parcor_2_lpc_32x16(WORD16 *parcor, WORD16 *lpc, WORD16 *scale, WORD order)
29*15dc779aSAndroid Build Coastguard Worker
30*15dc779aSAndroid Build Coastguard Worker {
31*15dc779aSAndroid Build Coastguard Worker WORD i, j, status;
32*15dc779aSAndroid Build Coastguard Worker WORD32 z1;
33*15dc779aSAndroid Build Coastguard Worker WORD16 z[MAX_ORDER + 1];
34*15dc779aSAndroid Build Coastguard Worker WORD16 w[MAX_ORDER + 1];
35*15dc779aSAndroid Build Coastguard Worker WORD32 accu1, accu2;
36*15dc779aSAndroid Build Coastguard Worker
37*15dc779aSAndroid Build Coastguard Worker status = 1;
38*15dc779aSAndroid Build Coastguard Worker *scale = 0;
39*15dc779aSAndroid Build Coastguard Worker while (status) {
40*15dc779aSAndroid Build Coastguard Worker status = 0;
41*15dc779aSAndroid Build Coastguard Worker
42*15dc779aSAndroid Build Coastguard Worker for (i = MAX_ORDER; i >= 0; i--) {
43*15dc779aSAndroid Build Coastguard Worker z[i] = 0;
44*15dc779aSAndroid Build Coastguard Worker w[i] = 0;
45*15dc779aSAndroid Build Coastguard Worker }
46*15dc779aSAndroid Build Coastguard Worker
47*15dc779aSAndroid Build Coastguard Worker accu1 = (0x7fffffff >> *scale);
48*15dc779aSAndroid Build Coastguard Worker
49*15dc779aSAndroid Build Coastguard Worker for (i = 0; i <= order; i++) {
50*15dc779aSAndroid Build Coastguard Worker z1 = accu1;
51*15dc779aSAndroid Build Coastguard Worker
52*15dc779aSAndroid Build Coastguard Worker for (j = 0; j < order; j++) {
53*15dc779aSAndroid Build Coastguard Worker w[j] = ixheaac_round16(accu1);
54*15dc779aSAndroid Build Coastguard Worker
55*15dc779aSAndroid Build Coastguard Worker accu1 = ixheaac_mac16x16in32_shl_sat(accu1, parcor[j], z[j]);
56*15dc779aSAndroid Build Coastguard Worker if (ixheaac_abs32_sat(accu1) == 0x7fffffff) status = 1;
57*15dc779aSAndroid Build Coastguard Worker }
58*15dc779aSAndroid Build Coastguard Worker for (j = (order - 1); j >= 0; j--) {
59*15dc779aSAndroid Build Coastguard Worker accu2 = ixheaac_deposit16h_in32(z[j]);
60*15dc779aSAndroid Build Coastguard Worker accu2 = ixheaac_mac16x16in32_shl_sat(accu2, parcor[j], w[j]);
61*15dc779aSAndroid Build Coastguard Worker z[j + 1] = ixheaac_round16(accu2);
62*15dc779aSAndroid Build Coastguard Worker if (ixheaac_abs32_sat(accu2) == 0x7fffffff) status = 1;
63*15dc779aSAndroid Build Coastguard Worker }
64*15dc779aSAndroid Build Coastguard Worker
65*15dc779aSAndroid Build Coastguard Worker z[0] = ixheaac_round16(z1);
66*15dc779aSAndroid Build Coastguard Worker lpc[i] = ixheaac_round16(accu1);
67*15dc779aSAndroid Build Coastguard Worker accu1 = 0;
68*15dc779aSAndroid Build Coastguard Worker }
69*15dc779aSAndroid Build Coastguard Worker accu1 = (status - 1);
70*15dc779aSAndroid Build Coastguard Worker if (accu1 == 0) {
71*15dc779aSAndroid Build Coastguard Worker *scale = *scale + 1;
72*15dc779aSAndroid Build Coastguard Worker }
73*15dc779aSAndroid Build Coastguard Worker }
74*15dc779aSAndroid Build Coastguard Worker }
75*15dc779aSAndroid Build Coastguard Worker
ixheaacd_res_tns_ar_filter_fixed_32x16(WORD32 * spectrum,WORD32 size,WORD32 inc,WORD16 * lpc,WORD32 order,WORD32 shift_value,WORD scale_spec)76*15dc779aSAndroid Build Coastguard Worker VOID ixheaacd_res_tns_ar_filter_fixed_32x16(WORD32 *spectrum, WORD32 size, WORD32 inc,
77*15dc779aSAndroid Build Coastguard Worker WORD16 *lpc, WORD32 order, WORD32 shift_value,
78*15dc779aSAndroid Build Coastguard Worker WORD scale_spec) {
79*15dc779aSAndroid Build Coastguard Worker WORD32 i, j;
80*15dc779aSAndroid Build Coastguard Worker WORD32 y, state[MAX_ORDER + 1];
81*15dc779aSAndroid Build Coastguard Worker
82*15dc779aSAndroid Build Coastguard Worker if ((order & 3) != 0) {
83*15dc779aSAndroid Build Coastguard Worker for (i = order + 1; i < ((WORD32)(order & 0xfffffffc) + 4); i++) {
84*15dc779aSAndroid Build Coastguard Worker lpc[i] = 0;
85*15dc779aSAndroid Build Coastguard Worker }
86*15dc779aSAndroid Build Coastguard Worker lpc[i] = 0;
87*15dc779aSAndroid Build Coastguard Worker order = ((order & 0xfffffffc) + 4);
88*15dc779aSAndroid Build Coastguard Worker order = order & 31;
89*15dc779aSAndroid Build Coastguard Worker }
90*15dc779aSAndroid Build Coastguard Worker
91*15dc779aSAndroid Build Coastguard Worker for (i = 0; i < order; i++) {
92*15dc779aSAndroid Build Coastguard Worker y = (*spectrum) << scale_spec;
93*15dc779aSAndroid Build Coastguard Worker for (j = i; j > 0; j--) {
94*15dc779aSAndroid Build Coastguard Worker y = ixheaac_sub32_sat(y, ixheaac_mult32x16in32_shl_sat(state[j - 1], lpc[j]));
95*15dc779aSAndroid Build Coastguard Worker state[j] = state[j - 1];
96*15dc779aSAndroid Build Coastguard Worker }
97*15dc779aSAndroid Build Coastguard Worker
98*15dc779aSAndroid Build Coastguard Worker state[0] = ixheaac_shl32_dir_sat_limit(y, shift_value);
99*15dc779aSAndroid Build Coastguard Worker *spectrum = y >> scale_spec;
100*15dc779aSAndroid Build Coastguard Worker spectrum += inc;
101*15dc779aSAndroid Build Coastguard Worker }
102*15dc779aSAndroid Build Coastguard Worker
103*15dc779aSAndroid Build Coastguard Worker for (i = order; i < size; i++) {
104*15dc779aSAndroid Build Coastguard Worker y = (*spectrum) << scale_spec;
105*15dc779aSAndroid Build Coastguard Worker
106*15dc779aSAndroid Build Coastguard Worker for (j = order; j > 0; j--) {
107*15dc779aSAndroid Build Coastguard Worker y = ixheaac_sub32_sat(y, ixheaac_mult32x16in32_shl_sat(state[j - 1], lpc[j]));
108*15dc779aSAndroid Build Coastguard Worker state[j] = state[j - 1];
109*15dc779aSAndroid Build Coastguard Worker }
110*15dc779aSAndroid Build Coastguard Worker
111*15dc779aSAndroid Build Coastguard Worker state[0] = ixheaac_shl32_dir_sat_limit(y, shift_value);
112*15dc779aSAndroid Build Coastguard Worker *spectrum = y >> scale_spec;
113*15dc779aSAndroid Build Coastguard Worker spectrum += inc;
114*15dc779aSAndroid Build Coastguard Worker }
115*15dc779aSAndroid Build Coastguard Worker }
116*15dc779aSAndroid Build Coastguard Worker
ixheaacd_res_calc_max_spectral_line(WORD32 * p_tmp,WORD32 size)117*15dc779aSAndroid Build Coastguard Worker WORD32 ixheaacd_res_calc_max_spectral_line(WORD32 *p_tmp, WORD32 size) {
118*15dc779aSAndroid Build Coastguard Worker WORD32 max_spectral_line = 0, i;
119*15dc779aSAndroid Build Coastguard Worker WORD count, remaining, temp_1, temp_2, temp3, temp4;
120*15dc779aSAndroid Build Coastguard Worker
121*15dc779aSAndroid Build Coastguard Worker count = size >> 3;
122*15dc779aSAndroid Build Coastguard Worker for (i = count; i--;) {
123*15dc779aSAndroid Build Coastguard Worker temp_1 = *p_tmp++;
124*15dc779aSAndroid Build Coastguard Worker temp_2 = *p_tmp++;
125*15dc779aSAndroid Build Coastguard Worker temp3 = *p_tmp++;
126*15dc779aSAndroid Build Coastguard Worker temp4 = *p_tmp++;
127*15dc779aSAndroid Build Coastguard Worker
128*15dc779aSAndroid Build Coastguard Worker max_spectral_line = ixheaac_abs32_nrm(temp_1) | max_spectral_line;
129*15dc779aSAndroid Build Coastguard Worker max_spectral_line = ixheaac_abs32_nrm(temp_2) | max_spectral_line;
130*15dc779aSAndroid Build Coastguard Worker max_spectral_line = ixheaac_abs32_nrm(temp3) | max_spectral_line;
131*15dc779aSAndroid Build Coastguard Worker max_spectral_line = ixheaac_abs32_nrm(temp4) | max_spectral_line;
132*15dc779aSAndroid Build Coastguard Worker temp_1 = *p_tmp++;
133*15dc779aSAndroid Build Coastguard Worker temp_2 = *p_tmp++;
134*15dc779aSAndroid Build Coastguard Worker temp3 = *p_tmp++;
135*15dc779aSAndroid Build Coastguard Worker temp4 = *p_tmp++;
136*15dc779aSAndroid Build Coastguard Worker
137*15dc779aSAndroid Build Coastguard Worker max_spectral_line = ixheaac_abs32_nrm(temp_1) | max_spectral_line;
138*15dc779aSAndroid Build Coastguard Worker max_spectral_line = ixheaac_abs32_nrm(temp_2) | max_spectral_line;
139*15dc779aSAndroid Build Coastguard Worker max_spectral_line = ixheaac_abs32_nrm(temp3) | max_spectral_line;
140*15dc779aSAndroid Build Coastguard Worker max_spectral_line = ixheaac_abs32_nrm(temp4) | max_spectral_line;
141*15dc779aSAndroid Build Coastguard Worker }
142*15dc779aSAndroid Build Coastguard Worker
143*15dc779aSAndroid Build Coastguard Worker remaining = size - (count << 3);
144*15dc779aSAndroid Build Coastguard Worker if (remaining) {
145*15dc779aSAndroid Build Coastguard Worker for (i = remaining; i--;) {
146*15dc779aSAndroid Build Coastguard Worker max_spectral_line = ixheaac_abs32_nrm(*p_tmp) | max_spectral_line;
147*15dc779aSAndroid Build Coastguard Worker p_tmp++;
148*15dc779aSAndroid Build Coastguard Worker }
149*15dc779aSAndroid Build Coastguard Worker }
150*15dc779aSAndroid Build Coastguard Worker
151*15dc779aSAndroid Build Coastguard Worker return ixheaac_norm32(max_spectral_line);
152*15dc779aSAndroid Build Coastguard Worker }
153