Coverage Report

Created: 2025-08-26 06:50

/src/aac/libSACdec/src/sac_process.cpp
Line
Count
Source (jump to first uncovered line)
1
/* -----------------------------------------------------------------------------
2
Software License for The Fraunhofer FDK AAC Codec Library for Android
3
4
© Copyright  1995 - 2021 Fraunhofer-Gesellschaft zur Förderung der angewandten
5
Forschung e.V. All rights reserved.
6
7
 1.    INTRODUCTION
8
The Fraunhofer FDK AAC Codec Library for Android ("FDK AAC Codec") is software
9
that implements the MPEG Advanced Audio Coding ("AAC") encoding and decoding
10
scheme for digital audio. This FDK AAC Codec software is intended to be used on
11
a wide variety of Android devices.
12
13
AAC's HE-AAC and HE-AAC v2 versions are regarded as today's most efficient
14
general perceptual audio codecs. AAC-ELD is considered the best-performing
15
full-bandwidth communications codec by independent studies and is widely
16
deployed. AAC has been standardized by ISO and IEC as part of the MPEG
17
specifications.
18
19
Patent licenses for necessary patent claims for the FDK AAC Codec (including
20
those of Fraunhofer) may be obtained through Via Licensing
21
(www.vialicensing.com) or through the respective patent owners individually for
22
the purpose of encoding or decoding bit streams in products that are compliant
23
with the ISO/IEC MPEG audio standards. Please note that most manufacturers of
24
Android devices already license these patent claims through Via Licensing or
25
directly from the patent owners, and therefore FDK AAC Codec software may
26
already be covered under those patent licenses when it is used for those
27
licensed purposes only.
28
29
Commercially-licensed AAC software libraries, including floating-point versions
30
with enhanced sound quality, are also available from Fraunhofer. Users are
31
encouraged to check the Fraunhofer website for additional applications
32
information and documentation.
33
34
2.    COPYRIGHT LICENSE
35
36
Redistribution and use in source and binary forms, with or without modification,
37
are permitted without payment of copyright license fees provided that you
38
satisfy the following conditions:
39
40
You must retain the complete text of this software license in redistributions of
41
the FDK AAC Codec or your modifications thereto in source code form.
42
43
You must retain the complete text of this software license in the documentation
44
and/or other materials provided with redistributions of the FDK AAC Codec or
45
your modifications thereto in binary form. You must make available free of
46
charge copies of the complete source code of the FDK AAC Codec and your
47
modifications thereto to recipients of copies in binary form.
48
49
The name of Fraunhofer may not be used to endorse or promote products derived
50
from this library without prior written permission.
51
52
You may not charge copyright license fees for anyone to use, copy or distribute
53
the FDK AAC Codec software or your modifications thereto.
54
55
Your modified versions of the FDK AAC Codec must carry prominent notices stating
56
that you changed the software and the date of any change. For modified versions
57
of the FDK AAC Codec, the term "Fraunhofer FDK AAC Codec Library for Android"
58
must be replaced by the term "Third-Party Modified Version of the Fraunhofer FDK
59
AAC Codec Library for Android."
60
61
3.    NO PATENT LICENSE
62
63
NO EXPRESS OR IMPLIED LICENSES TO ANY PATENT CLAIMS, including without
64
limitation the patents of Fraunhofer, ARE GRANTED BY THIS SOFTWARE LICENSE.
65
Fraunhofer provides no warranty of patent non-infringement with respect to this
66
software.
67
68
You may use this FDK AAC Codec software or modifications thereto only for
69
purposes that are authorized by appropriate patent licenses.
70
71
4.    DISCLAIMER
72
73
This FDK AAC Codec software is provided by Fraunhofer on behalf of the copyright
74
holders and contributors "AS IS" and WITHOUT ANY EXPRESS OR IMPLIED WARRANTIES,
75
including but not limited to the implied warranties of merchantability and
76
fitness for a particular purpose. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
77
CONTRIBUTORS BE LIABLE for any direct, indirect, incidental, special, exemplary,
78
or consequential damages, including but not limited to procurement of substitute
79
goods or services; loss of use, data, or profits, or business interruption,
80
however caused and on any theory of liability, whether in contract, strict
81
liability, or tort (including negligence), arising in any way out of the use of
82
this software, even if advised of the possibility of such damage.
83
84
5.    CONTACT INFORMATION
85
86
Fraunhofer Institute for Integrated Circuits IIS
87
Attention: Audio and Multimedia Departments - FDK AAC LL
88
Am Wolfsmantel 33
89
91058 Erlangen, Germany
90
91
www.iis.fraunhofer.de/amm
92
amm-info@iis.fraunhofer.de
93
----------------------------------------------------------------------------- */
94
95
/*********************** MPEG surround decoder library *************************
96
97
   Author(s):
98
99
   Description: SAC Processing
100
101
*******************************************************************************/
102
103
/* data structures and interfaces for spatial audio reference software */
104
#include "sac_process.h"
105
106
#include "sac_bitdec.h"
107
#include "sac_calcM1andM2.h"
108
#include "sac_smoothing.h"
109
#include "sac_rom.h"
110
111
#include "sac_dec_errorcodes.h"
112
113
#include "FDK_trigFcts.h"
114
#include "FDK_decorrelate.h"
115
116
88.2M
#define SAC_DEC_APPLY_M2_SCALE(spec, s) ((spec) >> (-(s)))
117
118
/**
119
 * \brief  Linear interpolation between two parameter values.
120
 *         a*alpha + b*(1-alpha)
121
 *       = a*alpha + b - b*alpha
122
 *
123
 * \param alpha               Weighting factor.
124
 * \param a                   Parameter a.
125
 * \param b                   Parameter b.
126
 *
127
 * \return Interpolated parameter value.
128
 */
129
FDK_INLINE FIXP_DBL interpolateParameter(const FIXP_SGL alpha, const FIXP_DBL a,
130
281M
                                         const FIXP_DBL b) {
131
281M
  return (b - fMult(alpha, b) + fMult(alpha, a));
132
281M
}
133
134
/**
135
 * \brief Map MPEG Surround channel indices to MPEG 4 PCE like channel indices.
136
 * \param self Spatial decoder handle.
137
 * \param ch MPEG Surround channel index.
138
 * \return MPEG 4 PCE style channel index, corresponding to the given MPEG
139
 * Surround channel index.
140
 */
