Coverage Report

Created: 2025-08-29 06:10

/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
87.8M
#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
286M
                                         const FIXP_DBL b) {
131
286M
  return (b - fMult(alpha, b) + fMult(alpha, a));
132
286M
}
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.3M
static UINT mapChannel(spatialDec *self, UINT ch) {
142
14.3M
  static const UCHAR chanelIdx[][8] = {
143
14.3M
      {0, 1, 2, 3, 4, 5, 6, 7}, /*  binaural, TREE_212, arbitrary tree */
144
14.3M
  };
145
146
14.3M
  int idx = 0;
147
148
14.3M
  return (chanelIdx[idx][ch]);
149
14.3M
}
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.14M
                                   const int numInputChannels) {
170
2.14M
  SACDEC_ERROR err = MPS_OK;
171
2.14M
  int ch, offset;
172
173
2.14M
  offset = self->pQmfDomain->globalConf.nBandsSynthesis *
174
2.14M
           self->pQmfDomain->globalConf.nQmfTimeSlots;
175
176
2.14M
  {
177
4.29M
    for (ch = 0; ch < numInputChannels; ch++) {
178
2.14M
      const PCM_MPS *inSamples =
179
2.14M
          &inData[ts * self->pQmfDomain->globalConf.nBandsAnalysis];
180
2.14M
      FIXP_DBL *pQmfRealAnalysis = qmfReal[ch]; /* no delay in blind mode */
181
2.14M
      FIXP_DBL *pQmfImagAnalysis = qmfImag[ch];
182
183
2.14M
      CalculateSpaceAnalysisQmf(&self->pQmfDomain->QmfDomainIn[ch].fb,
184
2.14M
                                inSamples + (ch * offset), pQmfRealAnalysis,
185
2.14M
                                pQmfImagAnalysis);
186
187
2.14M
      if (!isTwoChMode(self->upmixType) && !bypassMode) {
188
2.14M
        int i;
189
51.2M
        for (i = 0; i < self->qmfBands; i++) {
190
49.0M
          qmfReal[ch][i] = fMult(
191
49.0M
              scaleValueSaturate(qmfReal[ch][i], self->sacInDataHeadroom - (1)),
192
49.0M
              self->clipProtectGain__FDK);
193
49.0M
          qmfImag[ch][i] = fMult(
194
49.0M
              scaleValueSaturate(qmfImag[ch][i], self->sacInDataHeadroom - (1)),
195
49.0M
              self->clipProtectGain__FDK);
196
49.0M
        }
197
2.14M
      }
198
2.14M
    }
199
2.14M
  }
200
201
2.14M
  self->qmfInputDelayBufPos =
202
2.14M
      (self->qmfInputDelayBufPos + 1) % self->pc_filterdelay;
203
204
2.14M
  return err;
205
2.14M
}
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
5.03M
                               const INT numInputChannels) {
212
5.03M
  SACDEC_ERROR err = MPS_OK;
213
5.03M
  int ch;
214
215
5.03M
  {
216
10.0M
    for (ch = 0; ch < numInputChannels; ch++) {
217
5.03M
      FIXP_DBL *pQmfRealAnalysis =
218
5.03M
          qmfReal__FDK[ch]; /* no delay in blind mode */
219
5.03M
      FIXP_DBL *pQmfImagAnalysis = qmfImag__FDK[ch];
220
221
      /* Write Input data to pQmfRealAnalysis. */
222
5.03M
      if (self->bShareDelayWithSBR) {
223
5.01M
        FDK_QmfDomain_GetSlot(&self->pQmfDomain->QmfDomainIn[ch],
224
5.01M
                              ts + HYBRID_FILTER_DELAY, 0,
225
5.01M
                              MAX_QMF_BANDS_TO_HYBRID, pQmfRealAnalysis,
226
5.01M
                              pQmfImagAnalysis, 15 + (1));
227
5.01M
        FDK_QmfDomain_GetSlot(&self->pQmfDomain->QmfDomainIn[ch], ts,
228
5.01M
                              MAX_QMF_BANDS_TO_HYBRID, self->qmfBands,
229
5.01M
                              pQmfRealAnalysis, pQmfImagAnalysis, 15 + (1));
230
5.01M
      } else {
231
19.4k
        FDK_QmfDomain_GetSlot(&self->pQmfDomain->QmfDomainIn[ch], ts, 0,
232
19.4k
                              self->qmfBands, pQmfRealAnalysis,
233
19.4k
                              pQmfImagAnalysis, 15 + (1));
234
19.4k
      }
235
5.03M
      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
156k
        FDK_QmfDomain_SaveOverlap(&self->pQmfDomain->QmfDomainIn[ch], 0);
239
156k
      }
240
241
      /* Apply clip protection to output. */
242
5.03M
      if (!isTwoChMode(self->upmixType) && !bypassMode) {
243
5.03M
        int i;
244
327M
        for (i = 0; i < self->qmfBands; i++) {
245
322M
          qmfReal__FDK[ch][i] =
246
322M
              fMult(qmfReal__FDK[ch][i], self->clipProtectGain__FDK);
247
322M
          qmfImag__FDK[ch][i] =
248
322M
              fMult(qmfImag__FDK[ch][i], self->clipProtectGain__FDK);
249
322M
        }
250
5.03M
      }
251
252
5.03M
    } /* End of loop over numInputChannels */
253
5.03M
  }
254
255
5.03M
  self->qmfInputDelayBufPos =
256
5.03M
      (self->qmfInputDelayBufPos + 1) % self->pc_filterdelay;
257
258
5.03M
  return err;
