Coverage Report

Created: 2025-08-03 06:57

/src/libxaac/decoder/ixheaacd_aac_tns.c
Line
Count
Source (jump to first uncovered line)
1
/******************************************************************************
2
 *                                                                            *
3
 * Copyright (C) 2018 The Android Open Source Project
4
 *
5
 * Licensed under the Apache License, Version 2.0 (the "License");
6
 * you may not use this file except in compliance with the License.
7
 * You may obtain a copy of the License at:
8
 *
9
 * http://www.apache.org/licenses/LICENSE-2.0
10
 *
11
 * Unless required by applicable law or agreed to in writing, software
12
 * distributed under the License is distributed on an "AS IS" BASIS,
13
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14
 * See the License for the specific language governing permissions and
15
 * limitations under the License.
16
 *
17
 *****************************************************************************
18
 * Originally developed and contributed by Ittiam Systems Pvt. Ltd, Bangalore
19
*/
20
#include "ixheaacd_sbr_common.h"
21
#include "ixheaac_type_def.h"
22
23
#include "ixheaac_constants.h"
24
#include "ixheaac_basic_ops32.h"
25
#include "ixheaac_basic_ops16.h"
26
#include "ixheaac_basic_ops40.h"
27
#include "ixheaac_basic_ops.h"
28
29
#include "ixheaacd_defines.h"
30
#include "ixheaacd_bitbuffer.h"
31
32
#include "ixheaacd_error_codes.h"
33
#include "ixheaacd_aac_rom.h"
34
#include "ixheaacd_pulsedata.h"
35
36
#include "ixheaacd_pns.h"
37
#include "ixheaacd_drc_data_struct.h"
38
39
#include "ixheaacd_lt_predict.h"
40
41
#include "ixheaacd_cnst.h"
42
#include "ixheaacd_ec_defines.h"
43
#include "ixheaacd_ec_struct_def.h"
44
#include "ixheaacd_channelinfo.h"
45
#include "ixheaacd_drc_dec.h"
46
#include "ixheaacd_sbrdecoder.h"
47
#include "ixheaacd_tns.h"
48
#include "ixheaacd_intrinsics.h"
49
50
#include "ixheaacd_common_rom.h"
51
#include "ixheaacd_block.h"
52
#include "ixheaacd_channel.h"
53
#include "ixheaacd_audioobjtypes.h"
54
#include "ixheaacd_latmdemux.h"
55
56
#include "ixheaacd_aacdec.h"
57
58
static PLATFORM_INLINE WORD32 ixheaacd_mac32_tns_sat(WORD32 a, WORD32 b,
59
34.1M
                                                     WORD32 c) {
60
34.1M
  WORD32 result;
61
34.1M
  WORD64 temp_result;
62
63
34.1M
  temp_result = (WORD64)a * (WORD64)b;
64
34.1M
  result = (WORD32)(temp_result >> 32);
65
34.1M
  result = ixheaac_add32_sat(c, result);
66
34.1M
  return (result);
67
34.1M
}
68
69
VOID ixheaacd_tns_decode_coefficients(
70
    const ia_filter_info_struct *filter, WORD32 *a,
71
46.0k
    ia_aac_dec_tables_struct *ptr_aac_tables) {
72
46.0k
  WORD32 i;
73
46.0k
  WORD32 tmp;
74
46.0k
  WORD32 *aptr = a;
75
46.0k
  WORD32 *tns_coeff_ptr;
76
46.0k
  WORD8 ixheaacd_drc_offset;
77
78
46.0k
  tmp = filter->resolution;
79
46.0k
  if (tmp == 0) {
80
42.3k
    tns_coeff_ptr = ptr_aac_tables->pstr_block_tables->tns_coeff3;
81
42.3k
    ixheaacd_drc_offset = 4;
82
83
42.3k
  } else {
84
3.66k
    tns_coeff_ptr = ptr_aac_tables->pstr_block_tables->tns_coeff4;
85
3.66k
    ixheaacd_drc_offset = 8;
86
3.66k
  }
87
88
440k
  for (i = 0; i < filter->order; i++) {
89
394k
    *aptr++ = tns_coeff_ptr[filter->coef[i] + ixheaacd_drc_offset];
90
394k
  }
91
46.0k
}
92
93
VOID ixheaacd_tns_parcor_to_lpc(WORD32 *parcor, WORD32 *lpc, WORD16 *scale,
94
                                WORD32 order)