141
14.1M
static UINT mapChannel(spatialDec *self, UINT ch) {
142
14.1M
  static const UCHAR chanelIdx[][8] = {
143
14.1M
      {0, 1, 2, 3, 4, 5, 6, 7}, /*  binaural, TREE_212, arbitrary tree */
144
14.1M
  };
145
146
14.1M
  int idx = 0;
147
148
14.1M
  return (chanelIdx[idx][ch]);
149
14.1M
}
150
151
143k
FIXP_DBL getChGain(spatialDec *self, UINT ch, INT *scale) {
152
  /* init no gain modifier */
153
143k
  FIXP_DBL gain = 0x80000000;
154
143k
  *scale = 0;
155
156
143k
  if ((!isTwoChMode(self->upmixType)) &&
157
143k
      (self->upmixType != UPMIXTYPE_BYPASS)) {
158
143k
    if ((ch == 0) || (ch == 1) || (ch == 2)) {
159
      /* no modifier */
160
143k
    }
161
143k
  }
162
163
143k
  return gain;
164
143k
}
165
166
SACDEC_ERROR SpatialDecQMFAnalysis(spatialDec *self, const PCM_MPS *inData,
167
                                   const INT ts, const INT bypassMode,
168
                                   FIXP_DBL **qmfReal, FIXP_DBL **qmfImag,
169
2.17M
                                   const int numInputChannels) {
170
2.17M
  SACDEC_ERROR err = MPS_OK;
171
2.17M
  int ch, offset;
172
173
2.17M
  offset = self->pQmfDomain->globalConf.nBandsSynthesis *
174
2.17M
           self->pQmfDomain->globalConf.nQmfTimeSlots;
175
176
2.17M
  {
177
4.34M
    for (ch = 0; ch < numInputChannels; ch++) {
178
2.17M
      const PCM_MPS *inSamples =
179
2.17M
          &inData[ts * self->pQmfDomain->globalConf.nBandsAnalysis];
180
2.17M
      FIXP_DBL *pQmfRealAnalysis = qmfReal[ch]; /* no delay in blind mode */
181
2.17M
      FIXP_DBL *pQmfImagAnalysis = qmfImag[ch];
182
183
2.17M
      CalculateSpaceAnalysisQmf(&self->pQmfDomain->QmfDomainIn[ch].fb,
184
2.17M
                                inSamples + (ch * offset), pQmfRealAnalysis,
185
2.17M
                                pQmfImagAnalysis);
186
187
2.17M
      if (!isTwoChMode(self->upmixType) && !bypassMode) {
188
2.17M
        int i;
189
51.8M
        for (i = 0; i < self->qmfBands; i++) {
190
49.6M
          qmfReal[ch][i] = fMult(
191
49.6M
              scaleValueSaturate(qmfReal[ch][i], self->sacInDataHeadroom - (1)),
192
49.6M
              self->clipProtectGain__FDK);
193
49.6M
          qmfImag[ch][i] = fMult(
194
49.6M
              scaleValueSaturate(qmfImag[ch][i], self->sacInDataHeadroom - (1)),
195
49.6M
              self->clipProtectGain__FDK);
196
49.6M
        }
197
2.17M
      }
198
2.17M
    }
199
2.17M
  }
200
201
2.17M
  self->qmfInputDelayBufPos =
202
2.17M
      (self->qmfInputDelayBufPos + 1) % self->pc_filterdelay;
203
204
2.17M
  return err;
205
2.17M
}
206
207
SACDEC_ERROR SpatialDecFeedQMF(spatialDec *self, FIXP_DBL **qmfInDataReal,
208
                               FIXP_DBL **qmfInDataImag, const INT ts,
209
                               const INT bypassMode, FIXP_DBL **qmfReal__FDK,
210
                               FIXP_DBL **qmfImag__FDK,
211
4.92M
                               const INT numInputChannels) {
212
4.92M
  SACDEC_ERROR err = MPS_OK;
213
4.92M
  int ch;
214
215
4.92M
  {
216
9.85M
    for (ch = 0; ch < numInputChannels; ch++) {
217
4.92M
      FIXP_DBL *pQmfRealAnalysis =
218
4.92M
          qmfReal__FDK[ch]; /* no delay in blind mode */
219
4.92M
      FIXP_DBL *pQmfImagAnalysis = qmfImag__FDK[ch];
220
221
      /* Write Input data to pQmfRealAnalysis. */
222
4.92M
      if (self->bShareDelayWithSBR) {
223
4.90M
        FDK_QmfDomain_GetSlot(&self->pQmfDomain->QmfDomainIn[ch],
224
4.90M
                              ts + HYBRID_FILTER_DELAY, 0,
225
4.90M
                              MAX_QMF_BANDS_TO_HYBRID, pQmfRealAnalysis,
226
4.90M
                              pQmfImagAnalysis, 15 + (1));
227
4.90M
        FDK_QmfDomain_GetSlot(&self->pQmfDomain->QmfDomainIn[ch], ts,
228
4.90M
                              MAX_QMF_BANDS_TO_HYBRID, self->qmfBands,
229
4.90M
                              pQmfRealAnalysis, pQmfImagAnalysis, 15 + (1));
230
4.90M
      } else {
231
19.6k
        FDK_QmfDomain_GetSlot(&self->pQmfDomain->QmfDomainIn[ch], ts, 0,
232
19.6k
                              self->qmfBands, pQmfRealAnalysis,
233
19.6k
                              pQmfImagAnalysis, 15 + (1));
234
19.6k
      }
235
4.92M
      if (ts == self->pQmfDomain->globalConf.nQmfTimeSlots - 1) {
236
        /* Is currently also needed in case we dont have any overlap. We need to
237
         * save lb_scale to ov_lb_scale */
238
153k
        FDK_QmfDomain_SaveOverlap(&self->pQmfDomain->QmfDomainIn[ch], 0);
239
153k
      }
240
241
      /* Apply clip protection to output. */
242
4.92M
      if (!isTwoChMode(self->upmixType) && !bypassMode) {
243
4.92M
        int i;
244
320M
        for (i = 0; i < self->qmfBands; i++) {
245
315M
          qmfReal__FDK[ch][i] =
246
315M
              fMult(qmfReal__FDK[ch][i], self->clipProtectGain__FDK);
247
315M
          qmfImag__FDK[ch][i] =
248
315M
              fMult(qmfImag__FDK[ch][i], self->clipProtectGain__FDK);
249
315M
        }
250
4.92M
      }
251
252
4.92M
    } /* End of loop over numInputChannels */
253
4.92M
  }
254
255
4.92M
  self->qmfInputDelayBufPos =
256
4.92M
      (self->qmfInputDelayBufPos + 1) % self->pc_filterdelay;
257
258
4.92M
  return err;
259
4.92M
}
260
261
/*******************************************************************************
262
 Functionname: SpatialDecHybridAnalysis
263
 *******************************************************************************
264
265
 Description:
266
267
 Arguments:
268
269
 Input:
270
  float** pointers[4] leftReal, leftIm, rightReal, rightIm
271
272
 Output:
273
  float self->qmfInputReal[MAX_INPUT_CHANNELS][MAX_TIME_SLOTS][MAX_QMF_BANDS];
274
  float self->qmfInputImag[MAX_INPUT_CHANNELS][MAX_TIME_SLOTS][MAX_QMF_BANDS];
275
276
  float
277
self->hybInputReal[MAX_INPUT_CHANNELS][MAX_TIME_SLOTS][MAX_HYBRID_BANDS]; float
278
self->hybInputImag[MAX_INPUT_CHANNELS][MAX_TIME_SLOTS][MAX_HYBRID_BANDS];
279
280
281
*******************************************************************************/
282
SACDEC_ERROR SpatialDecHybridAnalysis(spatialDec *self, FIXP_DBL **qmfInputReal,
283
                                      FIXP_DBL **qmfInputImag,
284
                                      FIXP_DBL **hybOutputReal,
285
                                      FIXP_DBL **hybOutputImag, const INT ts,
286
7.09M
                                      const INT numInputChannels) {
287
7.09M
  SACDEC_ERROR err = MPS_OK;
288
7.09M
  int ch;
289
290
14.1M
  for (ch = 0; ch < numInputChannels;
291
7.09M
       ch++) /* hybrid filtering for down-mix signals */
292
7.09M
  {
293
7.09M
    if (self->pConfigCurrent->syntaxFlags & SACDEC_SYNTAX_LD) {
294
806k
      int k;
295
      /* No hybrid filtering. Just copy the QMF data. */
296
27.4M
      for (k = 0; k < self->hybridBands; k += 1) {
297
26.6M
        hybOutputReal[ch][k] = qmfInputReal[ch][k];
298
26.6M
        hybOutputImag[ch][k] = qmfInputImag[ch][k];
299
26.6M
      }
300
6.29M
    } else {
301
6.29M
      self->hybridAnalysis[ch].hfMode = self->bShareDelayWithSBR;
302
303
6.29M
      if (self->stereoConfigIndex == 3)
304
6.29M
        FDK_ASSERT(self->hybridAnalysis[ch].hfMode == 0);
305
6.29M
      FDKhybridAnalysisApply(&self->hybridAnalysis[ch], qmfInputReal[ch],
306
6.29M
                             qmfInputImag[ch], hybOutputReal[ch],
307
6.29M
                             hybOutputImag[ch]);
308
6.29M
    }
309
7.09M
  }
310
311
7.09M
  if ((self->pConfigCurrent->syntaxFlags & SACDEC_SYNTAX_USAC) &&
312
7.09M
      self->residualCoding) {
313
1.48M
    self->hybridAnalysis[numInputChannels].hfMode = 0;
314
1.48M
    FDKhybridAnalysisApply(
315
1.48M
        &self->hybridAnalysis[numInputChannels],
316
1.48M
        self->qmfResidualReal__FDK[0][0], self->qmfResidualImag__FDK[0][0],
317
1.48M
        self->hybResidualReal__FDK[0], self->hybResidualImag__FDK[0]);
318
1.48M
  }
319
320
7.09M
  return err;
321
7.09M
}
322
323
SACDEC_ERROR SpatialDecCreateX(spatialDec *self, FIXP_DBL **hybInputReal,
324
                               FIXP_DBL **hybInputImag, FIXP_DBL **pxReal,
325
7.09M
                               FIXP_DBL **pxImag) {
326
7.09M
  SACDEC_ERROR err = MPS_OK;
327
7.09M
  int row;
328
329
  /* Creating wDry */
330
14.1M
  for (row = 0; row < self->numInputChannels; row++) {
331
    /* pointer to direct signals */
332
7.09M
    pxReal[row] = hybInputReal[row];
333
7.09M
    pxImag[row] = hybInputImag[row];
334
7.09M
  }
335
336
7.09M
  return err;
337
7.09M
}
338
339
static void M2ParamToKernelMult(FIXP_SGL *RESTRICT pKernel,
340
                                FIXP_DBL *RESTRICT Mparam,
341
                                FIXP_DBL *RESTRICT MparamPrev,
342
                                int *RESTRICT pWidth, FIXP_SGL alpha__FDK,
343
23.4M
                                int nBands) {
344
23.4M
  int pb;
345
346
247M
  for (pb = 0; pb < nBands; pb++) {
347
224M
    FIXP_SGL tmp = FX_DBL2FX_SGL(
348
224M
        interpolateParameter(alpha__FDK, Mparam[pb], MparamPrev[pb]));
349
350
224M
    int i = pWidth[pb];
351
224M
    if (i & 1) *pKernel++ = tmp;
352
224M
    if (i & 2) {
353
98.3M
      *pKernel++ = tmp;
354
98.3M
      *pKernel++ = tmp;
355
98.3M
    }
356
520M
    for (i >>= 2; i--;) {
357
295M
      *pKernel++ = tmp;
358
295M
      *pKernel++ = tmp;
359
295M
      *pKernel++ = tmp;
360
295M
      *pKernel++ = tmp;
361
295M
    }
362
224M
  }
363
23.4M
}
364
365
SACDEC_ERROR SpatialDecApplyM1_CreateW_Mode212(
366
    spatialDec *self, const SPATIAL_BS_FRAME *frame, FIXP_DBL **xReal,
367
7.09M
    FIXP_DBL **xImag, FIXP_DBL **vReal, FIXP_DBL **vImag) {
368
7.09M
  SACDEC_ERROR err = MPS_OK;
369
7.09M
  int res;
370
7.09M
  FIXP_DBL *decorrInReal = vReal[0];
371
7.09M
  FIXP_DBL *decorrInImag = vImag[0];
372
373
  /* M1 does not do anything in 212 mode, so use simplified processing */
374
7.09M
  FDK_ASSERT(self->numVChannels == 2);
375
7.09M
  FDK_ASSERT(self->numDirektSignals == 1);
376
7.09M
  FDK_ASSERT(self->numDecorSignals == 1);
377
7.09M
  FDKmemcpy(vReal[0], xReal[0], self->hybridBands * sizeof(FIXP_DBL));
378
7.09M
  FDKmemcpy(vImag[0], xImag[0], self->hybridBands * sizeof(FIXP_DBL));
379
380
7.09M
  if (isTsdActive(frame->TsdData)) {
381
    /* Generate v_{x,nonTr} as input for allpass based decorrelator */
382
36.7k
    TsdGenerateNonTr(self->hybridBands, frame->TsdData, self->TsdTs, vReal[0],
383
36.7k
                     vImag[0], vReal[1], vImag[1], &decorrInReal,
384
36.7k
                     &decorrInImag);
385
36.7k
  }
386
  /* - Decorrelate */
387
7.09M
  res = SpatialDecGetResidualIndex(self, 1);
388
7.09M
  if (FDKdecorrelateApply(&self->apDecor[0], decorrInReal, decorrInImag,
389
7.09M
                          vReal[1], vImag[1],
390
7.09M
                          self->param2hyb[self->residualBands[res]])) {
391
0
    return MPS_NOTOK;
392
0
  }
393
7.09M
  if (isTsdActive(frame->TsdData)) {
394
    /* Generate v_{x,Tr}, apply transient decorrelator and add to allpass based
395
     * decorrelator output */
396
36.7k
    TsdApply(self->hybridBands, frame->TsdData, &self->TsdTs,
397
36.7k
             vReal[0], /* input: v_x */
398
36.7k
             vImag[0],
399
36.7k
             vReal[1], /* input: d_{x,nonTr}; output: d_{x,nonTr} + d_{x,Tr} */
400
36.7k
             vImag[1]);
401
36.7k
  }
402
403
  /* Write residual signal in approriate parameter bands */
404
7.09M
  if (self->residualBands[res] > 0) {
405
1.36M
    int stopBand = self->param2hyb[self->residualBands[res]];
406
1.36M
    FDKmemcpy(vReal[1], self->hybResidualReal__FDK[res],
407
1.36M
              fixMin(stopBand, self->hybridBands) * sizeof(FIXP_DBL));
408
1.36M
    FDKmemcpy(vImag[1], self->hybResidualImag__FDK[res],
409
1.36M
              fixMin(stopBand, self->hybridBands) * sizeof(FIXP_DBL));
410
1.36M
  } /* (self->residualBands[res]>0) */
411
412
7.09M
  return err;
413
7.09M
}
414
415
SACDEC_ERROR SpatialDecApplyM2_Mode212(spatialDec *self, INT ps,
416
                                       const FIXP_SGL alpha, FIXP_DBL **wReal,
417
                                       FIXP_DBL **wImag,
418
                                       FIXP_DBL **hybOutputRealDry,
419
1.25M
                                       FIXP_DBL **hybOutputImagDry) {
420
1.25M
  SACDEC_ERROR err = MPS_OK;
421
1.25M
  INT row;
422
423
1.25M
  INT *pWidth = self->kernels_width;
424
  /* for stereoConfigIndex == 3 case hybridBands is < 71 */
425
1.25M
  INT pb_max = self->kernels[self->hybridBands - 1] + 1;
426
1.25M
  INT max_row = self->numOutputChannels;
427
428
1.25M
  INT M2_exp = 0;
429
1.25M
  if (self->residualCoding) M2_exp = 3;
430
431
3.75M
  for (row = 0; row < max_row; row++)  // 2 times
432
2.50M
  {
433
2.50M
    FIXP_DBL *Mparam0 = self->M2Real__FDK[row][0];
434
2.50M
    FIXP_DBL *Mparam1 = self->M2Real__FDK[row][1];
435
2.50M
    FIXP_DBL *MparamPrev0 = self->M2RealPrev__FDK[row][0];
436
2.50M
    FIXP_DBL *MparamPrev1 = self->M2RealPrev__FDK[row][1];
437
438
2.50M
    FIXP_DBL *RESTRICT pHybOutRealDry = hybOutputRealDry[row];
439
2.50M
    FIXP_DBL *RESTRICT pHybOutImagDry = hybOutputImagDry[row];
440
441
2.50M
    FIXP_DBL *RESTRICT pWReal0 = wReal[0];
442
2.50M
    FIXP_DBL *RESTRICT pWReal1 = wReal[1];
443
2.50M
    FIXP_DBL *RESTRICT pWImag0 = wImag[0];
444
2.50M
    FIXP_DBL *RESTRICT pWImag1 = wImag[1];
445
23.3M
    for (INT pb = 0; pb < pb_max; pb++) {
446
20.8M
      FIXP_DBL tmp0, tmp1;
447
448
20.8M
      tmp0 = interpolateParameter(alpha, Mparam0[pb], MparamPrev0[pb]);
449
20.8M
      tmp1 = interpolateParameter(alpha, Mparam1[pb], MparamPrev1[pb]);
450
451
20.8M
      INT i = pWidth[pb];
452
453
20.8M
      do  // about 3-4 times
454
72.2M
      {
455
72.2M
        FIXP_DBL var0, var1, real, imag;
456
457
72.2M
        var0 = *pWReal0++;
458
72.2M
        var1 = *pWReal1++;
459
72.2M
        real = fMultDiv2(var0, tmp0);
460
72.2M
        var0 = *pWImag0++;
461
72.2M
        real = fMultAddDiv2(real, var1, tmp1);
462
72.2M
        var1 = *pWImag1++;
463
72.2M
        imag = fMultDiv2(var0, tmp0);
464
72.2M
        *pHybOutRealDry++ = real << (1 + M2_exp);
465
72.2M
        imag = fMultAddDiv2(imag, var1, tmp1);
466
72.2M
        *pHybOutImagDry++ = imag << (1 + M2_exp);
467
72.2M
      } while (--i != 0);
468
20.8M
    }
469
2.50M
  }
470
1.25M
  return err;
471
1.25M
}
472
473
SACDEC_ERROR SpatialDecApplyM2_Mode212_ResidualsPlusPhaseCoding(
474
    spatialDec *self, INT ps, const FIXP_SGL alpha, FIXP_DBL **wReal,
475
    FIXP_DBL **wImag, FIXP_DBL **hybOutputRealDry,
476
95.5k
    FIXP_DBL **hybOutputImagDry) {
477
95.5k
  SACDEC_ERROR err = MPS_OK;
478
95.5k
  INT row;
479
95.5k
  INT scale_param_m2;
480
95.5k
  INT *pWidth = self->kernels_width;
481
95.5k
  INT pb_max = self->kernels[self->hybridBands - 1] + 1;
482
483
95.5k
  scale_param_m2 = SCALE_PARAM_M2_212_PRED + SCALE_DATA_APPLY_M2;
484
485
286k
  for (row = 0; row < self->numM2rows; row++) {
486
191k
    INT qs, pb;
487
488
191k
    FIXP_DBL *RESTRICT pWReal0 = wReal[0];
489
191k
    FIXP_DBL *RESTRICT pWImag0 = wImag[0];
490
191k
    FIXP_DBL *RESTRICT pWReal1 = wReal[1];
491
191k
    FIXP_DBL *RESTRICT pWImag1 = wImag[1];
492
493
191k
    FIXP_DBL *MReal0 = self->M2Real__FDK[row][0];
494
191k
    FIXP_DBL *MImag0 = self->M2Imag__FDK[row][0];
495
191k
    FIXP_DBL *MReal1 = self->M2Real__FDK[row][1];
496
191k
    FIXP_DBL *MRealPrev0 = self->M2RealPrev__FDK[row][0];
497
191k
    FIXP_DBL *MImagPrev0 = self->M2ImagPrev__FDK[row][0];
498
191k
    FIXP_DBL *MRealPrev1 = self->M2RealPrev__FDK[row][1];
499
500
191k
    FIXP_DBL *RESTRICT pHybOutRealDry = hybOutputRealDry[row];
501
191k
    FIXP_DBL *RESTRICT pHybOutImagDry = hybOutputImagDry[row];
502
503
191k
    FDK_ASSERT(!(self->pConfigCurrent->syntaxFlags & SACDEC_SYNTAX_LD));
504
191k
    FDK_ASSERT((pWidth[0] + pWidth[1]) >= 3);
505
506
573k
    for (pb = 0, qs = 3; pb < 2; pb++) {
507
382k
      INT s;
508
382k
      FIXP_DBL maxVal;
509
382k
      FIXP_DBL mReal1;
510
382k
      FIXP_DBL mReal0, mImag0;
511
382k
      FIXP_DBL iReal0, iImag0, iReal1;
512
513
382k
      iReal0 = interpolateParameter(alpha, MReal0[pb], MRealPrev0[pb]);
514
382k
      iImag0 = -interpolateParameter(alpha, MImag0[pb], MImagPrev0[pb]);
515
382k
      iReal1 = interpolateParameter(alpha, MReal1[pb], MRealPrev1[pb]);
516
517
382k
      maxVal = fAbs(iReal0) | fAbs(iImag0);
518
382k
      maxVal |= fAbs(iReal1);
519
520
382k
      s = fMin(CntLeadingZeros(maxVal) - 2, scale_param_m2);
521
522
382k
      mReal0 = scaleValue(iReal0, s);
523
382k
      mImag0 = scaleValue(iImag0, s);
524
382k
      mReal1 = scaleValue(iReal1, s);
525
526
382k
      s = scale_param_m2 - s;
527
528
382k
      INT i = pWidth[pb];
529
530
1.19M
      do {
531
1.19M
        FIXP_DBL real, imag, wReal0, wImag0, wReal1, wImag1;
532
533
1.19M
        wReal0 = *pWReal0++;
534
1.19M
        wImag0 = *pWImag0++;
535
1.19M
        wReal1 = *pWReal1++;
536
1.19M
        wImag1 = *pWImag1++;
537
538
1.19M
        cplxMultDiv2(&real, &imag, wReal0, wImag0, mReal0, mImag0);
539
540
1.19M
        *pHybOutRealDry++ = fMultAddDiv2(real, wReal1, mReal1) << s;
541
1.19M
        *pHybOutImagDry++ = fMultAddDiv2(imag, wImag1, mReal1) << s;
542
543
1.19M
        if (qs > 0) {
544
573k
          mImag0 = -mImag0;
545
573k
          qs--;
546
573k
        }
547
1.19M
      } while (--i != 0);
548
382k
    }
549
550
2.34M
    for (; pb < pb_max; pb++) {
551
2.15M
      INT s;
552
2.15M
      FIXP_DBL maxVal;
553
2.15M
      FIXP_SGL mReal1;
554
2.15M
      FIXP_SGL mReal0, mImag0;
555
2.15M
      FIXP_DBL iReal0, iImag0, iReal1;
556
557
2.15M
      iReal0 = interpolateParameter(alpha, MReal0[pb], MRealPrev0[pb]);
558
2.15M
      iImag0 = interpolateParameter(alpha, MImag0[pb], MImagPrev0[pb]);
559
2.15M
      iReal1 = interpolateParameter(alpha, MReal1[pb], MRealPrev1[pb]);
560
561
2.15M
      maxVal = fAbs(iReal0) | fAbs(iImag0);
562
2.15M
      maxVal |= fAbs(iReal1);
563
564
2.15M
      s = fMin(CntLeadingZeros(maxVal) - 2, scale_param_m2);
565
566
2.15M
      mReal0 = FX_DBL2FX_SGL(scaleValue(iReal0, s));
567
2.15M
      mImag0 = FX_DBL2FX_SGL(scaleValue(iImag0, s));
568
2.15M
      mReal1 = FX_DBL2FX_SGL(scaleValue(iReal1, s));
569
570
2.15M
      s = scale_param_m2 - s;
571
572
2.15M
      INT i = pWidth[pb];
573
574
6.61M
      do {
575
6.61M
        FIXP_DBL real, imag, wReal0, wImag0, wReal1, wImag1;
576
577
6.61M
        wReal0 = *pWReal0++;
578
6.61M
        wImag0 = *pWImag0++;
579
6.61M
        wReal1 = *pWReal1++;
580
6.61M
        wImag1 = *pWImag1++;
581
582
6.61M
        cplxMultDiv2(&real, &imag, wReal0, wImag0, mReal0, mImag0);
583
584
6.61M
        *pHybOutRealDry++ = fMultAddDiv2(real, wReal1, mReal1) << s;
585
6.61M
        *pHybOutImagDry++ = fMultAddDiv2(imag, wImag1, mReal1) << s;
586
6.61M
      } while (--i != 0);
587
2.15M
    }
588
191k
  }
589
590
95.5k
  return err;
591
95.5k
}
592
593
SACDEC_ERROR SpatialDecApplyM2(spatialDec *self, INT ps, const FIXP_SGL alpha,
594
                               FIXP_DBL **wReal, FIXP_DBL **wImag,
595
                               FIXP_DBL **hybOutputRealDry,
596
                               FIXP_DBL **hybOutputImagDry,
597
                               FIXP_DBL **hybOutputRealWet,
598
5.75M
                               FIXP_DBL **hybOutputImagWet) {
599
5.75M
  SACDEC_ERROR err = MPS_OK;
600
601
5.75M
  {
602
5.75M
    int qs, row, col;
603
5.75M
    int complexHybBands;
604
5.75M
    int complexParBands;
605
5.75M
    int scale_param_m2 = 0;
606
5.75M
    int toolsDisabled;
607
608
5.75M
    UCHAR activParamBands;
609
5.75M
    FIXP_DBL *RESTRICT pWReal, *RESTRICT pWImag, *RESTRICT pHybOutRealDry,
610
5.75M
        *RESTRICT pHybOutImagDry, *RESTRICT pHybOutRealWet,
611
5.75M
        *RESTRICT pHybOutImagWet;
612
5.75M
    C_ALLOC_SCRATCH_START(pKernel, FIXP_SGL, MAX_HYBRID_BANDS);
613
614
    /* The wet signal is added to the dry signal directly in applyM2 if GES and
615
     * STP are disabled */
616
5.75M
    toolsDisabled =
617
5.75M
        ((self->tempShapeConfig == 1) || (self->tempShapeConfig == 2)) ? 0 : 1;
618
619
5.75M
    {
620
5.75M
      complexHybBands = self->hybridBands;
621
5.75M
      complexParBands = self->numParameterBands;
622
5.75M
    }
623
624
5.75M
    FDKmemclear(hybOutputImagDry[0],
625
5.75M
                self->createParams.maxNumOutputChannels *
626
5.75M
                    self->createParams.maxNumCmplxHybBands * sizeof(FIXP_DBL));
627
5.75M
    FDKmemclear(hybOutputRealDry[0], self->createParams.maxNumOutputChannels *
628
5.75M
                                         self->createParams.maxNumHybridBands *
629
5.75M
                                         sizeof(FIXP_DBL));
630
631
5.75M
    if (!toolsDisabled) {
632
5.48M
      FDKmemclear(hybOutputRealWet[0],
633
5.48M
                  self->createParams.maxNumOutputChannels *
634
5.48M
                      self->createParams.maxNumHybridBands * sizeof(FIXP_DBL));
635
5.48M
      FDKmemclear(hybOutputImagWet[0],
636
5.48M
                  self->createParams.maxNumOutputChannels *
637
5.48M
                      self->createParams.maxNumCmplxHybBands *
638
5.48M
                      sizeof(FIXP_DBL));
639
5.48M
    }
640
641
5.75M
    if (self->phaseCoding == 3) {
642
113k
      scale_param_m2 = -(SCALE_DATA_APPLY_M2_PC - 1);
643
113k
    }
644
645
17.2M
    for (row = 0; row < self->numM2rows; row++) {
646
11.5M
      pHybOutRealDry = hybOutputRealDry[row];
647
11.5M
      pHybOutImagDry = hybOutputImagDry[row];
648
649
11.5M
      if (toolsDisabled) {
650
544k
        pHybOutRealWet = hybOutputRealDry[row];
651
544k
        pHybOutImagWet = hybOutputImagDry[row];
652
10.9M
      } else {
653
10.9M
        pHybOutRealWet = hybOutputRealWet[row];
654
10.9M
        pHybOutImagWet = hybOutputImagWet[row];
655
10.9M
      }
656
657
23.0M
      for (col = 0; col < self->numDirektSignals; col++) {
658
11.5M
        if (self->pActivM2ParamBands ==
659
11.5M
            0) { /* default setting, calculate all rows and columns */
660
11.5M
          activParamBands = 1;
661
11.5M
        } else {
662
0
          if (self->pActivM2ParamBands[MAX_M2_INPUT * row +
663
0
                                       col]) /* table with activ and inactiv
664
                                                bands exists for current
665
                                                configuration */
666
0
            activParamBands = 1;
667
0
          else
668
0
            activParamBands = 0;
669
0
        }
670
11.5M
        if (activParamBands) {
671
11.5M
          pWReal = wReal[col];
672
11.5M
          pWImag = wImag[col];
673
674
11.5M
          M2ParamToKernelMult(pKernel, self->M2Real__FDK[row][col],
675
11.5M
                              self->M2RealPrev__FDK[row][col],
676
11.5M
                              self->kernels_width, alpha,
677
11.5M
                              self->numParameterBands);
678
679
11.5M
          if (1 && (self->phaseCoding != 3)) {
680
            /* direct signals */
681
11.2M
            {
682
              /* only one sample will be assigned to each row, hence
683
               * accumulation is not neccessary; that is valid for all
684
               * configurations */
685
737M
              for (qs = 0; qs < complexHybBands; qs++) {
686
726M
                pHybOutRealDry[qs] = fMult(pWReal[qs], pKernel[qs]);
687
726M
                pHybOutImagDry[qs] = fMult(pWImag[qs], pKernel[qs]);
688
726M
              }
689
11.2M
            }
690
11.2M
          } else { /*  isBinauralMode(self->upmixType)  */
691
692
11.2M
            for (qs = 0; qs < complexHybBands; qs++) {
693
11.0M
              pHybOutRealDry[qs] += SAC_DEC_APPLY_M2_SCALE(
694
11.0M
                  fMultDiv2(pWReal[qs], pKernel[qs]), scale_param_m2);
695
11.0M
              pHybOutImagDry[qs] += SAC_DEC_APPLY_M2_SCALE(
696
11.0M
                  fMultDiv2(pWImag[qs], pKernel[qs]), scale_param_m2);
697
11.0M
            }
698
699
226k
            M2ParamToKernelMult(pKernel, self->M2Imag__FDK[row][col],
700
226k
                                self->M2ImagPrev__FDK[row][col],
701
226k
                                self->kernels_width, alpha, complexParBands);
702
703
            /* direct signals sign is -1 for qs = 0,2 */
704
226k
            pHybOutRealDry[0] += SAC_DEC_APPLY_M2_SCALE(
705
226k
                fMultDiv2(pWImag[0], pKernel[0]), scale_param_m2);
706
226k
            pHybOutImagDry[0] -= SAC_DEC_APPLY_M2_SCALE(
707
226k
                fMultDiv2(pWReal[0], pKernel[0]), scale_param_m2);
708
709
226k
            pHybOutRealDry[2] += SAC_DEC_APPLY_M2_SCALE(
710
226k
                fMultDiv2(pWImag[2], pKernel[2]), scale_param_m2);
711
226k
            pHybOutImagDry[2] -= SAC_DEC_APPLY_M2_SCALE(
712
226k
                fMultDiv2(pWReal[2], pKernel[2]), scale_param_m2);
713
714
            /* direct signals sign is +1 for qs = 1,3,4,5,...,complexHybBands */
715
226k
            pHybOutRealDry[1] -= SAC_DEC_APPLY_M2_SCALE(
716
226k
                fMultDiv2(pWImag[1], pKernel[1]), scale_param_m2);
717
226k
            pHybOutImagDry[1] += SAC_DEC_APPLY_M2_SCALE(
718
226k
                fMultDiv2(pWReal[1], pKernel[1]), scale_param_m2);
719
720
10.5M
            for (qs = 3; qs < complexHybBands; qs++) {
721
10.3M
              pHybOutRealDry[qs] -= SAC_DEC_APPLY_M2_SCALE(
722
10.3M
                  fMultDiv2(pWImag[qs], pKernel[qs]), scale_param_m2);
723
10.3M
              pHybOutImagDry[qs] += SAC_DEC_APPLY_M2_SCALE(
724
10.3M
                  fMultDiv2(pWReal[qs], pKernel[qs]), scale_param_m2);
725
10.3M
            }
726
226k
          } /* self->upmixType */
727
11.5M
        }   /* if (activParamBands) */
728
11.5M
      }     /* self->numDirektSignals */
729
730
23.0M
      for (; col < self->numVChannels; col++) {
731
11.5M
        if (self->pActivM2ParamBands ==
732
11.5M
            0) { /* default setting, calculate all rows and columns */
733
11.5M
          activParamBands = 1;
734
11.5M
        } else {
735
0
          if (self->pActivM2ParamBands[MAX_M2_INPUT * row +
736
0
                                       col]) /* table with activ and inactiv
737
                                                bands exists for current
738
                                                configuration */
739
0
            activParamBands = 1;
740
0
          else
741
0
            activParamBands = 0;
742
0
        }
743
744
11.5M
        if (activParamBands) {
745
11.5M
          int resBandIndex;
746
11.5M
          int resHybIndex;
747
748
11.5M
          resBandIndex =
749
11.5M
              self->residualBands[SpatialDecGetResidualIndex(self, col)];
750
11.5M
          resHybIndex = self->param2hyb[resBandIndex];
751
752
11.5M
          pWReal = wReal[col];
753
11.5M
          pWImag = wImag[col];
754
755
11.5M
          M2ParamToKernelMult(pKernel, self->M2Real__FDK[row][col],
756
11.5M
                              self->M2RealPrev__FDK[row][col],
757
11.5M
                              self->kernels_width, alpha,
758
11.5M
                              self->numParameterBands);
759
760
11.5M
          if (1 && (self->phaseCoding != 3)) {
761
            /* residual signals */
762
14.3M
            for (qs = 0; qs < resHybIndex; qs++) {
763
3.08M
              pHybOutRealDry[qs] += fMult(pWReal[qs], pKernel[qs]);
764
3.08M
              pHybOutImagDry[qs] += fMult(pWImag[qs], pKernel[qs]);
765
3.08M
            }
766
            /* decor signals */
767
734M
            for (; qs < complexHybBands; qs++) {
768
723M
              pHybOutRealWet[qs] += fMult(pWReal[qs], pKernel[qs]);
769
723M
              pHybOutImagWet[qs] += fMult(pWImag[qs], pKernel[qs]);
770
723M
            }
771
11.2M
          } else { /* self->upmixType */
772
            /* residual signals */
773
226k
            FIXP_DBL *RESTRICT pHybOutReal;
774
226k
            FIXP_DBL *RESTRICT pHybOutImag;
775
776
1.37M
            for (qs = 0; qs < resHybIndex; qs++) {
777
1.14M
              pHybOutRealDry[qs] += SAC_DEC_APPLY_M2_SCALE(
778
1.14M
                  fMultDiv2(pWReal[qs], pKernel[qs]), scale_param_m2);
779
1.14M
              pHybOutImagDry[qs] += SAC_DEC_APPLY_M2_SCALE(
780
1.14M
                  fMultDiv2(pWImag[qs], pKernel[qs]), scale_param_m2);
781
1.14M
            }
782
            /* decor signals */
783
10.1M
            for (; qs < complexHybBands; qs++) {
784
9.88M
              pHybOutRealWet[qs] += SAC_DEC_APPLY_M2_SCALE(
785
9.88M
                  fMultDiv2(pWReal[qs], pKernel[qs]), scale_param_m2);
786
9.88M
              pHybOutImagWet[qs] += SAC_DEC_APPLY_M2_SCALE(
787
9.88M
                  fMultDiv2(pWImag[qs], pKernel[qs]), scale_param_m2);
788
9.88M
            }
789
790
226k
            M2ParamToKernelMult(pKernel, self->M2Imag__FDK[row][col],
791
226k
                                self->M2ImagPrev__FDK[row][col],
792
226k
                                self->kernels_width, alpha, complexParBands);
793
794
            /* direct signals sign is -1 for qs = 0,2 */
795
            /* direct signals sign is +1 for qs = 1,3.. */
796
226k
            if (toolsDisabled) {
797
0
              pHybOutRealDry[0] += SAC_DEC_APPLY_M2_SCALE(
798
0
                  fMultDiv2(pWImag[0], pKernel[0]), scale_param_m2);
799
0
              pHybOutImagDry[0] -= SAC_DEC_APPLY_M2_SCALE(
800
0
                  fMultDiv2(pWReal[0], pKernel[0]), scale_param_m2);
801
802
0
              pHybOutRealDry[1] -= SAC_DEC_APPLY_M2_SCALE(
803
0
                  fMultDiv2(pWImag[1], pKernel[1]), scale_param_m2);
804
0
              pHybOutImagDry[1] += SAC_DEC_APPLY_M2_SCALE(
805
0
                  fMultDiv2(pWReal[1], pKernel[1]), scale_param_m2);
806
807
0
              pHybOutRealDry[2] += SAC_DEC_APPLY_M2_SCALE(
808
0
                  fMultDiv2(pWImag[2], pKernel[2]), scale_param_m2);
809
0
              pHybOutImagDry[2] -= SAC_DEC_APPLY_M2_SCALE(
810
0
                  fMultDiv2(pWReal[2], pKernel[2]), scale_param_m2);
811
226k
            } else {
812
226k
              pHybOutReal = &pHybOutRealDry[0];
813
226k
              pHybOutImag = &pHybOutImagDry[0];
814
226k
              if (0 == resHybIndex) {
815
97.7k
                pHybOutReal = &pHybOutRealWet[0];
816
97.7k
                pHybOutImag = &pHybOutImagWet[0];
817
97.7k
              }
818
226k
              pHybOutReal[0] += SAC_DEC_APPLY_M2_SCALE(
819
226k
                  fMultDiv2(pWImag[0], pKernel[0]), scale_param_m2);
820
226k
              pHybOutImag[0] -= SAC_DEC_APPLY_M2_SCALE(
821
226k
                  fMultDiv2(pWReal[0], pKernel[0]), scale_param_m2);
822
823
226k
              if (1 == resHybIndex) {
824
0
                pHybOutReal = &pHybOutRealWet[0];
825
0
                pHybOutImag = &pHybOutImagWet[0];
826
0
              }
827
226k
              pHybOutReal[1] -= SAC_DEC_APPLY_M2_SCALE(
828
226k
                  fMultDiv2(pWImag[1], pKernel[1]), scale_param_m2);
829
226k
              pHybOutImag[1] += SAC_DEC_APPLY_M2_SCALE(
830
226k
                  fMultDiv2(pWReal[1], pKernel[1]), scale_param_m2);
831
832
226k
              if (2 == resHybIndex) {
833
14.1k
                pHybOutReal = &pHybOutRealWet[0];
834
14.1k
                pHybOutImag = &pHybOutImagWet[0];
835
14.1k
              }
836
226k
              pHybOutReal[2] += SAC_DEC_APPLY_M2_SCALE(
837
226k
                  fMultDiv2(pWImag[2], pKernel[2]), scale_param_m2);
838
226k
              pHybOutImag[2] -= SAC_DEC_APPLY_M2_SCALE(
839
226k
                  fMultDiv2(pWReal[2], pKernel[2]), scale_param_m2);
840
226k
            }
841
842
1.00M
            for (qs = 3; qs < resHybIndex; qs++) {
843
774k
              pHybOutRealDry[qs] -= SAC_DEC_APPLY_M2_SCALE(
844
774k
                  fMultDiv2(pWImag[qs], pKernel[qs]), scale_param_m2);
845
774k
              pHybOutImagDry[qs] += SAC_DEC_APPLY_M2_SCALE(
846
774k
                  fMultDiv2(pWReal[qs], pKernel[qs]), scale_param_m2);
847
774k
            }
848
            /* decor signals */
849
9.80M
            for (; qs < complexHybBands; qs++) {
850
9.57M
              pHybOutRealWet[qs] -= SAC_DEC_APPLY_M2_SCALE(
851
9.57M
                  fMultDiv2(pWImag[qs], pKernel[qs]), scale_param_m2);
852
9.57M
              pHybOutImagWet[qs] += SAC_DEC_APPLY_M2_SCALE(
853
9.57M
                  fMultDiv2(pWReal[qs], pKernel[qs]), scale_param_m2);
854
9.57M
            }
855
226k
          } /* self->upmixType */
856
11.5M
        }   /* if (activParamBands) { */
857
11.5M
      }     /*  self->numVChannels */
858
859
11.5M
      if (self->phaseCoding == 3) {
860
226k
        scaleValuesSaturate(pHybOutRealDry, complexHybBands,
861
226k
                            SCALE_PARAM_M2_212_PRED + SCALE_DATA_APPLY_M2_PC);
862
226k
        scaleValuesSaturate(pHybOutImagDry, complexHybBands,
863
226k
                            SCALE_PARAM_M2_212_PRED + SCALE_DATA_APPLY_M2_PC);
864
865
226k
        if (!toolsDisabled) {
866
226k
          scaleValuesSaturate(pHybOutRealWet, complexHybBands,
867
226k
                              SCALE_PARAM_M2_212_PRED + SCALE_DATA_APPLY_M2_PC);
868
226k
          scaleValuesSaturate(pHybOutImagWet, complexHybBands,
869
226k
                              SCALE_PARAM_M2_212_PRED + SCALE_DATA_APPLY_M2_PC);
870
226k
        }
871
226k
      }
872
11.5M
    }
873
874
5.75M
    C_ALLOC_SCRATCH_END(pKernel, FIXP_SGL, MAX_HYBRID_BANDS);
875
5.75M
  }
876
877
5.75M
  return err;
878
5.75M
}
879
880
SACDEC_ERROR SpatialDecSynthesis(spatialDec *self, const INT ts,
881
                                 FIXP_DBL **hybOutputReal,
882
                                 FIXP_DBL **hybOutputImag, PCM_MPS *timeOut,
883
                                 const INT numInputChannels,
884
7.09M
                                 const FDK_channelMapDescr *const mapDescr) {
885
7.09M
  SACDEC_ERROR err = MPS_OK;
886
887
7.09M
  int ch;
888
7.09M
  int stride, offset;
889
890
7.09M
  stride = self->numOutputChannelsAT;
891
7.09M
  offset = 1;
892
893
7.09M
  PCM_MPS *pTimeOut__FDK =
894
7.09M
      &timeOut[stride * self->pQmfDomain->globalConf.nBandsSynthesis * ts];
895
7.09M
  C_ALLOC_SCRATCH_START(pQmfReal, FIXP_DBL, QMF_MAX_SYNTHESIS_BANDS);
896
7.09M
  C_ALLOC_SCRATCH_START(pQmfImag, FIXP_DBL, QMF_MAX_SYNTHESIS_BANDS);
897
898
21.2M
  for (ch = 0; ch < self->numOutputChannelsAT; ch++) {
899
14.1M
    if (self->pConfigCurrent->syntaxFlags & SACDEC_SYNTAX_LD) {
900
1.61M
      int k;
901
      /* No hybrid filtering. Just copy the QMF data. */
902
54.8M
      for (k = 0; k < self->hybridBands; k += 1) {
903
53.2M
        pQmfReal[k] = hybOutputReal[ch][k];
904
53.2M
        pQmfImag[k] = hybOutputImag[ch][k];
905
53.2M
      }
906
12.5M
    } else {
907
12.5M
      FDKhybridSynthesisApply(&self->hybridSynthesis[ch], hybOutputReal[ch],
908
12.5M
                              hybOutputImag[ch], pQmfReal, pQmfImag);
909
12.5M
    }
910
911
    /* Map channel indices from MPEG Surround -> PCE style -> channelMapping[]
912
     */
913
14.1M
    FDK_ASSERT(self->numOutputChannelsAT <= 6);
914
14.1M
    int outCh = FDK_chMapDescr_getMapValue(mapDescr, mapChannel(self, ch),
915
14.1M
                                           self->numOutputChannelsAT);
916
917
14.1M
    {
918
14.1M
      if (self->stereoConfigIndex == 3) {
919
        /* MPS -> SBR */
920
2.76M
        int i;
921
2.76M
        FIXP_DBL *pWorkBufReal, *pWorkBufImag;
922
2.76M
        FDK_ASSERT((self->pQmfDomain->QmfDomainOut[outCh].fb.outGain_m ==
923
2.76M
                    (FIXP_DBL)0x80000000) &&
924
2.76M
                   (self->pQmfDomain->QmfDomainOut[outCh].fb.outGain_e == 0));
925
2.76M
        FDK_QmfDomain_GetWorkBuffer(&self->pQmfDomain->QmfDomainIn[outCh], ts,
926
2.76M
                                    &pWorkBufReal, &pWorkBufImag);
927
2.76M
        FDK_ASSERT(self->qmfBands <=
928
2.76M
                   self->pQmfDomain->QmfDomainIn[outCh].workBuf_nBands);
929
50.8M
        for (i = 0; i < self->qmfBands; i++) {
930
48.0M
          pWorkBufReal[i] = pQmfReal[i];
931
48.0M
          pWorkBufImag[i] = pQmfImag[i];
932
48.0M
        }
933
2.76M
        self->pQmfDomain->QmfDomainIn[outCh].scaling.lb_scale =
934
2.76M
            -7; /*-ALGORITHMIC_SCALING_IN_ANALYSIS_FILTERBANK;*/
935
2.76M
        self->pQmfDomain->QmfDomainIn[outCh].scaling.lb_scale -=
936
2.76M
            self->pQmfDomain->QmfDomainIn[outCh].fb.filterScale;
937
2.76M
        self->pQmfDomain->QmfDomainIn[outCh].scaling.lb_scale -=
938
2.76M
            self->clipProtectGainSF__FDK;
939
940
2.76M
        self->pQmfDomain->QmfDomainIn[outCh].scaling.lb_scale -= (1);
941
11.4M
      } else {
942
        /* Call the QMF synthesis for dry. */
943
11.4M
        err = CalculateSpaceSynthesisQmf(&self->pQmfDomain->QmfDomainOut[outCh],
944
11.4M
                                         pQmfReal, pQmfImag, stride,
945
11.4M
                                         pTimeOut__FDK + (offset * outCh));
946
11.4M
      }
947
14.1M
      if (err != MPS_OK) goto bail;
948
14.1M
    }
949
14.1M
  } /* ch loop */
950
951
7.09M
bail:
952
7.09M
  C_ALLOC_SCRATCH_END(pQmfImag, FIXP_DBL, QMF_MAX_SYNTHESIS_BANDS);
953
7.09M
  C_ALLOC_SCRATCH_END(pQmfReal, FIXP_DBL, QMF_MAX_SYNTHESIS_BANDS);
954
955
7.09M
  return err;
956
7.09M
}
957
958
276k
void SpatialDecBufferMatrices(spatialDec *self) {
959
276k
  int row, col;
960
276k
  int complexParBands;
961
276k
  complexParBands = self->numParameterBands;
962
963
  /*
964
    buffer matrices M2
965
  */
966
830k
  for (row = 0; row < self->numM2rows; row++) {
967
1.66M
    for (col = 0; col < self->numVChannels; col++) {
968
1.10M
      FDKmemcpy(self->M2RealPrev__FDK[row][col], self->M2Real__FDK[row][col],
969
1.10M
                self->numParameterBands * sizeof(FIXP_DBL));
970
1.10M
      if (0 || (self->phaseCoding == 3)) {
971
29.7k
        FDKmemcpy(self->M2ImagPrev__FDK[row][col], self->M2Imag__FDK[row][col],
972
29.7k
                  complexParBands * sizeof(FIXP_DBL));
973
29.7k
      }
974
1.10M
    }
975
553k
  }
976
977
  /* buffer phase */
978
276k
  FDKmemcpy(self->PhasePrevLeft__FDK, self->PhaseLeft__FDK,
979
276k
            self->numParameterBands * sizeof(FIXP_DBL));
980
276k
  FDKmemcpy(self->PhasePrevRight__FDK, self->PhaseRight__FDK,
981
276k
            self->numParameterBands * sizeof(FIXP_DBL));
982
276k
}
983
984
113M
#define PHASE_SCALE 2
985
986
#ifndef P_PI
987
#define P_PI 3.1415926535897932
988
#endif
989
990
/* For better precision, PI (pi_x2) is already doubled */
991
static FIXP_DBL interp_angle__FDK(FIXP_DBL angle1, FIXP_DBL angle2,
992
7.98M
                                  FIXP_SGL alpha, FIXP_DBL pi_x2) {
993
7.98M
  if (angle2 - angle1 > (pi_x2 >> 1)) angle2 -= pi_x2;
994
995
7.98M
  if (angle1 - angle2 > (pi_x2 >> 1)) angle1 -= pi_x2;
996
997
7.98M
  return interpolateParameter(alpha, angle2, angle1);
998
7.98M
}
999
1000
/*
1001
 *
1002
 */