259
5.03M
}
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.18M
                                      const INT numInputChannels) {
287
7.18M
  SACDEC_ERROR err = MPS_OK;
288
7.18M
  int ch;
289
290
14.3M
  for (ch = 0; ch < numInputChannels;
291
7.18M
       ch++) /* hybrid filtering for down-mix signals */
292
7.18M
  {
293
7.18M
    if (self->pConfigCurrent->syntaxFlags & SACDEC_SYNTAX_LD) {
294
792k
      int k;
295
      /* No hybrid filtering. Just copy the QMF data. */
296
26.9M
      for (k = 0; k < self->hybridBands; k += 1) {
297
26.1M
        hybOutputReal[ch][k] = qmfInputReal[ch][k];
298
26.1M
        hybOutputImag[ch][k] = qmfInputImag[ch][k];
299
26.1M
      }
300
6.39M
    } else {
301
6.39M
      self->hybridAnalysis[ch].hfMode = self->bShareDelayWithSBR;
302
303
6.39M
      if (self->stereoConfigIndex == 3)
304
6.39M
        FDK_ASSERT(self->hybridAnalysis[ch].hfMode == 0);
305
6.39M
      FDKhybridAnalysisApply(&self->hybridAnalysis[ch], qmfInputReal[ch],
306
6.39M
                             qmfInputImag[ch], hybOutputReal[ch],
307
6.39M
                             hybOutputImag[ch]);
308
6.39M
    }
309
7.18M
  }
310
311
7.18M
  if ((self->pConfigCurrent->syntaxFlags & SACDEC_SYNTAX_USAC) &&
312
7.18M
      self->residualCoding) {
313
1.47M
    self->hybridAnalysis[numInputChannels].hfMode = 0;
314
1.47M
    FDKhybridAnalysisApply(
315
1.47M
        &self->hybridAnalysis[numInputChannels],
316
1.47M
        self->qmfResidualReal__FDK[0][0], self->qmfResidualImag__FDK[0][0],
317
1.47M
        self->hybResidualReal__FDK[0], self->hybResidualImag__FDK[0]);
318
1.47M
  }
319
320
7.18M
  return err;
321
7.18M
}
322
323
SACDEC_ERROR SpatialDecCreateX(spatialDec *self, FIXP_DBL **hybInputReal,
324
                               FIXP_DBL **hybInputImag, FIXP_DBL **pxReal,
325
7.18M
                               FIXP_DBL **pxImag) {
326
7.18M
  SACDEC_ERROR err = MPS_OK;
327
7.18M
  int row;
328
329
  /* Creating wDry */
330
14.3M
  for (row = 0; row < self->numInputChannels; row++) {
331
    /* pointer to direct signals */
332
7.18M
    pxReal[row] = hybInputReal[row];
333
7.18M
    pxImag[row] = hybInputImag[row];
334
7.18M
  }
335
336
7.18M
  return err;
337
7.18M
}
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.8M
                                int nBands) {
344
23.8M
  int pb;
345
346
252M
  for (pb = 0; pb < nBands; pb++) {
347
228M
    FIXP_SGL tmp = FX_DBL2FX_SGL(
348
228M
        interpolateParameter(alpha__FDK, Mparam[pb], MparamPrev[pb]));
349
350
228M
    int i = pWidth[pb];
351
228M
    if (i & 1) *pKernel++ = tmp;
352
228M
    if (i & 2) {
353
100M
      *pKernel++ = tmp;
354
100M
      *pKernel++ = tmp;
355
100M
    }
356
529M
    for (i >>= 2; i--;) {
357
300M
      *pKernel++ = tmp;
358
300M
      *pKernel++ = tmp;
359
300M
      *pKernel++ = tmp;
360
300M
      *pKernel++ = tmp;
361
300M
    }
362
228M
  }
363
23.8M
}
364
365
SACDEC_ERROR SpatialDecApplyM1_CreateW_Mode212(
366
    spatialDec *self, const SPATIAL_BS_FRAME *frame, FIXP_DBL **xReal,
367
7.18M
    FIXP_DBL **xImag, FIXP_DBL **vReal, FIXP_DBL **vImag) {
368
7.18M
  SACDEC_ERROR err = MPS_OK;
369
7.18M
  int res;
370
7.18M
  FIXP_DBL *decorrInReal = vReal[0];
371
7.18M
  FIXP_DBL *decorrInImag = vImag[0];
372
373
  /* M1 does not do anything in 212 mode, so use simplified processing */
374
7.18M
  FDK_ASSERT(self->numVChannels == 2);
375
7.18M
  FDK_ASSERT(self->numDirektSignals == 1);
376
7.18M
  FDK_ASSERT(self->numDecorSignals == 1);
377
7.18M
  FDKmemcpy(vReal[0], xReal[0], self->hybridBands * sizeof(FIXP_DBL));
378
7.18M
  FDKmemcpy(vImag[0], xImag[0], self->hybridBands * sizeof(FIXP_DBL));
379
380
7.18M
  if (isTsdActive(frame->TsdData)) {
381
    /* Generate v_{x,nonTr} as input for allpass based decorrelator */
382
36.6k
    TsdGenerateNonTr(self->hybridBands, frame->TsdData, self->TsdTs, vReal[0],
383
36.6k
                     vImag[0], vReal[1], vImag[1], &decorrInReal,
384
36.6k
                     &decorrInImag);
385
36.6k
  }
386
  /* - Decorrelate */
387
7.18M
  res = SpatialDecGetResidualIndex(self, 1);
388
7.18M
  if (FDKdecorrelateApply(&self->apDecor[0], decorrInReal, decorrInImag,
389
7.18M
                          vReal[1], vImag[1],
390
7.18M
                          self->param2hyb[self->residualBands[res]])) {
391
0
    return MPS_NOTOK;
392
0
  }
393
7.18M
  if (isTsdActive(frame->TsdData)) {
394
    /* Generate v_{x,Tr}, apply transient decorrelator and add to allpass based
395
     * decorrelator output */
396
36.6k
    TsdApply(self->hybridBands, frame->TsdData, &self->TsdTs,
397
36.6k
             vReal[0], /* input: v_x */
398
36.6k
             vImag[0],
399
36.6k
             vReal[1], /* input: d_{x,nonTr}; output: d_{x,nonTr} + d_{x,Tr} */
400
36.6k
             vImag[1]);
401
36.6k
  }
402
403
  /* Write residual signal in approriate parameter bands */
404
7.18M
  if (self->residualBands[res] > 0) {
405
1.35M
    int stopBand = self->param2hyb[self->residualBands[res]];
406
1.35M
    FDKmemcpy(vReal[1], self->hybResidualReal__FDK[res],
407
1.35M
              fixMin(stopBand, self->hybridBands) * sizeof(FIXP_DBL));
408
1.35M
    FDKmemcpy(vImag[1], self->hybResidualImag__FDK[res],
409
1.35M
              fixMin(stopBand, self->hybridBands) * sizeof(FIXP_DBL));
410
1.35M
  } /* (self->residualBands[res]>0) */
411
412
7.18M
  return err;
413
7.18M
}
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.24M
                                       FIXP_DBL **hybOutputImagDry) {
420
1.24M
  SACDEC_ERROR err = MPS_OK;
421
1.24M
  INT row;
422
423
1.24M
  INT *pWidth = self->kernels_width;
424
  /* for stereoConfigIndex == 3 case hybridBands is < 71 */
425
1.24M
  INT pb_max = self->kernels[self->hybridBands - 1] + 1;
426
1.24M
  INT max_row = self->numOutputChannels;
427
428
1.24M
  INT M2_exp = 0;
429
1.24M
  if (self->residualCoding) M2_exp = 3;
430
431
3.72M
  for (row = 0; row < max_row; row++)  // 2 times
432
2.48M
  {
433
2.48M
    FIXP_DBL *Mparam0 = self->M2Real__FDK[row][0];
434
2.48M
    FIXP_DBL *Mparam1 = self->M2Real__FDK[row][1];
435
2.48M
    FIXP_DBL *MparamPrev0 = self->M2RealPrev__FDK[row][0];
436
2.48M
    FIXP_DBL *MparamPrev1 = self->M2RealPrev__FDK[row][1];
437
438
2.48M
    FIXP_DBL *RESTRICT pHybOutRealDry = hybOutputRealDry[row];
439
2.48M
    FIXP_DBL *RESTRICT pHybOutImagDry = hybOutputImagDry[row];
440
441
2.48M
    FIXP_DBL *RESTRICT pWReal0 = wReal[0];
442
2.48M
    FIXP_DBL *RESTRICT pWReal1 = wReal[1];
443
2.48M
    FIXP_DBL *RESTRICT pWImag0 = wImag[0];
444
2.48M
    FIXP_DBL *RESTRICT pWImag1 = wImag[1];
445
23.6M
    for (INT pb = 0; pb < pb_max; pb++) {
446
21.1M
      FIXP_DBL tmp0, tmp1;
447
448
21.1M
      tmp0 = interpolateParameter(alpha, Mparam0[pb], MparamPrev0[pb]);
449
21.1M
      tmp1 = interpolateParameter(alpha, Mparam1[pb], MparamPrev1[pb]);
450
451
21.1M
      INT i = pWidth[pb];
452
453
21.1M
      do  // about 3-4 times
454
72.7M
      {
455
72.7M
        FIXP_DBL var0, var1, real, imag;
456
457
72.7M
        var0 = *pWReal0++;
458
72.7M
        var1 = *pWReal1++;
459
72.7M
        real = fMultDiv2(var0, tmp0);
460
72.7M
        var0 = *pWImag0++;
461
72.7M
        real = fMultAddDiv2(real, var1, tmp1);
462
72.7M
        var1 = *pWImag1++;
463
72.7M
        imag = fMultDiv2(var0, tmp0);
464
72.7M
        *pHybOutRealDry++ = real << (1 + M2_exp);
465
72.7M
        imag = fMultAddDiv2(imag, var1, tmp1);
466
72.7M
        *pHybOutImagDry++ = imag << (1 + M2_exp);
467
72.7M
      } while (--i != 0);
468
21.1M
    }
469
2.48M
  }
470
1.24M
  return err;
471
1.24M
}
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
94.9k
    FIXP_DBL **hybOutputImagDry) {
477
94.9k
  SACDEC_ERROR err = MPS_OK;
478
94.9k
  INT row;
479
94.9k
  INT scale_param_m2;
480
94.9k
  INT *pWidth = self->kernels_width;
481
94.9k
  INT pb_max = self->kernels[self->hybridBands - 1] + 1;
482
483
94.9k
  scale_param_m2 = SCALE_PARAM_M2_212_PRED + SCALE_DATA_APPLY_M2;
484
485
284k
  for (row = 0; row < self->numM2rows; row++) {
486
189k
    INT qs, pb;
487
488
189k
    FIXP_DBL *RESTRICT pWReal0 = wReal[0];
489
189k
    FIXP_DBL *RESTRICT pWImag0 = wImag[0];
490
189k
    FIXP_DBL *RESTRICT pWReal1 = wReal[1];
491
189k
    FIXP_DBL *RESTRICT pWImag1 = wImag[1];
492
493
189k
    FIXP_DBL *MReal0 = self->M2Real__FDK[row][0];
494
189k
    FIXP_DBL *MImag0 = self->M2Imag__FDK[row][0];
495
189k
    FIXP_DBL *MReal1 = self->M2Real__FDK[row][1];
496
189k
    FIXP_DBL *MRealPrev0 = self->M2RealPrev__FDK[row][0];
497
189k
    FIXP_DBL *MImagPrev0 = self->M2ImagPrev__FDK[row][0];
498
189k
    FIXP_DBL *MRealPrev1 = self->M2RealPrev__FDK[row][1];
499
500
189k
    FIXP_DBL *RESTRICT pHybOutRealDry = hybOutputRealDry[row];
501
189k
    FIXP_DBL *RESTRICT pHybOutImagDry = hybOutputImagDry[row];
502
503
189k
    FDK_ASSERT(!(self->pConfigCurrent->syntaxFlags & SACDEC_SYNTAX_LD));
504
189k
    FDK_ASSERT((pWidth[0] + pWidth[1]) >= 3);
505
506
569k
    for (pb = 0, qs = 3; pb < 2; pb++) {
507
379k
      INT s;
508
379k
      FIXP_DBL maxVal;
509
379k
      FIXP_DBL mReal1;
510
379k
      FIXP_DBL mReal0, mImag0;
511
379k
      FIXP_DBL iReal0, iImag0, iReal1;
512
513
379k
      iReal0 = interpolateParameter(alpha, MReal0[pb], MRealPrev0[pb]);
514
379k
      iImag0 = -interpolateParameter(alpha, MImag0[pb], MImagPrev0[pb]);
515
379k
      iReal1 = interpolateParameter(alpha, MReal1[pb], MRealPrev1[pb]);
516
517
379k
      maxVal = fAbs(iReal0) | fAbs(iImag0);
518
379k
      maxVal |= fAbs(iReal1);
519
520
379k
      s = fMin(CntLeadingZeros(maxVal) - 2, scale_param_m2);
521
522
379k
      mReal0 = scaleValue(iReal0, s);
523
379k
      mImag0 = scaleValue(iImag0, s);
524
379k
      mReal1 = scaleValue(iReal1, s);
525
526
379k
      s = scale_param_m2 - s;
527
528
379k
      INT i = pWidth[pb];
529
530
1.18M
      do {
531
1.18M
        FIXP_DBL real, imag, wReal0, wImag0, wReal1, wImag1;
532
533
1.18M
        wReal0 = *pWReal0++;
534
1.18M
        wImag0 = *pWImag0++;
535
1.18M
        wReal1 = *pWReal1++;
536
1.18M
        wImag1 = *pWImag1++;
537
538
1.18M
        cplxMultDiv2(&real, &imag, wReal0, wImag0, mReal0, mImag0);
539
540
1.18M
        *pHybOutRealDry++ = fMultAddDiv2(real, wReal1, mReal1) << s;
541
1.18M
        *pHybOutImagDry++ = fMultAddDiv2(imag, wImag1, mReal1) << s;
542
543
1.18M
        if (qs > 0) {
544
569k
          mImag0 = -mImag0;
545
569k
          qs--;
546
569k
        }
547
1.18M
      } while (--i != 0);
548
379k
    }
549
550
2.32M
    for (; pb < pb_max; pb++) {
551
2.13M
      INT s;
552
2.13M
      FIXP_DBL maxVal;
553
2.13M
      FIXP_SGL mReal1;
554
2.13M
      FIXP_SGL mReal0, mImag0;
555
2.13M
      FIXP_DBL iReal0, iImag0, iReal1;
556
557
2.13M
      iReal0 = interpolateParameter(alpha, MReal0[pb], MRealPrev0[pb]);
558
2.13M
      iImag0 = interpolateParameter(alpha, MImag0[pb], MImagPrev0[pb]);
559
2.13M
      iReal1 = interpolateParameter(alpha, MReal1[pb], MRealPrev1[pb]);
560
561
2.13M
      maxVal = fAbs(iReal0) | fAbs(iImag0);
562
2.13M
      maxVal |= fAbs(iReal1);
563
564
2.13M
      s = fMin(CntLeadingZeros(maxVal) - 2, scale_param_m2);
565
566
2.13M
      mReal0 = FX_DBL2FX_SGL(scaleValue(iReal0, s));
567
2.13M
      mImag0 = FX_DBL2FX_SGL(scaleValue(iImag0, s));
568
2.13M
      mReal1 = FX_DBL2FX_SGL(scaleValue(iReal1, s));
569
570
2.13M
      s = scale_param_m2 - s;
571
572
2.13M
      INT i = pWidth[pb];
573
574
6.53M
      do {
575
6.53M
        FIXP_DBL real, imag, wReal0, wImag0, wReal1, wImag1;
576
577
6.53M
        wReal0 = *pWReal0++;
578
6.53M
        wImag0 = *pWImag0++;
579
6.53M
        wReal1 = *pWReal1++;
580
6.53M
        wImag1 = *pWImag1++;
581
582
6.53M
        cplxMultDiv2(&real, &imag, wReal0, wImag0, mReal0, mImag0);
583
584
6.53M
        *pHybOutRealDry++ = fMultAddDiv2(real, wReal1, mReal1) << s;
585
6.53M
        *pHybOutImagDry++ = fMultAddDiv2(imag, wImag1, mReal1) << s;
586
6.53M
      } while (--i != 0);
587
2.13M
    }
588
189k
  }
589
590
94.9k
  return err;
591
94.9k
}
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.84M
                               FIXP_DBL **hybOutputImagWet) {
599
5.84M
  SACDEC_ERROR err = MPS_OK;
600
601
5.84M
  {
602
5.84M
    int qs, row, col;
603
5.84M
    int complexHybBands;
604
5.84M
    int complexParBands;
605
5.84M
    int scale_param_m2 = 0;
606
5.84M
    int toolsDisabled;
607
608
5.84M
    UCHAR activParamBands;
609
5.84M
    FIXP_DBL *RESTRICT pWReal, *RESTRICT pWImag, *RESTRICT pHybOutRealDry,
610
5.84M
        *RESTRICT pHybOutImagDry, *RESTRICT pHybOutRealWet,
611
5.84M
        *RESTRICT pHybOutImagWet;
612
5.84M
    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.84M
    toolsDisabled =
617
5.84M
        ((self->tempShapeConfig == 1) || (self->tempShapeConfig == 2)) ? 0 : 1;
618
619
5.84M
    {
620
5.84M
      complexHybBands = self->hybridBands;
621
5.84M
      complexParBands = self->numParameterBands;
622
5.84M
    }
623
624
5.84M
    FDKmemclear(hybOutputImagDry[0],
625
5.84M
                self->createParams.maxNumOutputChannels *
626
5.84M
                    self->createParams.maxNumCmplxHybBands * sizeof(FIXP_DBL));
627
5.84M
    FDKmemclear(hybOutputRealDry[0], self->createParams.maxNumOutputChannels *
628
5.84M
                                         self->createParams.maxNumHybridBands *
629
5.84M
                                         sizeof(FIXP_DBL));
630
631
5.84M
    if (!toolsDisabled) {
632
5.58M
      FDKmemclear(hybOutputRealWet[0],
633
5.58M
                  self->createParams.maxNumOutputChannels *
634
5.58M
                      self->createParams.maxNumHybridBands * sizeof(FIXP_DBL));
635
5.58M
      FDKmemclear(hybOutputImagWet[0],
636
5.58M
                  self->createParams.maxNumOutputChannels *
637
5.58M
                      self->createParams.maxNumCmplxHybBands *
638
5.58M
                      sizeof(FIXP_DBL));
639
5.58M
    }
640
641
5.84M
    if (self->phaseCoding == 3) {
642
111k
      scale_param_m2 = -(SCALE_DATA_APPLY_M2_PC - 1);
643
111k
    }
644
645
17.5M
    for (row = 0; row < self->numM2rows; row++) {
646
11.6M
      pHybOutRealDry = hybOutputRealDry[row];
647
11.6M
      pHybOutImagDry = hybOutputImagDry[row];
648
649
11.6M
      if (toolsDisabled) {
650
529k
        pHybOutRealWet = hybOutputRealDry[row];
651
529k
        pHybOutImagWet = hybOutputImagDry[row];
652
11.1M
      } else {
653
11.1M
        pHybOutRealWet = hybOutputRealWet[row];
654
11.1M
        pHybOutImagWet = hybOutputImagWet[row];
655
11.1M
      }
656
657
23.3M
      for (col = 0; col < self->numDirektSignals; col++) {
658
11.6M
        if (self->pActivM2ParamBands ==
659
11.6M
            0) { /* default setting, calculate all rows and columns */
660
11.6M
          activParamBands = 1;
661
11.6M
        } 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.6M
        if (activParamBands) {
671
11.6M
          pWReal = wReal[col];
672
11.6M
          pWImag = wImag[col];
673
674
11.6M
          M2ParamToKernelMult(pKernel, self->M2Real__FDK[row][col],
675
11.6M
                              self->M2RealPrev__FDK[row][col],
676
11.6M
                              self->kernels_width, alpha,
677
11.6M
                              self->numParameterBands);
678
679
11.6M
          if (1 && (self->phaseCoding != 3)) {
680
            /* direct signals */
681
11.4M
            {
682
              /* only one sample will be assigned to each row, hence
683
               * accumulation is not neccessary; that is valid for all
684
               * configurations */
685
751M
              for (qs = 0; qs < complexHybBands; qs++) {
686
740M
                pHybOutRealDry[qs] = fMult(pWReal[qs], pKernel[qs]);
687
740M
                pHybOutImagDry[qs] = fMult(pWImag[qs], pKernel[qs]);
688
740M
              }
689
11.4M
            }
690
11.4M
          } else { /*  isBinauralMode(self->upmixType)  */
691
692
11.2M
            for (qs = 0; qs < complexHybBands; qs++) {
693
10.9M
              pHybOutRealDry[qs] += SAC_DEC_APPLY_M2_SCALE(
694
10.9M
                  fMultDiv2(pWReal[qs], pKernel[qs]), scale_param_m2);
695
10.9M
              pHybOutImagDry[qs] += SAC_DEC_APPLY_M2_SCALE(
696
10.9M
                  fMultDiv2(pWImag[qs], pKernel[qs]), scale_param_m2);
697
10.9M
            }
698
699
223k
            M2ParamToKernelMult(pKernel, self->M2Imag__FDK[row][col],
700
223k
                                self->M2ImagPrev__FDK[row][col],
701
223k
                                self->kernels_width, alpha, complexParBands);
702
703
            /* direct signals sign is -1 for qs = 0,2 */
704
223k
            pHybOutRealDry[0] += SAC_DEC_APPLY_M2_SCALE(
705
223k
                fMultDiv2(pWImag[0], pKernel[0]), scale_param_m2);
706
223k
            pHybOutImagDry[0] -= SAC_DEC_APPLY_M2_SCALE(
707
223k
                fMultDiv2(pWReal[0], pKernel[0]), scale_param_m2);
708
709
223k
            pHybOutRealDry[2] += SAC_DEC_APPLY_M2_SCALE(
710
223k
                fMultDiv2(pWImag[2], pKernel[2]), scale_param_m2);
711
223k
            pHybOutImagDry[2] -= SAC_DEC_APPLY_M2_SCALE(
712
223k
                fMultDiv2(pWReal[2], pKernel[2]), scale_param_m2);
713
714
            /* direct signals sign is +1 for qs = 1,3,4,5,...,complexHybBands */
715
223k
            pHybOutRealDry[1] -= SAC_DEC_APPLY_M2_SCALE(
716
223k
                fMultDiv2(pWImag[1], pKernel[1]), scale_param_m2);
717
223k
            pHybOutImagDry[1] += SAC_DEC_APPLY_M2_SCALE(
718
223k
                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
223k
          } /* self->upmixType */
727
11.6M
        }   /* if (activParamBands) */
728
11.6M
      }     /* self->numDirektSignals */
729
730
23.3M
      for (; col < self->numVChannels; col++) {
731
11.6M
        if (self->pActivM2ParamBands ==
732
11.6M
            0) { /* default setting, calculate all rows and columns */
733
11.6M
          activParamBands = 1;
734
11.6M
        } 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.6M
        if (activParamBands) {
745
11.6M
          int resBandIndex;
746
11.6M
          int resHybIndex;
747
748
11.6M
          resBandIndex =
749
11.6M
              self->residualBands[SpatialDecGetResidualIndex(self, col)];
750
11.6M
          resHybIndex = self->param2hyb[resBandIndex];
751
752
11.6M
          pWReal = wReal[col];
753
11.6M
          pWImag = wImag[col];
754
755
11.6M
          M2ParamToKernelMult(pKernel, self->M2Real__FDK[row][col],
756
11.6M
                              self->M2RealPrev__FDK[row][col],
757
11.6M
                              self->kernels_width, alpha,
758
11.6M
                              self->numParameterBands);
759
760
11.6M
          if (1 && (self->phaseCoding != 3)) {
761
            /* residual signals */
762
14.8M
            for (qs = 0; qs < resHybIndex; qs++) {
763
3.32M
              pHybOutRealDry[qs] += fMult(pWReal[qs], pKernel[qs]);
764
3.32M
              pHybOutImagDry[qs] += fMult(pWImag[qs], pKernel[qs]);
765
3.32M
            }
766
            /* decor signals */
767
748M
            for (; qs < complexHybBands; qs++) {
768
736M
              pHybOutRealWet[qs] += fMult(pWReal[qs], pKernel[qs]);
769
736M
              pHybOutImagWet[qs] += fMult(pWImag[qs], pKernel[qs]);
770
736M
            }
771
11.4M
          } else { /* self->upmixType */
772
            /* residual signals */
773
223k
            FIXP_DBL *RESTRICT pHybOutReal;
774
223k
            FIXP_DBL *RESTRICT pHybOutImag;
775
776
1.33M
            for (qs = 0; qs < resHybIndex; qs++) {
777
1.11M
              pHybOutRealDry[qs] += SAC_DEC_APPLY_M2_SCALE(
778
1.11M
                  fMultDiv2(pWReal[qs], pKernel[qs]), scale_param_m2);
779
1.11M
              pHybOutImagDry[qs] += SAC_DEC_APPLY_M2_SCALE(
780
1.11M
                  fMultDiv2(pWImag[qs], pKernel[qs]), scale_param_m2);
781
1.11M
            }
782
            /* decor signals */
783
10.0M
            for (; qs < complexHybBands; qs++) {
784
9.86M
              pHybOutRealWet[qs] += SAC_DEC_APPLY_M2_SCALE(
785
9.86M
                  fMultDiv2(pWReal[qs], pKernel[qs]), scale_param_m2);
786
9.86M
              pHybOutImagWet[qs] += SAC_DEC_APPLY_M2_SCALE(
787
9.86M
                  fMultDiv2(pWImag[qs], pKernel[qs]), scale_param_m2);
788
9.86M
            }
789
790
223k
            M2ParamToKernelMult(pKernel, self->M2Imag__FDK[row][col],
791
223k
                                self->M2ImagPrev__FDK[row][col],
792
223k
                                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
223k
            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
223k
            } else {
812
223k
              pHybOutReal = &pHybOutRealDry[0];
813
223k
              pHybOutImag = &pHybOutImagDry[0];
814
223k
              if (0 == resHybIndex) {
815
97.7k
                pHybOutReal = &pHybOutRealWet[0];
816
97.7k
                pHybOutImag = &pHybOutImagWet[0];
817
97.7k
              }
818
223k
              pHybOutReal[0] += SAC_DEC_APPLY_M2_SCALE(
819
223k
                  fMultDiv2(pWImag[0], pKernel[0]), scale_param_m2);
820
223k
              pHybOutImag[0] -= SAC_DEC_APPLY_M2_SCALE(
821
223k
                  fMultDiv2(pWReal[0], pKernel[0]), scale_param_m2);
822
823
223k
              if (1 == resHybIndex) {
824
0
                pHybOutReal = &pHybOutRealWet[0];
825
0
                pHybOutImag = &pHybOutImagWet[0];
826
0
              }
827
223k
              pHybOutReal[1] -= SAC_DEC_APPLY_M2_SCALE(
828
223k
                  fMultDiv2(pWImag[1], pKernel[1]), scale_param_m2);
829
223k
              pHybOutImag[1] += SAC_DEC_APPLY_M2_SCALE(
830
223k
                  fMultDiv2(pWReal[1], pKernel[1]), scale_param_m2);
831
832
223k
              if (2 == resHybIndex) {
833
14.0k
                pHybOutReal = &pHybOutRealWet[0];
834
14.0k
                pHybOutImag = &pHybOutImagWet[0];
835
14.0k
              }
836
223k
              pHybOutReal[2] += SAC_DEC_APPLY_M2_SCALE(
837
223k
                  fMultDiv2(pWImag[2], pKernel[2]), scale_param_m2);
838
223k
              pHybOutImag[2] -= SAC_DEC_APPLY_M2_SCALE(
839
223k
                  fMultDiv2(pWReal[2], pKernel[2]), scale_param_m2);
840
223k
            }
841
842
974k
            for (qs = 3; qs < resHybIndex; qs++) {
843
751k
              pHybOutRealDry[qs] -= SAC_DEC_APPLY_M2_SCALE(
844
751k
                  fMultDiv2(pWImag[qs], pKernel[qs]), scale_param_m2);
845
751k
              pHybOutImagDry[qs] += SAC_DEC_APPLY_M2_SCALE(
846
751k
                  fMultDiv2(pWReal[qs], pKernel[qs]), scale_param_m2);
847
751k
            }
848
            /* decor signals */
849
9.77M
            for (; qs < complexHybBands; qs++) {
850
9.55M
              pHybOutRealWet[qs] -= SAC_DEC_APPLY_M2_SCALE(
851
9.55M
                  fMultDiv2(pWImag[qs], pKernel[qs]), scale_param_m2);
852
9.55M
              pHybOutImagWet[qs] += SAC_DEC_APPLY_M2_SCALE(
853
9.55M
                  fMultDiv2(pWReal[qs], pKernel[qs]), scale_param_m2);
854
9.55M
            }
855
223k
          } /* self->upmixType */
856
11.6M
        }   /* if (activParamBands) { */
857
11.6M
      }     /*  self->numVChannels */
858
859
11.6M
      if (self->phaseCoding == 3) {
860
223k
        scaleValuesSaturate(pHybOutRealDry, complexHybBands,
861
223k
                            SCALE_PARAM_M2_212_PRED + SCALE_DATA_APPLY_M2_PC);
862
223k
        scaleValuesSaturate(pHybOutImagDry, complexHybBands,
863
223k
                            SCALE_PARAM_M2_212_PRED + SCALE_DATA_APPLY_M2_PC);
864
865
223k
        if (!toolsDisabled) {
866
223k
          scaleValuesSaturate(pHybOutRealWet, complexHybBands,
867
223k
                              SCALE_PARAM_M2_212_PRED + SCALE_DATA_APPLY_M2_PC);
868
223k
          scaleValuesSaturate(pHybOutImagWet, complexHybBands,
869
223k
                              SCALE_PARAM_M2_212_PRED + SCALE_DATA_APPLY_M2_PC);
870
223k
        }
871
223k
      }
872
11.6M
    }
873
874
5.84M
    C_ALLOC_SCRATCH_END(pKernel, FIXP_SGL, MAX_HYBRID_BANDS);
875
5.84M
  }
876
877
5.84M
  return err;
878
5.84M
}
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.18M
                                 const FDK_channelMapDescr *const mapDescr) {
885
7.18M
  SACDEC_ERROR err = MPS_OK;
886
887
7.18M
  int ch;
888
7.18M
  int stride, offset;
889
890
7.18M
  stride = self->numOutputChannelsAT;
891
7.18M
  offset = 1;
892
893
7.18M
  PCM_MPS *pTimeOut__FDK =
894
7.18M
      &timeOut[stride * self->pQmfDomain->globalConf.nBandsSynthesis * ts];
895
7.18M
  C_ALLOC_SCRATCH_START(pQmfReal, FIXP_DBL, QMF_MAX_SYNTHESIS_BANDS);
896
7.18M
  C_ALLOC_SCRATCH_START(pQmfImag, FIXP_DBL, QMF_MAX_SYNTHESIS_BANDS);
897
898
21.5M
  for (ch = 0; ch < self->numOutputChannelsAT; ch++) {
899
14.3M
    if (self->pConfigCurrent->syntaxFlags & SACDEC_SYNTAX_LD) {
900
1.58M
      int k;
901
      /* No hybrid filtering. Just copy the QMF data. */
902
53.9M
      for (k = 0; k < self->hybridBands; k += 1) {
903
52.3M
        pQmfReal[k] = hybOutputReal[ch][k];
904
52.3M
        pQmfImag[k] = hybOutputImag[ch][k];
905
52.3M
      }
906
12.7M
    } else {
907
12.7M
      FDKhybridSynthesisApply(&self->hybridSynthesis[ch], hybOutputReal[ch],
908
12.7M
                              hybOutputImag[ch], pQmfReal, pQmfImag);
909
12.7M
    }
910
911
    /* Map channel indices from MPEG Surround -> PCE style -> channelMapping[]
912
     */
913
14.3M
    FDK_ASSERT(self->numOutputChannelsAT <= 6);
914
14.3M
    int outCh = FDK_chMapDescr_getMapValue(mapDescr, mapChannel(self, ch),
915
14.3M
                                           self->numOutputChannelsAT);
916
917
14.3M
    {
918
14.3M
      if (self->stereoConfigIndex == 3) {
919
        /* MPS -> SBR */
920
2.75M
        int i;
921
2.75M
        FIXP_DBL *pWorkBufReal, *pWorkBufImag;
922
2.75M
        FDK_ASSERT((self->pQmfDomain->QmfDomainOut[outCh].fb.outGain_m ==
923
2.75M
                    (FIXP_DBL)0x80000000) &&
924
2.75M
                   (self->pQmfDomain->QmfDomainOut[outCh].fb.outGain_e == 0));
925
2.75M
        FDK_QmfDomain_GetWorkBuffer(&self->pQmfDomain->QmfDomainIn[outCh], ts,
926
2.75M
                                    &pWorkBufReal, &pWorkBufImag);
927
2.75M
        FDK_ASSERT(self->qmfBands <=
928
2.75M
                   self->pQmfDomain->QmfDomainIn[outCh].workBuf_nBands);
929
50.5M
        for (i = 0; i < self->qmfBands; i++) {
930
47.7M
          pWorkBufReal[i] = pQmfReal[i];
931
47.7M
          pWorkBufImag[i] = pQmfImag[i];
932
47.7M
        }
933
2.75M
        self->pQmfDomain->QmfDomainIn[outCh].scaling.lb_scale =
934
2.75M
            -7; /*-ALGORITHMIC_SCALING_IN_ANALYSIS_FILTERBANK;*/
935
2.75M
        self->pQmfDomain->QmfDomainIn[outCh].scaling.lb_scale -=
936
2.75M
            self->pQmfDomain->QmfDomainIn[outCh].fb.filterScale;
937
2.75M
        self->pQmfDomain->QmfDomainIn[outCh].scaling.lb_scale -=
938
2.75M
            self->clipProtectGainSF__FDK;
939
940
2.75M
        self->pQmfDomain->QmfDomainIn[outCh].scaling.lb_scale -= (1);
941
11.6M
      } else {
942
        /* Call the QMF synthesis for dry. */
943
11.6M
        err = CalculateSpaceSynthesisQmf(&self->pQmfDomain->QmfDomainOut[outCh],
944
11.6M
                                         pQmfReal, pQmfImag, stride,
945
11.6M
                                         pTimeOut__FDK + (offset * outCh));
946
11.6M
      }
947
14.3M
      if (err != MPS_OK) goto bail;
948
14.3M
    }
949
14.3M
  } /* ch loop */
950
951
7.18M
bail:
952
7.18M
  C_ALLOC_SCRATCH_END(pQmfImag, FIXP_DBL, QMF_MAX_SYNTHESIS_BANDS);
953
7.18M
  C_ALLOC_SCRATCH_END(pQmfReal, FIXP_DBL, QMF_MAX_SYNTHESIS_BANDS);
954
955
7.18M
  return err;
956
7.18M
}
957
958
279k
void SpatialDecBufferMatrices(spatialDec *self) {
959
279k
  int row, col;
960
279k
  int complexParBands;
961
279k
  complexParBands = self->numParameterBands;
962
963
  /*
964
    buffer matrices M2
965
  */
966
838k
  for (row = 0; row < self->numM2rows; row++) {
967
1.67M
    for (col = 0; col < self->numVChannels; col++) {
968
1.11M
      FDKmemcpy(self->M2RealPrev__FDK[row][col], self->M2Real__FDK[row][col],
969
1.11M
                self->numParameterBands * sizeof(FIXP_DBL));
970
1.11M
      if (0 || (self->phaseCoding == 3)) {
971
29.4k
        FDKmemcpy(self->M2ImagPrev__FDK[row][col], self->M2Imag__FDK[row][col],
972
29.4k
                  complexParBands * sizeof(FIXP_DBL));
973
29.4k
      }
974
1.11M
    }
975
559k
  }
976
977
  /* buffer phase */
978
279k
  FDKmemcpy(self->PhasePrevLeft__FDK, self->PhaseLeft__FDK,
979
279k
            self->numParameterBands * sizeof(FIXP_DBL));
980
279k
  FDKmemcpy(self->PhasePrevRight__FDK, self->PhaseRight__FDK,
981
279k
            self->numParameterBands * sizeof(FIXP_DBL));
982
279k
}
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
8.01M
                                  FIXP_SGL alpha, FIXP_DBL pi_x2) {
993
8.01M
  if (angle2 - angle1 > (pi_x2 >> 1)) angle2 -= pi_x2;
994
995
8.01M
  if (angle1 - angle2 > (pi_x2 >> 1)) angle1 -= pi_x2;
996
997
8.01M
  return interpolateParameter(alpha, angle2, angle1);
998
8.01M
}
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.40M
  for (pb = 0; pb < self->numParameterBands; pb++) {
1011
4.00M
    FIXP_DBL pl, pr;
1012
1013
4.00M
    pl = interp_angle__FDK(self->PhasePrevLeft__FDK[pb],
1014
4.00M
                           self->PhaseLeft__FDK[pb], alpha__FDK, pi_x2);
1015
4.00M
    pr = interp_angle__FDK(self->PhasePrevRight__FDK[pb],
1016
4.00M
                           self->PhaseRight__FDK[pb], alpha__FDK, pi_x2);
1017
1018
4.00M
    inline_fixp_cos_sin(pl, pr, IPD_SCALE, &ppb[4 * pb]);
1019
4.00M
  }
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
801k
    } else {
1050
801k
      cplxMultDiv2(&out_re, &out_im, *Dry_real0, *Dry_imag0, ppb[4 * pb + 0],
1051
801k
                   -ppb[4 * pb + 1]);
1052
801k
      out_re <<= PHASE_SCALE - 1;
1053
801k
      out_im <<= PHASE_SCALE - 1;
1054
801k
      *Dry_real0++ = out_re;
1055
801k
      *Dry_imag0++ = out_im;
1056
1057
801k
      cplxMultDiv2(&out_re, &out_im, *Dry_real1, *Dry_imag1, ppb[4 * pb + 2],
1058
801k
                   -ppb[4 * pb + 3]);
1059
801k
      out_re <<= PHASE_SCALE - 1;
1060
801k
      out_im <<= PHASE_SCALE - 1;
1061
801k
      *Dry_real1++ = out_re;
1062
801k
      *Dry_imag1++ = out_im;
1063
801k
    }
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
}