95
96
34.7k
{
97
34.7k
  WORD i, j, status;
98
34.7k
  WORD32 z1;
99
34.7k
  WORD32 z[MAX_ORDER + 1];
100
34.7k
  WORD32 w[MAX_ORDER + 1];
101
34.7k
  WORD32 accu1, accu2;
102
103
34.7k
  status = 1;
104
34.7k
  *scale = 1;
105
106
97.3k
  while (status) {
107
62.5k
    status = 0;
108
109
2.06M
    for (i = MAX_ORDER; i >= 0; i--) {
110
2.00M
      z[i] = 0;
111
2.00M
      w[i] = 0;
112
2.00M
    }
113
114
62.5k
    accu1 = (0x40000000 >> (*scale - 1));
115
116
779k
    for (i = 0; i <= order; i++) {
117
716k
      z1 = accu1;
118
119
8.64M
      for (j = 0; j < order; j++) {
120
7.92M
        w[j] = (accu1);
121
122
7.92M
        accu1 = ixheaac_add32_sat(accu1,
123
7.92M
                                   ixheaac_mult32_shl_sat(parcor[j], (z[j])));
124
7.92M
        if (ixheaac_abs32_sat(accu1) == 0x7fffffff) status = 1;
125
7.92M
      }
126
8.64M
      for (j = (order - 1); j >= 0; j--) {
127
7.92M
        accu2 = (z[j]);
128
7.92M
        accu2 = ixheaac_add32_sat(accu2,
129
7.92M
                                   ixheaac_mult32_shl_sat(parcor[j], (w[j])));
130
7.92M
        z[j + 1] = (accu2);
131
7.92M
        if (ixheaac_abs32_sat(accu2) == 0x7fffffff) status = 1;
132
7.92M
      }
133
134
716k
      z[0] = (z1);
135
716k
      lpc[i] = (accu1);
136
716k
      accu1 = 0;
137
716k
    }
138
139
62.5k
    accu1 = (status - 1);
140
141
62.5k
    if (accu1 == 0) {
142
27.7k
      *scale = *scale + 1;
143
27.7k
    }
144
62.5k
  }
145
34.7k
}
146
147
VOID ixheaacd_tns_parcor_lpc_convert_dec(WORD16 *parcor, WORD16 *lpc,
148
                                         WORD16 *scale, WORD order)
149
150
41.7k
{
151
41.7k
  WORD i, j, status;
152
41.7k
  WORD32 accu;
153
41.7k
  WORD16 temp_buf1[MAX_ORDER + 1];
154
41.7k
  WORD16 temp_buf2[MAX_ORDER + 1];
155
41.7k
  WORD32 accu1, accu2;
156
157
41.7k
  status = 1;
158
41.7k
  *scale = 0;
159
160
127k
  while (status) {
161
86.1k
    status = 0;
162
163
2.84M
    for (i = MAX_ORDER; i >= 0; i--) {
164
2.75M
      temp_buf1[i] = 0;
165
2.75M
      temp_buf2[i] = 0;
166
2.75M
    }
167
168
86.1k
    accu1 = (0x7fffffff >> *scale);
169
170
713k
    for (i = 0; i <= order; i++) {
171
627k
      accu = accu1;
172
173
4.91M
      for (j = 0; j < order; j++) {
174
4.28M
        temp_buf2[j] = ixheaac_round16(accu1);
175
4.28M
        accu1 = ixheaac_mac16x16in32_shl_sat(accu1, parcor[j], temp_buf1[j]);
176
177
4.28M
        if (ixheaac_abs32_sat(accu1) == 0x7fffffff) {
178
329k
          status = 1;
179
329k
        }
180
4.28M
      }
181
182
4.91M
      for (j = (order - 1); j >= 0; j--) {
183
4.28M
        accu2 = ixheaac_deposit16h_in32(temp_buf1[j]);
184
4.28M
        accu2 = ixheaac_mac16x16in32_shl_sat(accu2, parcor[j], temp_buf2[j]);
185
4.28M
        temp_buf1[j + 1] = ixheaac_round16(accu2);
186
4.28M
        if (ixheaac_abs32_sat(accu2) == 0x7fffffff) {
187
65.7k
          status = 1;
188
65.7k
        }
189
4.28M
      }
190
191
627k
      temp_buf1[0] = ixheaac_round16(accu);
192
627k
      lpc[i] = ixheaac_round16(accu1);
193
627k
      accu1 = 0;
194
627k
    }
195
196
86.1k
    accu1 = (status - 1);
197
198
86.1k
    if (accu1 == 0) {
199
44.3k
      *scale = *scale + 1;
200
44.3k
    }
201
86.1k
  }
202
41.7k
}
203
204
VOID ixheaacd_tns_ar_filter_fixed_dec(WORD32 *spectrum, WORD32 size, WORD32 inc,
205
                                      WORD32 *lpc, WORD32 order,
206
                                      WORD32 shift_value, WORD scale_spec)