1003
void SpatialDecApplyPhase(spatialDec *self, FIXP_SGL alpha__FDK,
1004
400k
                          int lastSlotOfParamSet) {
1005
400k
  int pb, qs;
1006
400k
  FIXP_DBL ppb[MAX_PARAMETER_BANDS *
1007
400k
               4]; /* left real, imag - right real, imag interleaved */
1008
1009
400k
  const FIXP_DBL pi_x2 = PIx2__IPD;
1010
4.39M
  for (pb = 0; pb < self->numParameterBands; pb++) {
1011
3.99M
    FIXP_DBL pl, pr;
1012
1013
3.99M
    pl = interp_angle__FDK(self->PhasePrevLeft__FDK[pb],
1014
3.99M
                           self->PhaseLeft__FDK[pb], alpha__FDK, pi_x2);
1015
3.99M
    pr = interp_angle__FDK(self->PhasePrevRight__FDK[pb],
1016
3.99M
                           self->PhaseRight__FDK[pb], alpha__FDK, pi_x2);
1017
1018
3.99M
    inline_fixp_cos_sin(pl, pr, IPD_SCALE, &ppb[4 * pb]);
1019
3.99M
  }
1020
1021
  /* sign is -1 for qs = 0,2 and +1 for qs = 1 */
1022
1023
400k
  const SCHAR *kernels = &self->kernels[0];
1024
1025
400k
  FIXP_DBL *Dry_real0 = &self->hybOutputRealDry__FDK[0][0];
1026
400k
  FIXP_DBL *Dry_imag0 = &self->hybOutputImagDry__FDK[0][0];
1027
400k
  FIXP_DBL *Dry_real1 = &self->hybOutputRealDry__FDK[1][0];
1028
400k
  FIXP_DBL *Dry_imag1 = &self->hybOutputImagDry__FDK[1][0];
1029
1030
1.60M
  for (qs = 2; qs >= 0; qs--) {
1031
1.20M
    FIXP_DBL out_re, out_im;
1032
1033
1.20M
    pb = *kernels++;
1034
1.20M
    if (qs == 1) /* sign[qs] >= 0 */
1035
400k
    {
1036
400k
      cplxMultDiv2(&out_re, &out_im, *Dry_real0, *Dry_imag0, ppb[4 * pb + 0],
1037
400k
                   ppb[4 * pb + 1]);
1038
400k
      out_re <<= PHASE_SCALE - 1;
1039
400k
      out_im <<= PHASE_SCALE - 1;
1040
400k
      *Dry_real0++ = out_re;
1041
400k
      *Dry_imag0++ = out_im;
1042
1043
400k
      cplxMultDiv2(&out_re, &out_im, *Dry_real1, *Dry_imag1, ppb[4 * pb + 2],
1044
400k
                   ppb[4 * pb + 3]);
1045
400k
      out_re <<= PHASE_SCALE - 1;
1046
400k
      out_im <<= PHASE_SCALE - 1;
1047
400k
      *Dry_real1++ = out_re;
1048
400k
      *Dry_imag1++ = out_im;
1049
800k
    } else {
1050
800k
      cplxMultDiv2(&out_re, &out_im, *Dry_real0, *Dry_imag0, ppb[4 * pb + 0],
1051
800k
                   -ppb[4 * pb + 1]);
1052
800k
      out_re <<= PHASE_SCALE - 1;
1053
800k
      out_im <<= PHASE_SCALE - 1;
1054
800k
      *Dry_real0++ = out_re;
1055
800k
      *Dry_imag0++ = out_im;
1056
1057
800k
      cplxMultDiv2(&out_re, &out_im, *Dry_real1, *Dry_imag1, ppb[4 * pb + 2],
1058
800k
                   -ppb[4 * pb + 3]);
1059
800k
      out_re <<= PHASE_SCALE - 1;
1060
800k
      out_im <<= PHASE_SCALE - 1;
1061
800k
      *Dry_real1++ = out_re;
1062
800k
      *Dry_imag1++ = out_im;
1063
800k
    }
1064
1.20M
  }
1065
1066
  /* sign is +1 for qs >=3 */
1067
27.6M
  for (qs = self->hybridBands - 3; qs--;) {
1068
27.2M
    FIXP_DBL out_re, out_im;
1069
1070
27.2M
    pb = *kernels++;
1071
27.2M
    cplxMultDiv2(&out_re, &out_im, *Dry_real0, *Dry_imag0, ppb[4 * pb + 0],
1072
27.2M
                 ppb[4 * pb + 1]);
1073
27.2M
    out_re <<= PHASE_SCALE - 1;
1074
27.2M
    out_im <<= PHASE_SCALE - 1;
1075
27.2M
    *Dry_real0++ = out_re;
1076
27.2M
    *Dry_imag0++ = out_im;
1077
1078
27.2M
    cplxMultDiv2(&out_re, &out_im, *Dry_real1, *Dry_imag1, ppb[4 * pb + 2],
1079
27.2M
                 ppb[4 * pb + 3]);
1080
27.2M
    out_re <<= PHASE_SCALE - 1;
1081
27.2M
    out_im <<= PHASE_SCALE - 1;
1082
27.2M
    *Dry_real1++ = out_re;
1083
27.2M
    *Dry_imag1++ = out_im;
1084
27.2M
  }
1085
400k
}