Coverage Report

Created: 2026-03-21 07:02

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libxaac/encoder/ixheaace_sbr_tran_det_hp.c
Line
Count
Source
1
/******************************************************************************
2
 *                                                                            *
3
 * Copyright (C) 2023 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
21
#include <math.h>
22
#include <string.h>
23
#include "ixheaac_type_def.h"
24
#include "ixheaac_constants.h"
25
#include "ixheaace_aac_constants.h"
26
#include "ixheaac_basic_ops32.h"
27
#include "ixheaac_basic_ops16.h"
28
#include "ixheaac_basic_ops40.h"
29
#include "ixheaac_basic_ops.h"
30
31
#include "ixheaace_common_rom.h"
32
#include "ixheaace_sbr_header.h"
33
#include "ixheaace_sbr_def.h"
34
#include "ixheaace_resampler.h"
35
#include "ixheaace_sbr_rom.h"
36
#include "ixheaace_sbr_tran_det.h"
37
#include "ixheaace_sbr_main.h"
38
#include "ixheaace_sbr_frame_info_gen.h"
39
40
static VOID ixheaace_calc_thresholds(FLOAT32 **ptr_energies, WORD32 num_cols, WORD32 num_rows,
41
368k
                                     FLOAT32 *ptr_thresholds, ixheaace_sbr_codec_type sbr_codec) {
42
368k
  FLOAT32 mean_val, std_val, thr;
43
368k
  FLOAT32 *ptr_energy;
44
368k
  FLOAT32 inv_num_cols = 1.0f / (FLOAT32)(num_cols + num_cols / 2);
45
368k
  FLOAT32 inv_num_cols_1 = 1.0f / (FLOAT32)(num_cols + num_cols / 2 - 1);
46
47
368k
  WORD32 i = 0;
48
368k
  WORD32 j;
49
50
23.9M
  while (i < num_rows) {
51
23.5M
    mean_val = std_val = 0;
52
53
23.5M
    j = num_cols >> 2;
54
306M
    while (j < num_cols) {
55
282M
      ptr_energy = &ptr_energies[j][i];
56
282M
      mean_val += (*ptr_energy);
57
282M
      ptr_energy += 64;
58
282M
      mean_val += (*ptr_energy);
59
282M
      j += 2;
60
282M
    }
61
62
23.5M
    mean_val *= inv_num_cols * 2.0f;
63
64
23.5M
    j = num_cols >> 2;
65
589M
    while (j < num_cols) {
66
565M
      FLOAT32 tmp_var;
67
565M
      tmp_var = (sbr_codec == HEAAC_SBR) ? mean_val - ptr_energies[j][i] : ptr_energies[j][i];
68
565M
      std_val += tmp_var * tmp_var;
69
565M
      j++;
70
565M
    }
71
72
23.5M
    std_val = (FLOAT32)((sbr_codec == HEAAC_SBR)
73
23.5M
                            ? sqrt(std_val * inv_num_cols_1)
74
23.5M
                            : sqrt(fabs((mean_val * mean_val) - std_val * inv_num_cols * 2.0f)));
75
76
23.5M
    thr = 0.66f * ptr_thresholds[i] + 0.34f * IXHEAACE_SBR_TRAN_STD_FAC * std_val;
77
23.5M
    ptr_thresholds[i] = MAX(thr, IXHEAACE_SBR_TRAN_ABS_THR);
78
79
23.5M
    i++;
80
23.5M
  }
81
368k
}
82
83
static VOID ixheaace_extract_transient_candidates(FLOAT32 **ptr_energies, FLOAT32 *ptr_thresholds,
84
                                                  FLOAT32 *ptr_transients, WORD32 num_cols,
85
                                                  WORD32 start_band, WORD32 stop_band,
86
                                                  WORD32 buf_len)
87
88
368k
{
89
368k
  WORD32 idx;
90
368k
  WORD32 buf_move = buf_len >> 1;
91
368k
  FLOAT32 dt_1, dt_2, dt_3, inv_thr;
92
368k
  WORD32 len = num_cols + (num_cols >> 1) - 3;
93
368k
  WORD32 band = start_band;
94
95
368k
  memmove(ptr_transients, ptr_transients + num_cols, buf_move * sizeof(ptr_transients[0]));
96
368k
  memset(ptr_transients + buf_move, 0, (buf_len - buf_move) * sizeof(ptr_transients[0]));
97
98
23.9M
  while (band < stop_band) {
99
23.5M
    inv_thr = (FLOAT32)1.0f / ptr_thresholds[band];
100
23.5M
    FLOAT32 temp_energy_1 = ptr_energies[((num_cols >> 1) - 2) / 2][band];
101
23.5M
    FLOAT32 temp_energy_2 = ptr_energies[((num_cols >> 1)) / 2][band];
102
23.5M
    FLOAT32 temp_energy_3 = ptr_energies[((num_cols >> 1) + 2) / 2][band];
103
1.08G
    for (idx = 0; idx < len; idx++) {
104
1.06G
      if (!idx) {
105
23.5M
        dt_1 = temp_energy_2 - temp_energy_1;
106
23.5M
        dt_2 = temp_energy_3 - temp_energy_1;
107
23.5M
        dt_3 = temp_energy_3 - ptr_energies[((num_cols >> 1) - 4) / 2][band];
108
1.03G
      } else {
109
1.03G
        FLOAT32 temp_energy_4 = ptr_energies[(idx + (num_cols >> 1) + 3) / 2][band];
110
1.03G
        dt_1 = temp_energy_3 - temp_energy_2;
111
1.03G
        dt_2 = temp_energy_3 - temp_energy_1 + dt_1;
112
1.03G
        dt_3 = temp_energy_4 - temp_energy_1 + dt_2;
113
1.03G
        temp_energy_1 = temp_energy_2;
114
1.03G
        temp_energy_2 = temp_energy_3;
115
1.03G
        temp_energy_3 = temp_energy_4;
116
1.03G
      }
117
1.06G
      if (dt_1 > ptr_thresholds[band]) {
118
20.8M
        ptr_transients[idx + buf_move] += dt_1 * inv_thr - 1.0f;
119
20.8M
      }
120
1.06G
      if (dt_2 > ptr_thresholds[band]) {
121
48.1M
        ptr_transients[idx + buf_move] += dt_2 * inv_thr - 1.0f;
122
48.1M
      }
123
1.06G
      if (dt_3 > ptr_thresholds[band]) {
124
67.1M
        ptr_transients[idx + buf_move] += dt_3 * inv_thr - 1.0f;
125
67.1M
      }
126
1.06G
    }
127
541M
    for (idx = 1; idx < len; idx += 2) {
128
518M
      ptr_transients[idx + buf_move + 1] = ptr_transients[idx + buf_move];
129
518M
    }
130
23.5M
    band++;
131
23.5M
  }
132
368k
}
133
134
VOID ixheaace_detect_transient(FLOAT32 **ptr_energies,
135
                               ixheaace_pstr_sbr_trans_detector pstr_sbr_trans_det,
136
                               WORD32 *ptr_tran_vector, WORD32 time_step,
137
368k
                               ixheaace_sbr_codec_type sbr_codec) {
138
368k
  WORD32 i;
139
368k
  WORD32 no_cols = pstr_sbr_trans_det->no_cols;
140
368k
  WORD32 qmf_start_sample = no_cols + time_step * 4;
141
368k
  FLOAT32 int_thr = (FLOAT32)pstr_sbr_trans_det->tran_thr / (FLOAT32)pstr_sbr_trans_det->no_rows;
142
368k
  FLOAT32 *ptr_trans = &(pstr_sbr_trans_det->ptr_transients[qmf_start_sample]);
143
144
368k
  ptr_tran_vector[0] = 0;
145
368k
  ptr_tran_vector[1] = 0;
146
147
368k
  ixheaace_calc_thresholds(ptr_energies, pstr_sbr_trans_det->no_cols, pstr_sbr_trans_det->no_rows,
148
368k
                           pstr_sbr_trans_det->ptr_thresholds, sbr_codec);
149
150
368k
  ixheaace_extract_transient_candidates(
151
368k
      ptr_energies, pstr_sbr_trans_det->ptr_thresholds, pstr_sbr_trans_det->ptr_transients,
152
368k
      pstr_sbr_trans_det->no_cols, 0, pstr_sbr_trans_det->no_rows,
153
368k
      pstr_sbr_trans_det->buffer_length);
154
155
10.7M
  for (i = 0; i < no_cols; i++) {
156
10.4M
    if ((ptr_trans[i] < 0.9f * ptr_trans[i - 1]) && (ptr_trans[i - 1] > int_thr)) {
157
89.7k
      ptr_tran_vector[0] = (WORD32)floor(i / time_step);
158
89.7k
      ptr_tran_vector[1] = 1;
159
89.7k
      break;
160
89.7k
    }
161
10.4M
  }
162
368k
}
163
164
static VOID ixheaace_calc_thresholds_4_1(FLOAT32 **ptr_energies, WORD32 num_cols, WORD32 num_rows,
165
61.0k
                                         FLOAT32 *ptr_thresholds, WORD32 time_step) {
166
61.0k
  FLOAT32 mean_val, std_val, thr;
167
61.0k
  FLOAT32 *ptr_energy;
168
61.0k
  FLOAT32 inv_num_cols = 1.0f / (FLOAT32)((num_cols + num_cols / 2) / time_step);
169
61.0k
  FLOAT32 inv_num_cols_1 = 1.0f / (FLOAT32)((num_cols + num_cols / 2 - 1) / time_step);
170
171
61.0k
  WORD32 i = 0;
172
61.0k
  WORD32 j;
173
61.0k
  WORD32 start_band = 8;
174
61.0k
  WORD32 end_band = 32;
175
176
3.96M
  while (i < num_rows) {
177
3.90M
    mean_val = std_val = 0;
178
179
3.90M
    j = start_band;
180
97.6M
    while (j < end_band) {
181
93.7M
      ptr_energy = &ptr_energies[j][i];
182
93.7M
      mean_val += (*ptr_energy);
183
93.7M
      j++;
184
93.7M
    }
185
186
3.90M
    mean_val *= inv_num_cols;
187
188
3.90M
    j = start_band;
189
97.6M
    while (j < end_band) {
190
93.7M
      FLOAT32 tmp_var;
191
93.7M
      tmp_var = mean_val - ptr_energies[j][i];
192
93.7M
      std_val += tmp_var * tmp_var;
193
93.7M
      j++;
194
93.7M
    }
195
196
3.90M
    std_val = (FLOAT32)sqrt(std_val * inv_num_cols_1);
197
198
3.90M
    thr = 0.66f * ptr_thresholds[i] + 0.34f * IXHEAACE_SBR_TRAN_STD_FAC * std_val;
199
3.90M
    ptr_thresholds[i] = MAX(thr, IXHEAACE_SBR_TRAN_ABS_THR);
200
201
3.90M
    i++;
202
3.90M
  }
203
61.0k
}
204
205
static VOID ixheaace_extract_transient_candidates_4_1(FLOAT32 **ptr_energies,
206
                                                      FLOAT32 *ptr_thresholds,
207
                                                      FLOAT32 *ptr_transients, WORD32 num_cols,
208
                                                      WORD32 start_band, WORD32 stop_band,
209
                                                      WORD32 time_step)
210
211
61.0k
{
212
61.0k
  WORD32 idx;
213
61.0k
  WORD32 buf_move = num_cols / 2;
214
61.0k
  WORD32 band = start_band;
215
216
61.0k
  memmove(ptr_transients, ptr_transients + num_cols, buf_move * sizeof(ptr_transients[0]));
217
61.0k
  memset(ptr_transients + buf_move, 0, num_cols * sizeof(ptr_transients[0]));
218
219
3.96M
  while (band < stop_band) {
220
253M
    for (idx = buf_move; idx < num_cols + buf_move; idx++) {
221
250M
      float l = 0, r = 0;
222
1.00G
      for (int d = 1; d < 4; d++) {
223
750M
        l = ptr_energies[(idx - d) / time_step][band];
224
750M
        r = ptr_energies[(idx + d) / time_step][band];
225
750M
        if (r - l > ptr_thresholds[band])
226
4.53M
          ptr_transients[idx] += (r - l - ptr_thresholds[band]) / ptr_thresholds[band];
227
750M
      }
228
250M
    }
229
3.90M
    band++;
230
3.90M
  }
231
61.0k
}
232
233
VOID ixheaace_detect_transient_4_1(FLOAT32 **ptr_energies,
234
                                   ixheaace_pstr_sbr_trans_detector pstr_sbr_trans_det,
235
61.0k
                                   WORD32 *ptr_tran_vector, WORD32 time_step) {
236
61.0k
  WORD32 i;
237
61.0k
  WORD32 no_cols = pstr_sbr_trans_det->no_cols;
238
61.0k
  WORD32 qmf_start_sample = time_step * 4;
239
61.0k
  FLOAT32 int_thr = (FLOAT32)pstr_sbr_trans_det->tran_thr / (FLOAT32)pstr_sbr_trans_det->no_rows;
240
61.0k
  FLOAT32 *ptr_trans = &(pstr_sbr_trans_det->ptr_transients[qmf_start_sample]);
241
242
61.0k
  ptr_tran_vector[0] = 0;
243
61.0k
  ptr_tran_vector[1] = 0;
244
245
61.0k
  ixheaace_calc_thresholds_4_1(ptr_energies, pstr_sbr_trans_det->no_cols,
246
61.0k
                               pstr_sbr_trans_det->no_rows, pstr_sbr_trans_det->ptr_thresholds,
247
61.0k
                               time_step);
248
249
61.0k
  ixheaace_extract_transient_candidates_4_1(
250
61.0k
      ptr_energies, pstr_sbr_trans_det->ptr_thresholds, pstr_sbr_trans_det->ptr_transients,
251
61.0k
      pstr_sbr_trans_det->no_cols, 0, pstr_sbr_trans_det->no_rows, time_step);
252
253
3.82M
  for (i = 0; i < no_cols; i++) {
254
3.76M
    if ((ptr_trans[i] < 0.9f * ptr_trans[i - 1]) && (ptr_trans[i - 1] > int_thr)) {
255
4.98k
      ptr_tran_vector[0] = (WORD32)floor(i / time_step);
256
4.98k
      ptr_tran_vector[1] = 1;
257
4.98k
      break;
258
4.98k
    }
259
3.76M
  }
260
61.0k
}
261
262
VOID ixheaace_detect_transient_eld(FLOAT32 **ptr_energies,
263
                                   ixheaace_pstr_sbr_trans_detector pstr_sbr_trans_det,
264
119k
                                   WORD32 *ptr_tran_vector) {
265
119k
  WORD32 i, band;
266
119k
  WORD32 max_idx = 0, is_transient = 0;
267
119k
  FLOAT32 delta_max = 0, tmp, min_energy;
268
119k
  WORD32 num_slots = pstr_sbr_trans_det->time_slots;
269
119k
  WORD32 look_ahead = pstr_sbr_trans_det->look_ahead;
270
119k
  WORD32 start_band = pstr_sbr_trans_det->start_band;
271
119k
  WORD32 stop_band = pstr_sbr_trans_det->stop_band;
272
119k
  FLOAT32 *ptr_energy = pstr_sbr_trans_det->energy;
273
119k
  FLOAT32 *ptr_delta_energy = pstr_sbr_trans_det->delta_energy;
274
119k
  FLOAT32 *ptr_transients = pstr_sbr_trans_det->ptr_transients;
275
119k
  WORD32 ts = look_ahead;
276
119k
  FLOAT32 weighted_energy;
277
278
119k
  ptr_tran_vector[0] = ptr_tran_vector[1] = 0;
279
119k
  ptr_tran_vector[2] = 0;
280
281
119k
  memset(ptr_transients + look_ahead, 0, num_slots * sizeof(ptr_transients[0]));
282
283
1.96M
  while (ts < num_slots + look_ahead) {
284
1.84M
    tmp = 0.0f;
285
1.84M
    max_idx = 0;
286
1.84M
    delta_max = 0;
287
1.84M
    is_transient = 0;
288
1.84M
    i = 0;
289
1.84M
    band = start_band;
290
291
41.7M
    while (band < stop_band) {
292
39.9M
      tmp += (ptr_energies[ts][band] * pstr_sbr_trans_det->coeff[i]);
293
39.9M
      band++;
294
39.9M
      i++;
295
39.9M
    }
296
297
1.84M
    ptr_energy[ts] = tmp;
298
1.84M
    min_energy = (ptr_energy[ts - 1]) + IXHEAACE_SMALL_ENERGY;
299
1.84M
    ptr_delta_energy[ts] = ptr_energy[ts] / min_energy;
300
301
1.84M
    weighted_energy = ptr_energy[ts] * (1.0f / 1.4f);
302
303
1.84M
    if ((ptr_delta_energy[ts] >= IXHEAACE_TRANSIENT_THRESHOLD) &&
304
53.5k
        (((ptr_transients[ts - 2] == 0) && (ptr_transients[ts - 1] == 0)) ||
305
53.5k
         (weighted_energy >= ptr_energy[ts - 1]) || (weighted_energy >= ptr_energy[ts - 2]))) {
306
53.5k
      ptr_transients[ts] = 1;
307
53.5k
    }
308
309
1.84M
    ts++;
310
1.84M
  }
311
312
1.96M
  for (ts = 0; ts < num_slots; ts++) {
313
1.84M
    if (ptr_transients[ts] && (ptr_delta_energy[ts] > delta_max)) {
314
40.9k
      delta_max = ptr_delta_energy[ts];
315
40.9k
      max_idx = ts;
316
40.9k
      is_transient = 1;
317
40.9k
    }
318
1.84M
  }
319
320
119k
  if (is_transient) {
321
37.7k
    ptr_tran_vector[0] = max_idx;
322
37.7k
    ptr_tran_vector[1] = 1;
323
37.7k
  }
324
325
357k
  for (ts = 0; ts < look_ahead; ts++) {
326
238k
    if (ptr_transients[ts + num_slots]) {
327
2.91k
      ptr_tran_vector[2] = 1;
328
2.91k
    }
329
238k
    ptr_energy[ts] = ptr_energy[num_slots + ts];
330
238k
    ptr_transients[ts] = ptr_transients[num_slots + ts];
331
238k
    ptr_delta_energy[ts] = ptr_delta_energy[num_slots + ts];
332
238k
  }
333
119k
}