207
208
17.8k
{
209
17.8k
  WORD32 i, j;
210
17.8k
  WORD32 y, state[MAX_ORDER + 1];
211
17.8k
  WORD32 acc;
212
213
17.8k
  if ((order & 3) != 0) {
214
16.5k
    for (i = order + 1; i < ((WORD32)(order & 0xfffffffc) + 4); i++) {
215
8.01k
      lpc[i] = 0;
216
8.01k
    }
217
8.50k
    lpc[i] = 0;
218
8.50k
    order = ((order & 0xfffffffc) + 4);
219
8.50k
    order = order & 31;
220
8.50k
  }
221
17.8k
  {
222
202k
    for (i = 0; i < order; i++) {
223
184k
      y = ixheaac_shl32_sat((*spectrum), scale_spec);
224
184k
      acc = 0;
225
226
1.11M
      for (j = i; j > 0; j--) {
227
927k
        acc = ixheaacd_mac32_tns_sat(state[j - 1], lpc[j], acc);
228
927k
        state[j] = state[j - 1];
229
927k
      }
230
184k
      y = ixheaac_sub32_sat(y, ixheaac_shl32_sat(acc, 1));
231
184k
      state[0] = ixheaac_shl32_sat(y, shift_value);
232
233
184k
      *spectrum = y >> scale_spec;
234
184k
      spectrum += inc;
235
184k
    }
236
3.15M
    for (i = order; i < size; i++) {
237
3.14M
      y = ixheaac_shl32_sat((*spectrum), scale_spec);
238
3.14M
      acc = 0;
239
36.3M
      for (j = order; j > 0; j--) {
240
33.2M
        acc = ixheaacd_mac32_tns_sat(state[j - 1], lpc[j], acc);
241
33.2M
        state[j] = state[j - 1];
242
33.2M
      }
243
3.14M
      y = ixheaac_sub32_sat(y, ixheaac_shl32_sat(acc, 1));
244
3.14M
      state[0] = ixheaac_shl32_sat(y, shift_value);
245
246
3.14M
      *spectrum = y >> scale_spec;
247
3.14M
      spectrum += inc;
248
3.14M
    }
249
17.8k
  }
250
17.8k
}
251
252
VOID ixheaacd_tns_ar_filter_fixed_non_neon_armv7(WORD32 *spectrum, WORD32 size,
253
                                                 WORD32 inc, WORD32 *lpc,
254
                                                 WORD32 order,
255
                                                 WORD32 shift_value,
256
0
                                                 WORD scale_spec) {
257
0
  WORD32 i, j;
258
0
  WORD32 y, state[MAX_ORDER + 1];
259
0
  WORD32 acc;
260
261
0
  if ((order & 3) != 0) {
262
0
    for (i = order + 1; i < ((WORD32)(order & 0xfffffffc) + 4); i++) {
263
0
      lpc[i] = 0;
264
0
    }
265
0
    lpc[i] = 0;
266
0
    order = ((order & 0xfffffffc) + 4);
267
0
  }
268
0
  {
269
0
    for (i = 0; i < order; i++) {
270
0
      y = ixheaac_shl32_sat((*spectrum), scale_spec);
271
0
      acc = 0;
272
273
0
      for (j = i; j > 0; j--) {
274
0
        acc = ixheaacd_mac32_tns_sat(state[j - 1], lpc[j], acc);
275
0
        state[j] = state[j - 1];
276
0
      }
277
0
      y = ixheaac_sub32_sat(y, ixheaac_shl32_sat(acc, 1));
278
0
      state[0] = ixheaac_shl32_sat(y, shift_value);
279
280
0
      *spectrum = y >> scale_spec;
281
0
      spectrum += inc;
282
0
    }
283
0
    for (i = order; i < size; i++) {
284
0
      WORD64 acc = 0;
285
0
      WORD32 acc1;
286
0
      y = ixheaac_shl32_sat((*spectrum), scale_spec);
287
0
      for (j = order; j > 0; j--) {
288
0
        acc = ixheaac_mac32x32in64_dual(state[j - 1], lpc[j], acc);
289
0
        state[j] = state[j - 1];
290
0
      }
291
0
      acc1 = (WORD32)(acc >> 32);
292
293
0
      y = ixheaac_sub32_sat(y, ixheaac_shl32_sat(acc1, 1));
294
0
      state[0] = ixheaac_shl32_sat(y, shift_value);
295
296
0
      *spectrum = y >> scale_spec;
297
0
      spectrum += inc;
298
0
    }
299
0
  }
300
0
}
301
302
VOID ixheaacd_tns_ar_filter_fixed_armv8(WORD32 *spectrum, WORD32 size,
303
                                        WORD32 inc, WORD32 *lpc, WORD32 order,
304
0
                                        WORD32 shift_value, WORD scale_spec) {
305
0
  WORD32 i, j;
306
0
  WORD32 y, state[MAX_ORDER + 1];
307
0
  WORD32 acc;
308
309
0
  if ((order & 3) != 0) {
310
0
    for (i = order + 1; i < ((WORD32)(order & 0xfffffffc) + 4); i++) {
311
0
      lpc[i] = 0;
312
0
    }
313
0
    lpc[i] = 0;
314
0
    order = ((order & 0xfffffffc) + 4);
315
0
  }
316
0
  {
317
0
    for (i = 0; i < order; i++) {
318
0
      y = ixheaac_shl32_sat((*spectrum), scale_spec);
319
0
      acc = 0;
320
321
0
      for (j = i; j > 0; j--) {
322
0
        acc = ixheaacd_mac32_tns_sat(state[j - 1], lpc[j], acc);
323
0
        state[j] = state[j - 1];
324
0
      }
325
0
      y = ixheaac_sub32_sat(y, ixheaac_shl32_sat(acc, 1));
326
0
      state[0] = ixheaac_shl32_sat(y, shift_value);
327
328
0
      *spectrum = y >> scale_spec;
329
0
      spectrum += inc;
330
0
    }
331
0
    for (i = order; i < size; i++) {
332
0
      WORD64 acc = 0;
333
0
      WORD32 acc1;
334
0
      y = ixheaac_shl32_sat((*spectrum), scale_spec);
335
0
      for (j = order; j > 0; j--) {
336
0
        acc = ixheaac_mac32x32in64_dual(state[j - 1], lpc[j], acc);
337
0
        state[j] = state[j - 1];
338
0
      }
339
0
      acc1 = (WORD32)(acc >> 32);
340
341
0
      y = ixheaac_sub32_sat(y, ixheaac_shl32_sat(acc1, 1));
342
0
      state[0] = ixheaac_shl32_sat(y, shift_value);
343
344
0
      *spectrum = y >> scale_spec;
345
0
      spectrum += inc;
346
0
    }
347
0
  }
348
0
}
349
350
void ixheaacd_tns_ma_filter_fixed_ld(WORD32 *spectrum, WORD32 size, WORD32 inc,
351
                                     WORD32 *lpc, WORD32 order,
352
16.9k
                                     WORD16 shift_value) {
353
16.9k
  WORD32 i, j;
354
16.9k
  WORD32 y, state[MAX_ORDER];
355
356
178k
  for (i = 0; i < order; i++) state[i] = 0;
357
358
3.21M
  for (i = 0; i < size; i++) {
359
3.20M
    y = *spectrum;
360
361
35.3M
    for (j = 0; j < order; j++) y += ixheaac_mult32_shl(state[j], lpc[j + 1]);
362
363
32.1M
    for (j = order - 1; j > 0; j--) state[j] = state[j - 1];
364
365
3.20M
    state[0] = ixheaac_shl32_dir_sat(*spectrum, shift_value);
366
3.20M
    *spectrum = y;
367
3.20M
    spectrum += inc;
368
3.20M
  }
369
16.9k
}
370
371
VOID ixheaacd_tns_ar_filter_dec(WORD32 *spectrum, WORD32 size, WORD32 inc,
372
                                WORD16 *lpc, WORD32 order, WORD32 shift_value,
373
38.1k
                                WORD scale_spec, WORD32 *ptr_filter_state) {
374
38.1k
  WORD32 i, j;
375
38.1k
  WORD32 y;
376
38.1k
  WORD32 acc;
377
378
38.1k
  if ((order & 3) != 0) {
379
42.1k
    for (i = order + 1; i < ((WORD32)(order & (~3)) + 4); i++) {
380
8.83k
      lpc[i] = 0;
381
8.83k
    }
382
33.2k
    if (i < (MAX_ORDER + 1)) {
383
33.2k
      lpc[i] = 0;
384
33.2k
      order = ((order & (~3)) + 4);
385
33.2k
    } else {
386
0
      order = MAX_ORDER;
387
0
    }
388
33.2k
  }
389
390
315k
  for (i = 0; i < order; i++) {
391
276k
    y = ixheaac_shl32_sat((*spectrum), scale_spec);
392
276k
    acc = 0;
393
394
1.20M
    for (j = i; j > 0; j--) {
395
926k
      acc = ixheaac_add32_sat(
396
926k
          acc, ixheaac_mult32x16in32(ptr_filter_state[j - 1], lpc[j]));
397
926k
      ptr_filter_state[j] = ptr_filter_state[j - 1];
398
926k
    }
399
400
276k
    y = ixheaac_sub32_sat(y, ixheaac_shl32_sat(acc, 1));
401
276k
    ptr_filter_state[0] = ixheaac_shl32_sat(y, shift_value);
402
276k
    *spectrum = y >> scale_spec;
403
276k
    spectrum += inc;
404
276k
  }
405
406
2.10M
  for (i = order; i < size; i++) {
407
2.07M
    y = ixheaac_shl32_sat((*spectrum), scale_spec);
408
2.07M
    acc = 0;
409
15.9M
    for (j = order; j > 0; j--) {
410
13.8M
      acc = ixheaac_add32_sat(
411
13.8M
          acc, ixheaac_mult32x16in32(ptr_filter_state[j - 1], lpc[j]));
412
13.8M
      ptr_filter_state[j] = ptr_filter_state[j - 1];
413
13.8M
    }
414
415
2.07M
    y = ixheaac_sub32_sat(y, ixheaac_shl32_sat(acc, 1));
416
2.07M
    ptr_filter_state[0] = ixheaac_shl32_sat(y, shift_value);
417
2.07M
    *spectrum = y >> scale_spec;
418
2.07M
    spectrum += inc;
419
2.07M
  }
420
38.1k
}
421
422
483k
WORD32 ixheaacd_calc_max_spectral_line_dec(WORD32 *ptr_tmp, WORD32 size) {
423
483k
  WORD32 max_spec_line = 0, i;
424
483k
  WORD unroll_cnt, rem;
425
426
483k
  unroll_cnt = size >> 3;
427
38.3M
  for (i = unroll_cnt; i--;) {
428
37.8M
    max_spec_line = ixheaac_abs32_nrm(*ptr_tmp++) | max_spec_line;
429
37.8M
    max_spec_line = ixheaac_abs32_nrm(*ptr_tmp++) | max_spec_line;
430
37.8M
    max_spec_line = ixheaac_abs32_nrm(*ptr_tmp++) | max_spec_line;
431
37.8M
    max_spec_line = ixheaac_abs32_nrm(*ptr_tmp++) | max_spec_line;
432
433
37.8M
    max_spec_line = ixheaac_abs32_nrm(*ptr_tmp++) | max_spec_line;
434
37.8M
    max_spec_line = ixheaac_abs32_nrm(*ptr_tmp++) | max_spec_line;
435
37.8M
    max_spec_line = ixheaac_abs32_nrm(*ptr_tmp++) | max_spec_line;
436
37.8M
    max_spec_line = ixheaac_abs32_nrm(*ptr_tmp++) | max_spec_line;
437
37.8M
  }
438
439
483k
  rem = size - (unroll_cnt << 3);
440
441
483k
  if (rem) {
442
280k
    for (i = rem; i--;) {
443
224k
      max_spec_line = ixheaac_abs32_nrm(*ptr_tmp++) | max_spec_line;
444
224k
    }
445
56.1k
  }
446
447
483k
  return ixheaac_norm32(max_spec_line);
448
483k
}