• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* -----------------------------------------------------------------------------
2 Software License for The Fraunhofer FDK AAC Codec Library for Android
3 
4 © Copyright  1995 - 2018 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 /**************************** SBR decoder library ******************************
96 
97    Author(s):
98 
99    Description:
100 
101 *******************************************************************************/
102 
103 /*!
104   \file
105   \brief  Low Power Profile Transposer
106   This module provides the transposer. The main entry point is lppTransposer().
107   The function generates high frequency content by copying data from the low
108   band (provided by core codec) into the high band. This process is also
109   referred to as "patching". The function also implements spectral whitening by
110   means of inverse filtering based on LPC coefficients.
111 
112   Together with the QMF filterbank the transposer can be tested using a supplied
113   test program. See main_audio.cpp for details. This module does use fractional
114   arithmetic and the accuracy of the computations has an impact on the overall
115   sound quality. The module also needs to take into account the different
116   scaling of spectral data.
117 
118   \sa lppTransposer(), main_audio.cpp, sbr_scale.h, \ref documentationOverview
119 */
120 
121 #ifdef __ANDROID__
122 #include "log/log.h"
123 #endif
124 
125 #include "lpp_tran.h"
126 
127 #include "sbr_ram.h"
128 #include "sbr_rom.h"
129 
130 #include "genericStds.h"
131 #include "autocorr2nd.h"
132 
133 #include "HFgen_preFlat.h"
134 
135 #if defined(__arm__)
136 #include "arm/lpp_tran_arm.cpp"
137 #endif
138 
139 #define LPC_SCALE_FACTOR 2
140 
141 /*!
142  *
143  * \brief Get bandwidth expansion factor from filtering level
144  *
145  * Returns a filter parameter (bandwidth expansion factor) depending on
146  * the desired filtering level signalled in the bitstream.
147  * When switching the filtering level from LOW to OFF, an additional
148  * level is being inserted to achieve a smooth transition.
149  */
150 
mapInvfMode(INVF_MODE mode,INVF_MODE prevMode,WHITENING_FACTORS whFactors)151 static FIXP_DBL mapInvfMode(INVF_MODE mode, INVF_MODE prevMode,
152                             WHITENING_FACTORS whFactors) {
153   switch (mode) {
154     case INVF_LOW_LEVEL:
155       if (prevMode == INVF_OFF)
156         return whFactors.transitionLevel;
157       else
158         return whFactors.lowLevel;
159 
160     case INVF_MID_LEVEL:
161       return whFactors.midLevel;
162 
163     case INVF_HIGH_LEVEL:
164       return whFactors.highLevel;
165 
166     default:
167       if (prevMode == INVF_LOW_LEVEL)
168         return whFactors.transitionLevel;
169       else
170         return whFactors.off;
171   }
172 }
173 
174 /*!
175  *
176  * \brief Perform inverse filtering level emphasis
177  *
178  * Retrieve bandwidth expansion factor and apply smoothing for each filter band
179  *
180  */
181 
inverseFilteringLevelEmphasis(HANDLE_SBR_LPP_TRANS hLppTrans,UCHAR nInvfBands,INVF_MODE * sbr_invf_mode,INVF_MODE * sbr_invf_mode_prev,FIXP_DBL * bwVector)182 static void inverseFilteringLevelEmphasis(
183     HANDLE_SBR_LPP_TRANS hLppTrans, /*!< Handle of lpp transposer  */
184     UCHAR nInvfBands,              /*!< Number of bands for inverse filtering */
185     INVF_MODE *sbr_invf_mode,      /*!< Current inverse filtering modes */
186     INVF_MODE *sbr_invf_mode_prev, /*!< Previous inverse filtering modes */
187     FIXP_DBL *bwVector             /*!< Resulting filtering levels */
188 ) {
189   for (int i = 0; i < nInvfBands; i++) {
190     FIXP_DBL accu;
191     FIXP_DBL bwTmp = mapInvfMode(sbr_invf_mode[i], sbr_invf_mode_prev[i],
192                                  hLppTrans->pSettings->whFactors);
193 
194     if (bwTmp < hLppTrans->bwVectorOld[i]) {
195       accu = fMultDiv2(FL2FXCONST_DBL(0.75f), bwTmp) +
196              fMultDiv2(FL2FXCONST_DBL(0.25f), hLppTrans->bwVectorOld[i]);
197     } else {
198       accu = fMultDiv2(FL2FXCONST_DBL(0.90625f), bwTmp) +
199              fMultDiv2(FL2FXCONST_DBL(0.09375f), hLppTrans->bwVectorOld[i]);
200     }
201 
202     if (accu<FL2FXCONST_DBL(0.015625f)>> 1) {
203       bwVector[i] = FL2FXCONST_DBL(0.0f);
204     } else {
205       bwVector[i] = fixMin(accu << 1, FL2FXCONST_DBL(0.99609375f));
206     }
207   }
208 }
209 
210 /* Resulting autocorrelation determinant exponent */
211 #define ACDET_EXP \
212   (2 * (DFRACT_BITS + sbrScaleFactor->lb_scale + 10 - ac.det_scale))
213 #define AC_EXP (-sbrScaleFactor->lb_scale + LPC_SCALE_FACTOR)
214 #define ALPHA_EXP (-sbrScaleFactor->lb_scale + LPC_SCALE_FACTOR + 1)
215 /* Resulting transposed QMF values exponent 16 bit normalized samplebits
216  * assumed. */
217 #define QMFOUT_EXP ((SAMPLE_BITS - 15) - sbrScaleFactor->lb_scale)
218 
calc_qmfBufferReal(FIXP_DBL ** qmfBufferReal,const FIXP_DBL * const lowBandReal,const int startSample,const int stopSample,const UCHAR hiBand,const int dynamicScale,const int descale,const FIXP_SGL a0r,const FIXP_SGL a1r)219 static inline void calc_qmfBufferReal(FIXP_DBL **qmfBufferReal,
220                                       const FIXP_DBL *const lowBandReal,
221                                       const int startSample,
222                                       const int stopSample, const UCHAR hiBand,
223                                       const int dynamicScale, const int descale,
224                                       const FIXP_SGL a0r, const FIXP_SGL a1r) {
225   FIXP_DBL accu1, accu2;
226   int i;
227 
228   for (i = 0; i < stopSample - startSample; i++) {
229     accu1 = fMultDiv2(a1r, lowBandReal[i]);
230     accu1 = (fMultDiv2(a0r, lowBandReal[i + 1]) + accu1);
231     accu1 = accu1 >> dynamicScale;
232 
233     accu1 <<= 1;
234     accu2 = (lowBandReal[i + 2] >> descale);
235     qmfBufferReal[i + startSample][hiBand] = accu1 + accu2;
236   }
237 }
238 
239 /*!
240  *
241  * \brief Perform transposition by patching of subband samples.
242  * This function serves as the main entry point into the module. The function
243  * determines the areas for the patching process (these are the source range as
244  * well as the target range) and implements spectral whitening by means of
245  * inverse filtering. The function autoCorrelation2nd() is an auxiliary function
246  * for calculating the LPC coefficients for the filtering.  The actual
247  * calculation of the LPC coefficients and the implementation of the filtering
248  * are done as part of lppTransposer().
249  *
250  * Note that the filtering is done on all available QMF subsamples, whereas the
251  * patching is only done on those QMF subsamples that will be used in the next
252  * QMF synthesis. The filtering is also implemented before the patching includes
253  * further dependencies on parameters from the SBR data.
254  *
255  */
256 
lppTransposer(HANDLE_SBR_LPP_TRANS hLppTrans,QMF_SCALE_FACTOR * sbrScaleFactor,FIXP_DBL ** qmfBufferReal,FIXP_DBL * degreeAlias,FIXP_DBL ** qmfBufferImag,const int useLP,const int fPreWhitening,const int v_k_master0,const int timeStep,const int firstSlotOffs,const int lastSlotOffs,const int nInvfBands,INVF_MODE * sbr_invf_mode,INVF_MODE * sbr_invf_mode_prev)257 void lppTransposer(
258     HANDLE_SBR_LPP_TRANS hLppTrans,   /*!< Handle of lpp transposer  */
259     QMF_SCALE_FACTOR *sbrScaleFactor, /*!< Scaling factors */
260     FIXP_DBL **qmfBufferReal, /*!< Pointer to pointer to real part of subband
261                                  samples (source) */
262 
263     FIXP_DBL *degreeAlias,    /*!< Vector for results of aliasing estimation */
264     FIXP_DBL **qmfBufferImag, /*!< Pointer to pointer to imaginary part of
265                                  subband samples (source) */
266     const int useLP, const int fPreWhitening, const int v_k_master0,
267     const int timeStep,       /*!< Time step of envelope */
268     const int firstSlotOffs,  /*!< Start position in time */
269     const int lastSlotOffs,   /*!< Number of overlap-slots into next frame */
270     const int nInvfBands,     /*!< Number of bands for inverse filtering */
271     INVF_MODE *sbr_invf_mode, /*!< Current inverse filtering modes */
272     INVF_MODE *sbr_invf_mode_prev /*!< Previous inverse filtering modes */
273 ) {
274   INT bwIndex[MAX_NUM_PATCHES];
275   FIXP_DBL bwVector[MAX_NUM_PATCHES]; /*!< pole moving factors */
276   FIXP_DBL preWhiteningGains[(64) / 2];
277   int preWhiteningGains_exp[(64) / 2];
278 
279   int i;
280   int loBand, start, stop;
281   TRANSPOSER_SETTINGS *pSettings = hLppTrans->pSettings;
282   PATCH_PARAM *patchParam = pSettings->patchParam;
283   int patch;
284 
285   FIXP_SGL alphar[LPC_ORDER], a0r, a1r;
286   FIXP_SGL alphai[LPC_ORDER], a0i = 0, a1i = 0;
287   FIXP_SGL bw = FL2FXCONST_SGL(0.0f);
288 
289   int autoCorrLength;
290 
291   FIXP_DBL k1, k1_below = 0, k1_below2 = 0;
292 
293   ACORR_COEFS ac;
294   int startSample;
295   int stopSample;
296   int stopSampleClear;
297 
298   int comLowBandScale;
299   int ovLowBandShift;
300   int lowBandShift;
301   /*  int ovHighBandShift;*/
302 
303   alphai[0] = FL2FXCONST_SGL(0.0f);
304   alphai[1] = FL2FXCONST_SGL(0.0f);
305 
306   startSample = firstSlotOffs * timeStep;
307   stopSample = pSettings->nCols + lastSlotOffs * timeStep;
308   FDK_ASSERT((lastSlotOffs * timeStep) <= pSettings->overlap);
309 
310   inverseFilteringLevelEmphasis(hLppTrans, nInvfBands, sbr_invf_mode,
311                                 sbr_invf_mode_prev, bwVector);
312 
313   stopSampleClear = stopSample;
314 
315   autoCorrLength = pSettings->nCols + pSettings->overlap;
316 
317   if (pSettings->noOfPatches > 0) {
318     /* Set upper subbands to zero:
319        This is required in case that the patches do not cover the complete
320        highband (because the last patch would be too short). Possible
321        optimization: Clearing bands up to usb would be sufficient here. */
322     int targetStopBand =
323         patchParam[pSettings->noOfPatches - 1].targetStartBand +
324         patchParam[pSettings->noOfPatches - 1].numBandsInPatch;
325 
326     int memSize = ((64) - targetStopBand) * sizeof(FIXP_DBL);
327 
328     if (!useLP) {
329       for (i = startSample; i < stopSampleClear; i++) {
330         FDKmemclear(&qmfBufferReal[i][targetStopBand], memSize);
331         FDKmemclear(&qmfBufferImag[i][targetStopBand], memSize);
332       }
333     } else {
334       for (i = startSample; i < stopSampleClear; i++) {
335         FDKmemclear(&qmfBufferReal[i][targetStopBand], memSize);
336       }
337     }
338   }
339 #ifdef __ANDROID__
340   else {
341     // Safetynet logging
342     android_errorWriteLog(0x534e4554, "112160868");
343   }
344 #endif
345 
346   /* init bwIndex for each patch */
347   FDKmemclear(bwIndex, sizeof(bwIndex));
348 
349   /*
350     Calc common low band scale factor
351   */
352   comLowBandScale =
353       fixMin(sbrScaleFactor->ov_lb_scale, sbrScaleFactor->lb_scale);
354 
355   ovLowBandShift = sbrScaleFactor->ov_lb_scale - comLowBandScale;
356   lowBandShift = sbrScaleFactor->lb_scale - comLowBandScale;
357   /*  ovHighBandShift = firstSlotOffs == 0 ? ovLowBandShift:0;*/
358 
359   if (fPreWhitening) {
360     sbrDecoder_calculateGainVec(
361         qmfBufferReal, qmfBufferImag,
362         DFRACT_BITS - 1 - 16 -
363             sbrScaleFactor->ov_lb_scale, /* convert scale to exponent */
364         DFRACT_BITS - 1 - 16 -
365             sbrScaleFactor->lb_scale, /* convert scale to exponent */
366         pSettings->overlap, preWhiteningGains, preWhiteningGains_exp,
367         v_k_master0, startSample, stopSample);
368   }
369 
370   /* outer loop over bands to do analysis only once for each band */
371 
372   if (!useLP) {
373     start = pSettings->lbStartPatching;
374     stop = pSettings->lbStopPatching;
375   } else {
376     start = fixMax(1, pSettings->lbStartPatching - 2);
377     stop = patchParam[0].targetStartBand;
378   }
379 
380   for (loBand = start; loBand < stop; loBand++) {
381     FIXP_DBL lowBandReal[(((1024) / (32) * (4) / 2) + (3 * (4))) + LPC_ORDER];
382     FIXP_DBL *plowBandReal = lowBandReal;
383     FIXP_DBL **pqmfBufferReal =
384         qmfBufferReal + firstSlotOffs * timeStep /* + pSettings->overlap */;
385     FIXP_DBL lowBandImag[(((1024) / (32) * (4) / 2) + (3 * (4))) + LPC_ORDER];
386     FIXP_DBL *plowBandImag = lowBandImag;
387     FIXP_DBL **pqmfBufferImag =
388         qmfBufferImag + firstSlotOffs * timeStep /* + pSettings->overlap */;
389     int resetLPCCoeffs = 0;
390     int dynamicScale = DFRACT_BITS - 1 - LPC_SCALE_FACTOR;
391     int acDetScale = 0; /* scaling of autocorrelation determinant */
392 
393     for (i = 0;
394          i < LPC_ORDER + firstSlotOffs * timeStep /*+pSettings->overlap*/;
395          i++) {
396       *plowBandReal++ = hLppTrans->lpcFilterStatesRealLegSBR[i][loBand];
397       if (!useLP)
398         *plowBandImag++ = hLppTrans->lpcFilterStatesImagLegSBR[i][loBand];
399     }
400 
401     /*
402       Take old slope length qmf slot source values out of (overlap)qmf buffer
403     */
404     if (!useLP) {
405       for (i = 0;
406            i < pSettings->nCols + pSettings->overlap - firstSlotOffs * timeStep;
407            i++) {
408         *plowBandReal++ = (*pqmfBufferReal++)[loBand];
409         *plowBandImag++ = (*pqmfBufferImag++)[loBand];
410       }
411     } else {
412       /* pSettings->overlap is always even */
413       FDK_ASSERT((pSettings->overlap & 1) == 0);
414       for (i = 0; i < ((pSettings->nCols + pSettings->overlap -
415                         firstSlotOffs * timeStep) >>
416                        1);
417            i++) {
418         *plowBandReal++ = (*pqmfBufferReal++)[loBand];
419         *plowBandReal++ = (*pqmfBufferReal++)[loBand];
420       }
421       if (pSettings->nCols & 1) {
422         *plowBandReal++ = (*pqmfBufferReal++)[loBand];
423       }
424     }
425 
426     /*
427       Determine dynamic scaling value.
428      */
429     dynamicScale =
430         fixMin(dynamicScale,
431                getScalefactor(lowBandReal, LPC_ORDER + pSettings->overlap) +
432                    ovLowBandShift);
433     dynamicScale =
434         fixMin(dynamicScale,
435                getScalefactor(&lowBandReal[LPC_ORDER + pSettings->overlap],
436                               pSettings->nCols) +
437                    lowBandShift);
438     if (!useLP) {
439       dynamicScale =
440           fixMin(dynamicScale,
441                  getScalefactor(lowBandImag, LPC_ORDER + pSettings->overlap) +
442                      ovLowBandShift);
443       dynamicScale =
444           fixMin(dynamicScale,
445                  getScalefactor(&lowBandImag[LPC_ORDER + pSettings->overlap],
446                                 pSettings->nCols) +
447                      lowBandShift);
448     }
449     dynamicScale = fixMax(
450         0, dynamicScale - 1); /* one additional bit headroom to prevent -1.0 */
451 
452     /*
453       Scale temporal QMF buffer.
454      */
455     scaleValues(&lowBandReal[0], LPC_ORDER + pSettings->overlap,
456                 dynamicScale - ovLowBandShift);
457     scaleValues(&lowBandReal[LPC_ORDER + pSettings->overlap], pSettings->nCols,
458                 dynamicScale - lowBandShift);
459 
460     if (!useLP) {
461       scaleValues(&lowBandImag[0], LPC_ORDER + pSettings->overlap,
462                   dynamicScale - ovLowBandShift);
463       scaleValues(&lowBandImag[LPC_ORDER + pSettings->overlap],
464                   pSettings->nCols, dynamicScale - lowBandShift);
465     }
466 
467     if (!useLP) {
468       acDetScale += autoCorr2nd_cplx(&ac, lowBandReal + LPC_ORDER,
469                                      lowBandImag + LPC_ORDER, autoCorrLength);
470     } else {
471       acDetScale +=
472           autoCorr2nd_real(&ac, lowBandReal + LPC_ORDER, autoCorrLength);
473     }
474 
475     /* Examine dynamic of determinant in autocorrelation. */
476     acDetScale += 2 * (comLowBandScale + dynamicScale);
477     acDetScale *= 2;            /* two times reflection coefficent scaling */
478     acDetScale += ac.det_scale; /* ac scaling of determinant */
479 
480     /* In case of determinant < 10^-38, resetLPCCoeffs=1 has to be enforced. */
481     if (acDetScale > 126) {
482       resetLPCCoeffs = 1;
483     }
484 
485     alphar[1] = FL2FXCONST_SGL(0.0f);
486     if (!useLP) alphai[1] = FL2FXCONST_SGL(0.0f);
487 
488     if (ac.det != FL2FXCONST_DBL(0.0f)) {
489       FIXP_DBL tmp, absTmp, absDet;
490 
491       absDet = fixp_abs(ac.det);
492 
493       if (!useLP) {
494         tmp = (fMultDiv2(ac.r01r, ac.r12r) >> (LPC_SCALE_FACTOR - 1)) -
495               ((fMultDiv2(ac.r01i, ac.r12i) + fMultDiv2(ac.r02r, ac.r11r)) >>
496                (LPC_SCALE_FACTOR - 1));
497       } else {
498         tmp = (fMultDiv2(ac.r01r, ac.r12r) >> (LPC_SCALE_FACTOR - 1)) -
499               (fMultDiv2(ac.r02r, ac.r11r) >> (LPC_SCALE_FACTOR - 1));
500       }
501       absTmp = fixp_abs(tmp);
502 
503       /*
504         Quick check: is first filter coeff >= 1(4)
505        */
506       {
507         INT scale;
508         FIXP_DBL result = fDivNorm(absTmp, absDet, &scale);
509         scale = scale + ac.det_scale;
510 
511         if ((scale > 0) && (result >= (FIXP_DBL)MAXVAL_DBL >> scale)) {
512           resetLPCCoeffs = 1;
513         } else {
514           alphar[1] = FX_DBL2FX_SGL(scaleValue(result, scale));
515           if ((tmp < FL2FX_DBL(0.0f)) ^ (ac.det < FL2FX_DBL(0.0f))) {
516             alphar[1] = -alphar[1];
517           }
518         }
519       }
520 
521       if (!useLP) {
522         tmp = (fMultDiv2(ac.r01i, ac.r12r) >> (LPC_SCALE_FACTOR - 1)) +
523               ((fMultDiv2(ac.r01r, ac.r12i) -
524                 (FIXP_DBL)fMultDiv2(ac.r02i, ac.r11r)) >>
525                (LPC_SCALE_FACTOR - 1));
526 
527         absTmp = fixp_abs(tmp);
528 
529         /*
530         Quick check: is second filter coeff >= 1(4)
531         */
532         {
533           INT scale;
534           FIXP_DBL result = fDivNorm(absTmp, absDet, &scale);
535           scale = scale + ac.det_scale;
536 
537           if ((scale > 0) &&
538               (result >= /*FL2FXCONST_DBL(1.f)*/ (FIXP_DBL)MAXVAL_DBL >>
539                scale)) {
540             resetLPCCoeffs = 1;
541           } else {
542             alphai[1] = FX_DBL2FX_SGL(scaleValue(result, scale));
543             if ((tmp < FL2FX_DBL(0.0f)) ^ (ac.det < FL2FX_DBL(0.0f))) {
544               alphai[1] = -alphai[1];
545             }
546           }
547         }
548       }
549     }
550 
551     alphar[0] = FL2FXCONST_SGL(0.0f);
552     if (!useLP) alphai[0] = FL2FXCONST_SGL(0.0f);
553 
554     if (ac.r11r != FL2FXCONST_DBL(0.0f)) {
555       /* ac.r11r is always >=0 */
556       FIXP_DBL tmp, absTmp;
557 
558       if (!useLP) {
559         tmp = (ac.r01r >> (LPC_SCALE_FACTOR + 1)) +
560               (fMultDiv2(alphar[1], ac.r12r) + fMultDiv2(alphai[1], ac.r12i));
561       } else {
562         if (ac.r01r >= FL2FXCONST_DBL(0.0f))
563           tmp = (ac.r01r >> (LPC_SCALE_FACTOR + 1)) +
564                 fMultDiv2(alphar[1], ac.r12r);
565         else
566           tmp = -((-ac.r01r) >> (LPC_SCALE_FACTOR + 1)) +
567                 fMultDiv2(alphar[1], ac.r12r);
568       }
569 
570       absTmp = fixp_abs(tmp);
571 
572       /*
573         Quick check: is first filter coeff >= 1(4)
574       */
575 
576       if (absTmp >= (ac.r11r >> 1)) {
577         resetLPCCoeffs = 1;
578       } else {
579         INT scale;
580         FIXP_DBL result = fDivNorm(absTmp, fixp_abs(ac.r11r), &scale);
581         alphar[0] = FX_DBL2FX_SGL(scaleValue(result, scale + 1));
582 
583         if ((tmp > FL2FX_DBL(0.0f)) ^ (ac.r11r < FL2FX_DBL(0.0f)))
584           alphar[0] = -alphar[0];
585       }
586 
587       if (!useLP) {
588         tmp = (ac.r01i >> (LPC_SCALE_FACTOR + 1)) +
589               (fMultDiv2(alphai[1], ac.r12r) - fMultDiv2(alphar[1], ac.r12i));
590 
591         absTmp = fixp_abs(tmp);
592 
593         /*
594         Quick check: is second filter coeff >= 1(4)
595         */
596         if (absTmp >= (ac.r11r >> 1)) {
597           resetLPCCoeffs = 1;
598         } else {
599           INT scale;
600           FIXP_DBL result = fDivNorm(absTmp, fixp_abs(ac.r11r), &scale);
601           alphai[0] = FX_DBL2FX_SGL(scaleValue(result, scale + 1));
602           if ((tmp > FL2FX_DBL(0.0f)) ^ (ac.r11r < FL2FX_DBL(0.0f)))
603             alphai[0] = -alphai[0];
604         }
605       }
606     }
607 
608     if (!useLP) {
609       /* Now check the quadratic criteria */
610       if ((fMultDiv2(alphar[0], alphar[0]) + fMultDiv2(alphai[0], alphai[0])) >=
611           FL2FXCONST_DBL(0.5f))
612         resetLPCCoeffs = 1;
613       if ((fMultDiv2(alphar[1], alphar[1]) + fMultDiv2(alphai[1], alphai[1])) >=
614           FL2FXCONST_DBL(0.5f))
615         resetLPCCoeffs = 1;
616     }
617 
618     if (resetLPCCoeffs) {
619       alphar[0] = FL2FXCONST_SGL(0.0f);
620       alphar[1] = FL2FXCONST_SGL(0.0f);
621       if (!useLP) {
622         alphai[0] = FL2FXCONST_SGL(0.0f);
623         alphai[1] = FL2FXCONST_SGL(0.0f);
624       }
625     }
626 
627     if (useLP) {
628       /* Aliasing detection */
629       if (ac.r11r == FL2FXCONST_DBL(0.0f)) {
630         k1 = FL2FXCONST_DBL(0.0f);
631       } else {
632         if (fixp_abs(ac.r01r) >= fixp_abs(ac.r11r)) {
633           if (fMultDiv2(ac.r01r, ac.r11r) < FL2FX_DBL(0.0f)) {
634             k1 = (FIXP_DBL)MAXVAL_DBL /*FL2FXCONST_SGL(1.0f)*/;
635           } else {
636             /* Since this value is squared later, it must not ever become -1.0f.
637              */
638             k1 = (FIXP_DBL)(MINVAL_DBL + 1) /*FL2FXCONST_SGL(-1.0f)*/;
639           }
640         } else {
641           INT scale;
642           FIXP_DBL result =
643               fDivNorm(fixp_abs(ac.r01r), fixp_abs(ac.r11r), &scale);
644           k1 = scaleValue(result, scale);
645 
646           if (!((ac.r01r < FL2FX_DBL(0.0f)) ^ (ac.r11r < FL2FX_DBL(0.0f)))) {
647             k1 = -k1;
648           }
649         }
650       }
651       if ((loBand > 1) && (loBand < v_k_master0)) {
652         /* Check if the gain should be locked */
653         FIXP_DBL deg =
654             /*FL2FXCONST_DBL(1.0f)*/ (FIXP_DBL)MAXVAL_DBL - fPow2(k1_below);
655         degreeAlias[loBand] = FL2FXCONST_DBL(0.0f);
656         if (((loBand & 1) == 0) && (k1 < FL2FXCONST_DBL(0.0f))) {
657           if (k1_below < FL2FXCONST_DBL(0.0f)) { /* 2-Ch Aliasing Detection */
658             degreeAlias[loBand] = (FIXP_DBL)MAXVAL_DBL /*FL2FXCONST_DBL(1.0f)*/;
659             if (k1_below2 >
660                 FL2FXCONST_DBL(0.0f)) { /* 3-Ch Aliasing Detection */
661               degreeAlias[loBand - 1] = deg;
662             }
663           } else if (k1_below2 >
664                      FL2FXCONST_DBL(0.0f)) { /* 3-Ch Aliasing Detection */
665             degreeAlias[loBand] = deg;
666           }
667         }
668         if (((loBand & 1) == 1) && (k1 > FL2FXCONST_DBL(0.0f))) {
669           if (k1_below > FL2FXCONST_DBL(0.0f)) { /* 2-CH Aliasing Detection */
670             degreeAlias[loBand] = (FIXP_DBL)MAXVAL_DBL /*FL2FXCONST_DBL(1.0f)*/;
671             if (k1_below2 <
672                 FL2FXCONST_DBL(0.0f)) { /* 3-CH Aliasing Detection */
673               degreeAlias[loBand - 1] = deg;
674             }
675           } else if (k1_below2 <
676                      FL2FXCONST_DBL(0.0f)) { /* 3-CH Aliasing Detection */
677             degreeAlias[loBand] = deg;
678           }
679         }
680       }
681       /* remember k1 values of the 2 QMF channels below the current channel */
682       k1_below2 = k1_below;
683       k1_below = k1;
684     }
685 
686     patch = 0;
687 
688     while (patch < pSettings->noOfPatches) { /* inner loop over every patch */
689 
690       int hiBand = loBand + patchParam[patch].targetBandOffs;
691 
692       if (loBand < patchParam[patch].sourceStartBand ||
693           loBand >= patchParam[patch].sourceStopBand
694           //|| hiBand >= hLppTrans->pSettings->noChannels
695       ) {
696         /* Lowband not in current patch - proceed */
697         patch++;
698         continue;
699       }
700 
701       FDK_ASSERT(hiBand < (64));
702 
703       /* bwIndex[patch] is already initialized with value from previous band
704        * inside this patch */
705       while (hiBand >= pSettings->bwBorders[bwIndex[patch]] &&
706              bwIndex[patch] < MAX_NUM_PATCHES - 1) {
707         bwIndex[patch]++;
708       }
709 
710       /*
711         Filter Step 2: add the left slope with the current filter to the buffer
712                        pure source values are already in there
713       */
714       bw = FX_DBL2FX_SGL(bwVector[bwIndex[patch]]);
715 
716       a0r = FX_DBL2FX_SGL(
717           fMult(bw, alphar[0])); /* Apply current bandwidth expansion factor */
718 
719       if (!useLP) a0i = FX_DBL2FX_SGL(fMult(bw, alphai[0]));
720       bw = FX_DBL2FX_SGL(fPow2(bw));
721       a1r = FX_DBL2FX_SGL(fMult(bw, alphar[1]));
722       if (!useLP) a1i = FX_DBL2FX_SGL(fMult(bw, alphai[1]));
723 
724       /*
725         Filter Step 3: insert the middle part which won't be windowed
726       */
727       if (bw <= FL2FXCONST_SGL(0.0f)) {
728         if (!useLP) {
729           int descale =
730               fixMin(DFRACT_BITS - 1, (LPC_SCALE_FACTOR + dynamicScale));
731           for (i = startSample; i < stopSample; i++) {
732             FIXP_DBL accu1, accu2;
733             accu1 = lowBandReal[LPC_ORDER + i] >> descale;
734             accu2 = lowBandImag[LPC_ORDER + i] >> descale;
735             if (fPreWhitening) {
736               accu1 = scaleValueSaturate(
737                   fMultDiv2(accu1, preWhiteningGains[loBand]),
738                   preWhiteningGains_exp[loBand] + 1);
739               accu2 = scaleValueSaturate(
740                   fMultDiv2(accu2, preWhiteningGains[loBand]),
741                   preWhiteningGains_exp[loBand] + 1);
742             }
743             qmfBufferReal[i][hiBand] = accu1;
744             qmfBufferImag[i][hiBand] = accu2;
745           }
746         } else {
747           int descale =
748               fixMin(DFRACT_BITS - 1, (LPC_SCALE_FACTOR + dynamicScale));
749           for (i = startSample; i < stopSample; i++) {
750             qmfBufferReal[i][hiBand] = lowBandReal[LPC_ORDER + i] >> descale;
751           }
752         }
753       } else { /* bw <= 0 */
754 
755         if (!useLP) {
756           int descale =
757               fixMin(DFRACT_BITS - 1, (LPC_SCALE_FACTOR + dynamicScale));
758 #ifdef FUNCTION_LPPTRANSPOSER_func1
759           lppTransposer_func1(
760               lowBandReal + LPC_ORDER + startSample,
761               lowBandImag + LPC_ORDER + startSample,
762               qmfBufferReal + startSample, qmfBufferImag + startSample,
763               stopSample - startSample, (int)hiBand, dynamicScale, descale, a0r,
764               a0i, a1r, a1i, fPreWhitening, preWhiteningGains[loBand],
765               preWhiteningGains_exp[loBand] + 1);
766 #else
767           for (i = startSample; i < stopSample; i++) {
768             FIXP_DBL accu1, accu2;
769 
770             accu1 = (fMultDiv2(a0r, lowBandReal[LPC_ORDER + i - 1]) -
771                      fMultDiv2(a0i, lowBandImag[LPC_ORDER + i - 1]) +
772                      fMultDiv2(a1r, lowBandReal[LPC_ORDER + i - 2]) -
773                      fMultDiv2(a1i, lowBandImag[LPC_ORDER + i - 2])) >>
774                     dynamicScale;
775             accu2 = (fMultDiv2(a0i, lowBandReal[LPC_ORDER + i - 1]) +
776                      fMultDiv2(a0r, lowBandImag[LPC_ORDER + i - 1]) +
777                      fMultDiv2(a1i, lowBandReal[LPC_ORDER + i - 2]) +
778                      fMultDiv2(a1r, lowBandImag[LPC_ORDER + i - 2])) >>
779                     dynamicScale;
780 
781             accu1 = (lowBandReal[LPC_ORDER + i] >> descale) + (accu1 << 1);
782             accu2 = (lowBandImag[LPC_ORDER + i] >> descale) + (accu2 << 1);
783             if (fPreWhitening) {
784               accu1 = scaleValueSaturate(
785                   fMultDiv2(accu1, preWhiteningGains[loBand]),
786                   preWhiteningGains_exp[loBand] + 1);
787               accu2 = scaleValueSaturate(
788                   fMultDiv2(accu2, preWhiteningGains[loBand]),
789                   preWhiteningGains_exp[loBand] + 1);
790             }
791             qmfBufferReal[i][hiBand] = accu1;
792             qmfBufferImag[i][hiBand] = accu2;
793           }
794 #endif
795         } else {
796           FDK_ASSERT(dynamicScale >= 0);
797           calc_qmfBufferReal(
798               qmfBufferReal, &(lowBandReal[LPC_ORDER + startSample - 2]),
799               startSample, stopSample, hiBand, dynamicScale,
800               fMin(DFRACT_BITS - 1, (LPC_SCALE_FACTOR + dynamicScale)), a0r,
801               a1r);
802         }
803       } /* bw <= 0 */
804 
805       patch++;
806 
807     } /* inner loop over patches */
808 
809     /*
810      * store the unmodified filter coefficients if there is
811      * an overlapping envelope
812      *****************************************************************/
813 
814   } /* outer loop over bands (loBand) */
815 
816   if (useLP) {
817     for (loBand = pSettings->lbStartPatching;
818          loBand < pSettings->lbStopPatching; loBand++) {
819       patch = 0;
820       while (patch < pSettings->noOfPatches) {
821         UCHAR hiBand = loBand + patchParam[patch].targetBandOffs;
822 
823         if (loBand < patchParam[patch].sourceStartBand ||
824             loBand >= patchParam[patch].sourceStopBand ||
825             hiBand >= (64) /* Highband out of range (biterror) */
826         ) {
827           /* Lowband not in current patch or highband out of range (might be
828            * caused by biterrors)- proceed */
829           patch++;
830           continue;
831         }
832 
833         if (hiBand != patchParam[patch].targetStartBand)
834           degreeAlias[hiBand] = degreeAlias[loBand];
835 
836         patch++;
837       }
838     } /* end  for loop */
839   }
840 
841   for (i = 0; i < nInvfBands; i++) {
842     hLppTrans->bwVectorOld[i] = bwVector[i];
843   }
844 
845   /*
846     set high band scale factor
847   */
848   sbrScaleFactor->hb_scale = comLowBandScale - (LPC_SCALE_FACTOR);
849 }
850 
lppTransposerHBE(HANDLE_SBR_LPP_TRANS hLppTrans,HANDLE_HBE_TRANSPOSER hQmfTransposer,QMF_SCALE_FACTOR * sbrScaleFactor,FIXP_DBL ** qmfBufferReal,FIXP_DBL ** qmfBufferImag,const int timeStep,const int firstSlotOffs,const int lastSlotOffs,const int nInvfBands,INVF_MODE * sbr_invf_mode,INVF_MODE * sbr_invf_mode_prev)851 void lppTransposerHBE(
852     HANDLE_SBR_LPP_TRANS hLppTrans, /*!< Handle of lpp transposer  */
853     HANDLE_HBE_TRANSPOSER hQmfTransposer,
854     QMF_SCALE_FACTOR *sbrScaleFactor, /*!< Scaling factors */
855     FIXP_DBL **qmfBufferReal, /*!< Pointer to pointer to real part of subband
856                                  samples (source) */
857     FIXP_DBL **qmfBufferImag, /*!< Pointer to pointer to imaginary part of
858                                  subband samples (source) */
859     const int timeStep,       /*!< Time step of envelope */
860     const int firstSlotOffs,  /*!< Start position in time */
861     const int lastSlotOffs,   /*!< Number of overlap-slots into next frame */
862     const int nInvfBands,     /*!< Number of bands for inverse filtering */
863     INVF_MODE *sbr_invf_mode, /*!< Current inverse filtering modes */
864     INVF_MODE *sbr_invf_mode_prev /*!< Previous inverse filtering modes */
865 ) {
866   INT bwIndex;
867   FIXP_DBL bwVector[MAX_NUM_PATCHES_HBE]; /*!< pole moving factors */
868 
869   int i;
870   int loBand, start, stop;
871   TRANSPOSER_SETTINGS *pSettings = hLppTrans->pSettings;
872   PATCH_PARAM *patchParam = pSettings->patchParam;
873 
874   FIXP_SGL alphar[LPC_ORDER], a0r, a1r;
875   FIXP_SGL alphai[LPC_ORDER], a0i = 0, a1i = 0;
876   FIXP_SGL bw = FL2FXCONST_SGL(0.0f);
877 
878   int autoCorrLength;
879 
880   ACORR_COEFS ac;
881   int startSample;
882   int stopSample;
883   int stopSampleClear;
884 
885   int comBandScale;
886   int ovLowBandShift;
887   int lowBandShift;
888   /*  int ovHighBandShift;*/
889 
890   alphai[0] = FL2FXCONST_SGL(0.0f);
891   alphai[1] = FL2FXCONST_SGL(0.0f);
892 
893   startSample = firstSlotOffs * timeStep;
894   stopSample = pSettings->nCols + lastSlotOffs * timeStep;
895 
896   inverseFilteringLevelEmphasis(hLppTrans, nInvfBands, sbr_invf_mode,
897                                 sbr_invf_mode_prev, bwVector);
898 
899   stopSampleClear = stopSample;
900 
901   autoCorrLength = pSettings->nCols + pSettings->overlap;
902 
903   if (pSettings->noOfPatches > 0) {
904     /* Set upper subbands to zero:
905        This is required in case that the patches do not cover the complete
906        highband (because the last patch would be too short). Possible
907        optimization: Clearing bands up to usb would be sufficient here. */
908     int targetStopBand =
909         patchParam[pSettings->noOfPatches - 1].targetStartBand +
910         patchParam[pSettings->noOfPatches - 1].numBandsInPatch;
911 
912     int memSize = ((64) - targetStopBand) * sizeof(FIXP_DBL);
913 
914     for (i = startSample; i < stopSampleClear; i++) {
915       FDKmemclear(&qmfBufferReal[i][targetStopBand], memSize);
916       FDKmemclear(&qmfBufferImag[i][targetStopBand], memSize);
917     }
918   }
919 #ifdef __ANDROID__
920   else {
921     // Safetynet logging
922     android_errorWriteLog(0x534e4554, "112160868");
923   }
924 #endif
925 
926   /*
927   Calc common low band scale factor
928   */
929   comBandScale = sbrScaleFactor->hb_scale;
930 
931   ovLowBandShift = sbrScaleFactor->hb_scale - comBandScale;
932   lowBandShift = sbrScaleFactor->hb_scale - comBandScale;
933   /*  ovHighBandShift = firstSlotOffs == 0 ? ovLowBandShift:0;*/
934 
935   /* outer loop over bands to do analysis only once for each band */
936 
937   start = hQmfTransposer->startBand;
938   stop = hQmfTransposer->stopBand;
939 
940   for (loBand = start; loBand < stop; loBand++) {
941     bwIndex = 0;
942 
943     FIXP_DBL lowBandReal[(((1024) / (32) * (4) / 2) + (3 * (4))) + LPC_ORDER];
944     FIXP_DBL lowBandImag[(((1024) / (32) * (4) / 2) + (3 * (4))) + LPC_ORDER];
945 
946     int resetLPCCoeffs = 0;
947     int dynamicScale = DFRACT_BITS - 1 - LPC_SCALE_FACTOR;
948     int acDetScale = 0; /* scaling of autocorrelation determinant */
949 
950     for (i = 0; i < LPC_ORDER; i++) {
951       lowBandReal[i] = hLppTrans->lpcFilterStatesRealHBE[i][loBand];
952       lowBandImag[i] = hLppTrans->lpcFilterStatesImagHBE[i][loBand];
953     }
954 
955     for (; i < LPC_ORDER + firstSlotOffs * timeStep; i++) {
956       lowBandReal[i] = hLppTrans->lpcFilterStatesRealHBE[i][loBand];
957       lowBandImag[i] = hLppTrans->lpcFilterStatesImagHBE[i][loBand];
958     }
959 
960     /*
961     Take old slope length qmf slot source values out of (overlap)qmf buffer
962     */
963     for (i = firstSlotOffs * timeStep;
964          i < pSettings->nCols + pSettings->overlap; i++) {
965       lowBandReal[i + LPC_ORDER] = qmfBufferReal[i][loBand];
966       lowBandImag[i + LPC_ORDER] = qmfBufferImag[i][loBand];
967     }
968 
969     /* store unmodified values to buffer */
970     for (i = 0; i < LPC_ORDER + pSettings->overlap; i++) {
971       hLppTrans->lpcFilterStatesRealHBE[i][loBand] =
972           qmfBufferReal[pSettings->nCols - LPC_ORDER + i][loBand];
973       hLppTrans->lpcFilterStatesImagHBE[i][loBand] =
974           qmfBufferImag[pSettings->nCols - LPC_ORDER + i][loBand];
975     }
976 
977     /*
978     Determine dynamic scaling value.
979     */
980     dynamicScale =
981         fixMin(dynamicScale,
982                getScalefactor(lowBandReal, LPC_ORDER + pSettings->overlap) +
983                    ovLowBandShift);
984     dynamicScale =
985         fixMin(dynamicScale,
986                getScalefactor(&lowBandReal[LPC_ORDER + pSettings->overlap],
987                               pSettings->nCols) +
988                    lowBandShift);
989     dynamicScale =
990         fixMin(dynamicScale,
991                getScalefactor(lowBandImag, LPC_ORDER + pSettings->overlap) +
992                    ovLowBandShift);
993     dynamicScale =
994         fixMin(dynamicScale,
995                getScalefactor(&lowBandImag[LPC_ORDER + pSettings->overlap],
996                               pSettings->nCols) +
997                    lowBandShift);
998 
999     dynamicScale = fixMax(
1000         0, dynamicScale - 1); /* one additional bit headroom to prevent -1.0 */
1001 
1002     /*
1003     Scale temporal QMF buffer.
1004     */
1005     scaleValues(&lowBandReal[0], LPC_ORDER + pSettings->overlap,
1006                 dynamicScale - ovLowBandShift);
1007     scaleValues(&lowBandReal[LPC_ORDER + pSettings->overlap], pSettings->nCols,
1008                 dynamicScale - lowBandShift);
1009     scaleValues(&lowBandImag[0], LPC_ORDER + pSettings->overlap,
1010                 dynamicScale - ovLowBandShift);
1011     scaleValues(&lowBandImag[LPC_ORDER + pSettings->overlap], pSettings->nCols,
1012                 dynamicScale - lowBandShift);
1013 
1014     acDetScale += autoCorr2nd_cplx(&ac, lowBandReal + LPC_ORDER,
1015                                    lowBandImag + LPC_ORDER, autoCorrLength);
1016 
1017     /* Examine dynamic of determinant in autocorrelation. */
1018     acDetScale += 2 * (comBandScale + dynamicScale);
1019     acDetScale *= 2;            /* two times reflection coefficent scaling */
1020     acDetScale += ac.det_scale; /* ac scaling of determinant */
1021 
1022     /* In case of determinant < 10^-38, resetLPCCoeffs=1 has to be enforced. */
1023     if (acDetScale > 126) {
1024       resetLPCCoeffs = 1;
1025     }
1026 
1027     alphar[1] = FL2FXCONST_SGL(0.0f);
1028     alphai[1] = FL2FXCONST_SGL(0.0f);
1029 
1030     if (ac.det != FL2FXCONST_DBL(0.0f)) {
1031       FIXP_DBL tmp, absTmp, absDet;
1032 
1033       absDet = fixp_abs(ac.det);
1034 
1035       tmp = (fMultDiv2(ac.r01r, ac.r12r) >> (LPC_SCALE_FACTOR - 1)) -
1036             ((fMultDiv2(ac.r01i, ac.r12i) + fMultDiv2(ac.r02r, ac.r11r)) >>
1037              (LPC_SCALE_FACTOR - 1));
1038       absTmp = fixp_abs(tmp);
1039 
1040       /*
1041       Quick check: is first filter coeff >= 1(4)
1042       */
1043       {
1044         INT scale;
1045         FIXP_DBL result = fDivNorm(absTmp, absDet, &scale);
1046         scale = scale + ac.det_scale;
1047 
1048         if ((scale > 0) && (result >= (FIXP_DBL)MAXVAL_DBL >> scale)) {
1049           resetLPCCoeffs = 1;
1050         } else {
1051           alphar[1] = FX_DBL2FX_SGL(scaleValue(result, scale));
1052           if ((tmp < FL2FX_DBL(0.0f)) ^ (ac.det < FL2FX_DBL(0.0f))) {
1053             alphar[1] = -alphar[1];
1054           }
1055         }
1056       }
1057 
1058       tmp = (fMultDiv2(ac.r01i, ac.r12r) >> (LPC_SCALE_FACTOR - 1)) +
1059             ((fMultDiv2(ac.r01r, ac.r12i) -
1060               (FIXP_DBL)fMultDiv2(ac.r02i, ac.r11r)) >>
1061              (LPC_SCALE_FACTOR - 1));
1062 
1063       absTmp = fixp_abs(tmp);
1064 
1065       /*
1066       Quick check: is second filter coeff >= 1(4)
1067       */
1068       {
1069         INT scale;
1070         FIXP_DBL result = fDivNorm(absTmp, absDet, &scale);
1071         scale = scale + ac.det_scale;
1072 
1073         if ((scale > 0) &&
1074             (result >= /*FL2FXCONST_DBL(1.f)*/ (FIXP_DBL)MAXVAL_DBL >> scale)) {
1075           resetLPCCoeffs = 1;
1076         } else {
1077           alphai[1] = FX_DBL2FX_SGL(scaleValue(result, scale));
1078           if ((tmp < FL2FX_DBL(0.0f)) ^ (ac.det < FL2FX_DBL(0.0f))) {
1079             alphai[1] = -alphai[1];
1080           }
1081         }
1082       }
1083     }
1084 
1085     alphar[0] = FL2FXCONST_SGL(0.0f);
1086     alphai[0] = FL2FXCONST_SGL(0.0f);
1087 
1088     if (ac.r11r != FL2FXCONST_DBL(0.0f)) {
1089       /* ac.r11r is always >=0 */
1090       FIXP_DBL tmp, absTmp;
1091 
1092       tmp = (ac.r01r >> (LPC_SCALE_FACTOR + 1)) +
1093             (fMultDiv2(alphar[1], ac.r12r) + fMultDiv2(alphai[1], ac.r12i));
1094 
1095       absTmp = fixp_abs(tmp);
1096 
1097       /*
1098       Quick check: is first filter coeff >= 1(4)
1099       */
1100 
1101       if (absTmp >= (ac.r11r >> 1)) {
1102         resetLPCCoeffs = 1;
1103       } else {
1104         INT scale;
1105         FIXP_DBL result = fDivNorm(absTmp, fixp_abs(ac.r11r), &scale);
1106         alphar[0] = FX_DBL2FX_SGL(scaleValue(result, scale + 1));
1107 
1108         if ((tmp > FL2FX_DBL(0.0f)) ^ (ac.r11r < FL2FX_DBL(0.0f)))
1109           alphar[0] = -alphar[0];
1110       }
1111 
1112       tmp = (ac.r01i >> (LPC_SCALE_FACTOR + 1)) +
1113             (fMultDiv2(alphai[1], ac.r12r) - fMultDiv2(alphar[1], ac.r12i));
1114 
1115       absTmp = fixp_abs(tmp);
1116 
1117       /*
1118       Quick check: is second filter coeff >= 1(4)
1119       */
1120       if (absTmp >= (ac.r11r >> 1)) {
1121         resetLPCCoeffs = 1;
1122       } else {
1123         INT scale;
1124         FIXP_DBL result = fDivNorm(absTmp, fixp_abs(ac.r11r), &scale);
1125         alphai[0] = FX_DBL2FX_SGL(scaleValue(result, scale + 1));
1126         if ((tmp > FL2FX_DBL(0.0f)) ^ (ac.r11r < FL2FX_DBL(0.0f))) {
1127           alphai[0] = -alphai[0];
1128         }
1129       }
1130     }
1131 
1132     /* Now check the quadratic criteria */
1133     if ((fMultDiv2(alphar[0], alphar[0]) + fMultDiv2(alphai[0], alphai[0])) >=
1134         FL2FXCONST_DBL(0.5f)) {
1135       resetLPCCoeffs = 1;
1136     }
1137     if ((fMultDiv2(alphar[1], alphar[1]) + fMultDiv2(alphai[1], alphai[1])) >=
1138         FL2FXCONST_DBL(0.5f)) {
1139       resetLPCCoeffs = 1;
1140     }
1141 
1142     if (resetLPCCoeffs) {
1143       alphar[0] = FL2FXCONST_SGL(0.0f);
1144       alphar[1] = FL2FXCONST_SGL(0.0f);
1145       alphai[0] = FL2FXCONST_SGL(0.0f);
1146       alphai[1] = FL2FXCONST_SGL(0.0f);
1147     }
1148 
1149     while (bwIndex < MAX_NUM_PATCHES - 1 &&
1150            loBand >= pSettings->bwBorders[bwIndex]) {
1151       bwIndex++;
1152     }
1153 
1154     /*
1155     Filter Step 2: add the left slope with the current filter to the buffer
1156     pure source values are already in there
1157     */
1158     bw = FX_DBL2FX_SGL(bwVector[bwIndex]);
1159 
1160     a0r = FX_DBL2FX_SGL(
1161         fMult(bw, alphar[0])); /* Apply current bandwidth expansion factor */
1162     a0i = FX_DBL2FX_SGL(fMult(bw, alphai[0]));
1163     bw = FX_DBL2FX_SGL(fPow2(bw));
1164     a1r = FX_DBL2FX_SGL(fMult(bw, alphar[1]));
1165     a1i = FX_DBL2FX_SGL(fMult(bw, alphai[1]));
1166 
1167     /*
1168     Filter Step 3: insert the middle part which won't be windowed
1169     */
1170     if (bw <= FL2FXCONST_SGL(0.0f)) {
1171       int descale = fixMin(DFRACT_BITS - 1, (LPC_SCALE_FACTOR + dynamicScale));
1172       for (i = startSample; i < stopSample; i++) {
1173         qmfBufferReal[i][loBand] = lowBandReal[LPC_ORDER + i] >> descale;
1174         qmfBufferImag[i][loBand] = lowBandImag[LPC_ORDER + i] >> descale;
1175       }
1176     } else { /* bw <= 0 */
1177 
1178       int descale = fixMin(DFRACT_BITS - 1, (LPC_SCALE_FACTOR + dynamicScale));
1179 
1180       for (i = startSample; i < stopSample; i++) {
1181         FIXP_DBL accu1, accu2;
1182 
1183         accu1 = (fMultDiv2(a0r, lowBandReal[LPC_ORDER + i - 1]) -
1184                  fMultDiv2(a0i, lowBandImag[LPC_ORDER + i - 1]) +
1185                  fMultDiv2(a1r, lowBandReal[LPC_ORDER + i - 2]) -
1186                  fMultDiv2(a1i, lowBandImag[LPC_ORDER + i - 2])) >>
1187                 dynamicScale;
1188         accu2 = (fMultDiv2(a0i, lowBandReal[LPC_ORDER + i - 1]) +
1189                  fMultDiv2(a0r, lowBandImag[LPC_ORDER + i - 1]) +
1190                  fMultDiv2(a1i, lowBandReal[LPC_ORDER + i - 2]) +
1191                  fMultDiv2(a1r, lowBandImag[LPC_ORDER + i - 2])) >>
1192                 dynamicScale;
1193 
1194         qmfBufferReal[i][loBand] =
1195             (lowBandReal[LPC_ORDER + i] >> descale) + (accu1 << 1);
1196         qmfBufferImag[i][loBand] =
1197             (lowBandImag[LPC_ORDER + i] >> descale) + (accu2 << 1);
1198       }
1199     } /* bw <= 0 */
1200 
1201     /*
1202      * store the unmodified filter coefficients if there is
1203      * an overlapping envelope
1204      *****************************************************************/
1205 
1206   } /* outer loop over bands (loBand) */
1207 
1208   for (i = 0; i < nInvfBands; i++) {
1209     hLppTrans->bwVectorOld[i] = bwVector[i];
1210   }
1211 
1212   /*
1213   set high band scale factor
1214   */
1215   sbrScaleFactor->hb_scale = comBandScale - (LPC_SCALE_FACTOR);
1216 }
1217 
1218 /*!
1219  *
1220  * \brief Initialize one low power transposer instance
1221  *
1222  *
1223  */
1224 SBR_ERROR
createLppTransposer(HANDLE_SBR_LPP_TRANS hs,TRANSPOSER_SETTINGS * pSettings,const int highBandStartSb,UCHAR * v_k_master,const int numMaster,const int usb,const int timeSlots,const int nCols,UCHAR * noiseBandTable,const int noNoiseBands,UINT fs,const int chan,const int overlap)1225 createLppTransposer(
1226     HANDLE_SBR_LPP_TRANS hs,        /*!< Handle of low power transposer  */
1227     TRANSPOSER_SETTINGS *pSettings, /*!< Pointer to settings */
1228     const int highBandStartSb,      /*!< ? */
1229     UCHAR *v_k_master,              /*!< Master table */
1230     const int numMaster,            /*!< Valid entries in master table */
1231     const int usb,                  /*!< Highband area stop subband */
1232     const int timeSlots,            /*!< Number of time slots */
1233     const int nCols,                /*!< Number of colums (codec qmf bank) */
1234     UCHAR *noiseBandTable,  /*!< Mapping of SBR noise bands to QMF bands */
1235     const int noNoiseBands, /*!< Number of noise bands */
1236     UINT fs,                /*!< Sample Frequency */
1237     const int chan,         /*!< Channel number */
1238     const int overlap) {
1239   /* FB inverse filtering settings */
1240   hs->pSettings = pSettings;
1241 
1242   pSettings->nCols = nCols;
1243   pSettings->overlap = overlap;
1244 
1245   switch (timeSlots) {
1246     case 15:
1247     case 16:
1248       break;
1249 
1250     default:
1251       return SBRDEC_UNSUPPORTED_CONFIG; /* Unimplemented */
1252   }
1253 
1254   if (chan == 0) {
1255     /* Init common data only once */
1256     hs->pSettings->nCols = nCols;
1257 
1258     return resetLppTransposer(hs, highBandStartSb, v_k_master, numMaster,
1259                               noiseBandTable, noNoiseBands, usb, fs);
1260   }
1261   return SBRDEC_OK;
1262 }
1263 
findClosestEntry(UCHAR goalSb,UCHAR * v_k_master,UCHAR numMaster,UCHAR direction)1264 static int findClosestEntry(UCHAR goalSb, UCHAR *v_k_master, UCHAR numMaster,
1265                             UCHAR direction) {
1266   int index;
1267 
1268   if (goalSb <= v_k_master[0]) return v_k_master[0];
1269 
1270   if (goalSb >= v_k_master[numMaster]) return v_k_master[numMaster];
1271 
1272   if (direction) {
1273     index = 0;
1274     while (v_k_master[index] < goalSb) {
1275       index++;
1276     }
1277   } else {
1278     index = numMaster;
1279     while (v_k_master[index] > goalSb) {
1280       index--;
1281     }
1282   }
1283 
1284   return v_k_master[index];
1285 }
1286 
1287 /*!
1288  *
1289  * \brief Reset memory for one lpp transposer instance
1290  *
1291  * \return SBRDEC_OK on success, SBRDEC_UNSUPPORTED_CONFIG on error
1292  */
1293 SBR_ERROR
resetLppTransposer(HANDLE_SBR_LPP_TRANS hLppTrans,UCHAR highBandStartSb,UCHAR * v_k_master,UCHAR numMaster,UCHAR * noiseBandTable,UCHAR noNoiseBands,UCHAR usb,UINT fs)1294 resetLppTransposer(
1295     HANDLE_SBR_LPP_TRANS hLppTrans, /*!< Handle of lpp transposer  */
1296     UCHAR highBandStartSb,          /*!< High band area: start subband */
1297     UCHAR *v_k_master,              /*!< Master table */
1298     UCHAR numMaster,                /*!< Valid entries in master table */
1299     UCHAR *noiseBandTable, /*!< Mapping of SBR noise bands to QMF bands */
1300     UCHAR noNoiseBands,    /*!< Number of noise bands */
1301     UCHAR usb,             /*!< High band area: stop subband */
1302     UINT fs                /*!< SBR output sampling frequency */
1303 ) {
1304   TRANSPOSER_SETTINGS *pSettings = hLppTrans->pSettings;
1305   PATCH_PARAM *patchParam = pSettings->patchParam;
1306 
1307   int i, patch;
1308   int targetStopBand;
1309   int sourceStartBand;
1310   int patchDistance;
1311   int numBandsInPatch;
1312 
1313   int lsb = v_k_master[0]; /* Start subband expressed in "non-critical" sampling
1314                               terms*/
1315   int xoverOffset = highBandStartSb -
1316                     lsb; /* Calculate distance in QMF bands between k0 and kx */
1317   int startFreqHz;
1318 
1319   int desiredBorder;
1320 
1321   usb = fixMin(usb, v_k_master[numMaster]); /* Avoid endless loops (compare with
1322                                                float code). */
1323 
1324   /*
1325    * Plausibility check
1326    */
1327 
1328   if (pSettings->nCols == 64) {
1329     if (lsb < 4) {
1330       /* 4:1 SBR Requirement k0 >= 4 missed! */
1331       return SBRDEC_UNSUPPORTED_CONFIG;
1332     }
1333   } else if (lsb - SHIFT_START_SB < 4) {
1334     return SBRDEC_UNSUPPORTED_CONFIG;
1335   }
1336 
1337   /*
1338    * Initialize the patching parameter
1339    */
1340   /* ISO/IEC 14496-3 (Figure 4.48): goalSb = round( 2.048e6 / fs ) */
1341   desiredBorder = (((2048000 * 2) / fs) + 1) >> 1;
1342 
1343   desiredBorder = findClosestEntry(desiredBorder, v_k_master, numMaster,
1344                                    1); /* Adapt region to master-table */
1345 
1346   /* First patch */
1347   sourceStartBand = SHIFT_START_SB + xoverOffset;
1348   targetStopBand = lsb + xoverOffset; /* upperBand */
1349 
1350   /* Even (odd) numbered channel must be patched to even (odd) numbered channel
1351    */
1352   patch = 0;
1353   while (targetStopBand < usb) {
1354     /* Too many patches?
1355        Allow MAX_NUM_PATCHES+1 patches here.
1356        we need to check later again, since patch might be the highest patch
1357        AND contain less than 3 bands => actual number of patches will be reduced
1358        by 1.
1359     */
1360     if (patch > MAX_NUM_PATCHES) {
1361       return SBRDEC_UNSUPPORTED_CONFIG;
1362     }
1363 
1364     patchParam[patch].guardStartBand = targetStopBand;
1365     patchParam[patch].targetStartBand = targetStopBand;
1366 
1367     numBandsInPatch =
1368         desiredBorder - targetStopBand; /* Get the desired range of the patch */
1369 
1370     if (numBandsInPatch >= lsb - sourceStartBand) {
1371       /* Desired number bands are not available -> patch whole source range */
1372       patchDistance =
1373           targetStopBand - sourceStartBand; /* Get the targetOffset */
1374       patchDistance =
1375           patchDistance & ~1; /* Rounding off odd numbers and make all even */
1376       numBandsInPatch =
1377           lsb - (targetStopBand -
1378                  patchDistance); /* Update number of bands to be patched */
1379       numBandsInPatch = findClosestEntry(targetStopBand + numBandsInPatch,
1380                                          v_k_master, numMaster, 0) -
1381                         targetStopBand; /* Adapt region to master-table */
1382     }
1383 
1384     if (pSettings->nCols == 64) {
1385       if (numBandsInPatch == 0 && sourceStartBand == SHIFT_START_SB) {
1386         return SBRDEC_UNSUPPORTED_CONFIG;
1387       }
1388     }
1389 
1390     /* Desired number bands are available -> get the minimal even patching
1391      * distance */
1392     patchDistance =
1393         numBandsInPatch + targetStopBand - lsb; /* Get minimal distance */
1394     patchDistance = (patchDistance + 1) &
1395                     ~1; /* Rounding up odd numbers and make all even */
1396 
1397     if (numBandsInPatch > 0) {
1398       patchParam[patch].sourceStartBand = targetStopBand - patchDistance;
1399       patchParam[patch].targetBandOffs = patchDistance;
1400       patchParam[patch].numBandsInPatch = numBandsInPatch;
1401       patchParam[patch].sourceStopBand =
1402           patchParam[patch].sourceStartBand + numBandsInPatch;
1403 
1404       targetStopBand += patchParam[patch].numBandsInPatch;
1405       patch++;
1406     }
1407 
1408     /* All patches but first */
1409     sourceStartBand = SHIFT_START_SB;
1410 
1411     /* Check if we are close to desiredBorder */
1412     if (desiredBorder - targetStopBand < 3) /* MPEG doc */
1413     {
1414       desiredBorder = usb;
1415     }
1416   }
1417 
1418   patch--;
1419 
1420   /* If highest patch contains less than three subband: skip it */
1421   if ((patch > 0) && (patchParam[patch].numBandsInPatch < 3)) {
1422     patch--;
1423     targetStopBand =
1424         patchParam[patch].targetStartBand + patchParam[patch].numBandsInPatch;
1425   }
1426 
1427   /* now check if we don't have one too many */
1428   if (patch >= MAX_NUM_PATCHES) {
1429     return SBRDEC_UNSUPPORTED_CONFIG;
1430   }
1431 
1432   pSettings->noOfPatches = patch + 1;
1433 
1434   /* Check lowest and highest source subband */
1435   pSettings->lbStartPatching = targetStopBand;
1436   pSettings->lbStopPatching = 0;
1437   for (patch = 0; patch < pSettings->noOfPatches; patch++) {
1438     pSettings->lbStartPatching =
1439         fixMin(pSettings->lbStartPatching, patchParam[patch].sourceStartBand);
1440     pSettings->lbStopPatching =
1441         fixMax(pSettings->lbStopPatching, patchParam[patch].sourceStopBand);
1442   }
1443 
1444   for (i = 0; i < noNoiseBands; i++) {
1445     pSettings->bwBorders[i] = noiseBandTable[i + 1];
1446   }
1447   for (; i < MAX_NUM_NOISE_VALUES; i++) {
1448     pSettings->bwBorders[i] = 255;
1449   }
1450 
1451   /*
1452    * Choose whitening factors
1453    */
1454 
1455   startFreqHz =
1456       ((lsb + xoverOffset) * fs) >> 7; /* Shift does a division by 2*(64) */
1457 
1458   for (i = 1; i < NUM_WHFACTOR_TABLE_ENTRIES; i++) {
1459     if (startFreqHz < FDK_sbrDecoder_sbr_whFactorsIndex[i]) break;
1460   }
1461   i--;
1462 
1463   pSettings->whFactors.off = FDK_sbrDecoder_sbr_whFactorsTable[i][0];
1464   pSettings->whFactors.transitionLevel =
1465       FDK_sbrDecoder_sbr_whFactorsTable[i][1];
1466   pSettings->whFactors.lowLevel = FDK_sbrDecoder_sbr_whFactorsTable[i][2];
1467   pSettings->whFactors.midLevel = FDK_sbrDecoder_sbr_whFactorsTable[i][3];
1468   pSettings->whFactors.highLevel = FDK_sbrDecoder_sbr_whFactorsTable[i][4];
1469 
1470   return SBRDEC_OK;
1471 }
1472