• 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 /**************************** AAC decoder library ******************************
96 
97    Author(s):   Josef Hoepfl
98 
99    Description: independent channel concealment
100 
101 *******************************************************************************/
102 
103 /*!
104   \page concealment AAC core concealment
105 
106   This AAC core implementation includes a concealment function, which can be
107   enabled using the several defines during compilation.
108 
109   There are various tests inside the core, starting with simple CRC tests and
110   ending in a variety of plausibility checks. If such a check indicates an
111   invalid bitstream, then concealment is applied.
112 
113   Concealment is also applied when the calling main program indicates a
114   distorted or missing data frame using the frameOK flag. This is used for error
115   detection on the transport layer. (See below)
116 
117   There are three concealment-modes:
118 
119   1) Muting: The spectral data is simply set to zero in case of an detected
120   error.
121 
122   2) Noise substitution: In case of an detected error, concealment copies the
123   last frame and adds attenuates the spectral data. For this mode you have to
124   set the #CONCEAL_NOISE define. Noise substitution adds no additional delay.
125 
126   3) Interpolation: The interpolation routine swaps the spectral data from the
127   previous and the current frame just before the final frequency to time
128   conversion. In case a single frame is corrupted, concealmant interpolates
129   between the last good and the first good frame to create the spectral data for
130   the missing frame. If multiple frames are corrupted, concealment implements
131   first a fade out based on slightly modified spectral values from the last good
132      frame. As soon as good frames are available, concealmant fades in the new
133   spectral data. For this mode you have to set the #CONCEAL_INTER define. Note
134   that in this case, you also need to set #SBR_BS_DELAY_ENABLE, which basically
135   adds approriate delay in the SBR decoder. Note that the
136   Interpolating-Concealment increases the delay of your decoder by one frame and
137   that it does require additional resources such as memory and computational
138   complexity.
139 
140   <h2>How concealment can be used with errors on the transport layer</h2>
141 
142   Many errors can or have to be detected on the transport layer. For example in
143   IP based systems packet loss can occur. The transport protocol used should
144   indicate such packet loss by inserting an empty frame with frameOK=0.
145 */
146 
147 #include "conceal.h"
148 
149 #include "aac_rom.h"
150 #include "genericStds.h"
151 
152 /* PNS (of block) */
153 #include "aacdec_pns.h"
154 #include "block.h"
155 
156 #define CONCEAL_DFLT_COMF_NOISE_LEVEL (0x100000)
157 
158 #define CONCEAL_NOT_DEFINED ((UCHAR)-1)
159 
160 /* default settings */
161 #define CONCEAL_DFLT_FADEOUT_FRAMES (6)
162 #define CONCEAL_DFLT_FADEIN_FRAMES (5)
163 #define CONCEAL_DFLT_MUTE_RELEASE_FRAMES (0)
164 
165 #define CONCEAL_DFLT_FADE_FACTOR (0.707106781186548f) /* 1/sqrt(2) */
166 
167 /* some often used constants: */
168 #define FIXP_ZERO FL2FXCONST_DBL(0.0f)
169 #define FIXP_ONE FL2FXCONST_DBL(1.0f)
170 #define FIXP_FL_CORRECTION FL2FXCONST_DBL(0.53333333333333333f)
171 
172 /* For parameter conversion */
173 #define CONCEAL_PARAMETER_BITS (8)
174 #define CONCEAL_MAX_QUANT_FACTOR ((1 << CONCEAL_PARAMETER_BITS) - 1)
175 /*#define CONCEAL_MIN_ATTENUATION_FACTOR_025  ( FL2FXCONST_DBL(0.971627951577106174) )*/ /* -0.25 dB */
176 #define CONCEAL_MIN_ATTENUATION_FACTOR_025_LD \
177   FL2FXCONST_DBL(-0.041524101186092029596853445212299)
178 /*#define CONCEAL_MIN_ATTENUATION_FACTOR_050  ( FL2FXCONST_DBL(0.944060876285923380) )*/ /* -0.50 dB */
179 #define CONCEAL_MIN_ATTENUATION_FACTOR_050_LD \
180   FL2FXCONST_DBL(-0.083048202372184059253597008145293)
181 
182 typedef enum {
183   CConcealment_NoExpand,
184   CConcealment_Expand,
185   CConcealment_Compress
186 } CConcealmentExpandType;
187 
188 static const FIXP_SGL facMod4Table[4] = {
189     FL2FXCONST_SGL(0.500000000f), /* FIXP_SGL(0x4000),  2^-(1-0,00) */
190     FL2FXCONST_SGL(0.594603558f), /* FIXP_SGL(0x4c1b),  2^-(1-0,25) */
191     FL2FXCONST_SGL(0.707106781f), /* FIXP_SGL(0x5a82),  2^-(1-0,50) */
192     FL2FXCONST_SGL(0.840896415f)  /* FIXP_SGL(0x6ba2)   2^-(1-0,75) */
193 };
194 
195 static void CConcealment_CalcBandEnergy(
196     FIXP_DBL *spectrum, const SamplingRateInfo *pSamplingRateInfo,
197     const int blockType, CConcealmentExpandType ex, int *sfbEnergy);
198 
199 static void CConcealment_InterpolateBuffer(FIXP_DBL *spectrum,
200                                            SHORT *pSpecScalePrev,
201                                            SHORT *pSpecScaleAct,
202                                            SHORT *pSpecScaleOut, int *enPrv,
203                                            int *enAct, int sfbCnt,
204                                            const SHORT *pSfbOffset);
205 
206 static int CConcealment_ApplyInter(
207     CConcealmentInfo *pConcealmentInfo,
208     CAacDecoderChannelInfo *pAacDecoderChannelInfo,
209     const SamplingRateInfo *pSamplingRateInfo, const int samplesPerFrame,
210     const int improveTonal, const int frameOk, const int mute_release_active);
211 
212 static int CConcealment_ApplyNoise(
213     CConcealmentInfo *pConcealmentInfo,
214     CAacDecoderChannelInfo *pAacDecoderChannelInfo,
215     CAacDecoderStaticChannelInfo *pAacDecoderStaticChannelInfo,
216     const SamplingRateInfo *pSamplingRateInfo, const int samplesPerFrame,
217     const UINT flags);
218 
219 static void CConcealment_UpdateState(
220     CConcealmentInfo *pConcealmentInfo, int frameOk,
221     CAacDecoderStaticChannelInfo *pAacDecoderStaticChannelInfo,
222     const int samplesPerFrame, CAacDecoderChannelInfo *pAacDecoderChannelInfo);
223 
224 static void CConcealment_ApplyRandomSign(int iRandomPhase, FIXP_DBL *spec,
225                                          int samplesPerFrame);
226 
227 /* TimeDomainFading */
228 static void CConcealment_TDFadePcmAtt(int start, int len, FIXP_DBL fadeStart,
229                                       FIXP_DBL fadeStop, FIXP_PCM *pcmdata);
230 static void CConcealment_TDFadeFillFadingStations(FIXP_DBL *fadingStations,
231                                                   int *fadingSteps,
232                                                   FIXP_DBL fadeStop,
233                                                   FIXP_DBL fadeStart,
234                                                   TDfadingType fadingType);
235 static void CConcealment_TDFading_doLinearFadingSteps(int *fadingSteps);
236 
237 /* Streamline the state machine */
238 static int CConcealment_ApplyFadeOut(
239     int mode, CConcealmentInfo *pConcealmentInfo,
240     CAacDecoderStaticChannelInfo *pAacDecoderStaticChannelInfo,
241     const int samplesPerFrame, CAacDecoderChannelInfo *pAacDecoderChannelInfo);
242 
243 static int CConcealment_TDNoise_Random(ULONG *seed);
244 static void CConcealment_TDNoise_Apply(CConcealmentInfo *const pConcealmentInfo,
245                                        const int len, FIXP_PCM *const pcmdata);
246 
CConcealment_GetWinSeq(int prevWinSeq)247 static BLOCK_TYPE CConcealment_GetWinSeq(int prevWinSeq) {
248   BLOCK_TYPE newWinSeq = BLOCK_LONG;
249 
250   /* Try to have only long blocks */
251   if (prevWinSeq == BLOCK_START || prevWinSeq == BLOCK_SHORT) {
252     newWinSeq = BLOCK_STOP;
253   }
254 
255   return (newWinSeq);
256 }
257 
258 /*!
259   \brief Init common concealment information data
260 
261   \param pConcealCommonData Pointer to the concealment common data structure.
262 */
CConcealment_InitCommonData(CConcealParams * pConcealCommonData)263 void CConcealment_InitCommonData(CConcealParams *pConcealCommonData) {
264   if (pConcealCommonData != NULL) {
265     int i;
266 
267     /* Set default error concealment technique */
268     pConcealCommonData->method = ConcealMethodInter;
269 
270     pConcealCommonData->numFadeOutFrames = CONCEAL_DFLT_FADEOUT_FRAMES;
271     pConcealCommonData->numFadeInFrames = CONCEAL_DFLT_FADEIN_FRAMES;
272     pConcealCommonData->numMuteReleaseFrames = CONCEAL_DFLT_MUTE_RELEASE_FRAMES;
273 
274     pConcealCommonData->comfortNoiseLevel =
275         (FIXP_DBL)CONCEAL_DFLT_COMF_NOISE_LEVEL;
276 
277     /* Init fade factors (symetric) */
278     pConcealCommonData->fadeOutFactor[0] =
279         FL2FXCONST_SGL(CONCEAL_DFLT_FADE_FACTOR);
280     pConcealCommonData->fadeInFactor[0] = pConcealCommonData->fadeOutFactor[0];
281 
282     for (i = 1; i < CONCEAL_MAX_NUM_FADE_FACTORS; i++) {
283       pConcealCommonData->fadeOutFactor[i] =
284           FX_DBL2FX_SGL(fMult(pConcealCommonData->fadeOutFactor[i - 1],
285                               FL2FXCONST_SGL(CONCEAL_DFLT_FADE_FACTOR)));
286       pConcealCommonData->fadeInFactor[i] =
287           pConcealCommonData->fadeOutFactor[i];
288     }
289   }
290 }
291 
292 /*!
293   \brief Get current concealment method.
294 
295   \param pConcealCommonData Pointer to common concealment data (for all
296   channels)
297 */
CConcealment_GetMethod(CConcealParams * pConcealCommonData)298 CConcealmentMethod CConcealment_GetMethod(CConcealParams *pConcealCommonData) {
299   CConcealmentMethod method = ConcealMethodNone;
300 
301   if (pConcealCommonData != NULL) {
302     method = pConcealCommonData->method;
303   }
304 
305   return (method);
306 }
307 
308 /*!
309   \brief Init concealment information for each channel
310 
311   \param pConcealChannelInfo Pointer to the channel related concealment info
312   structure to be initialized. \param pConcealCommonData  Pointer to common
313   concealment data (for all channels) \param initRenderMode      Initial render
314   mode to be set for the current channel. \param samplesPerFrame     The number
315   of samples per frame.
316 */
CConcealment_InitChannelData(CConcealmentInfo * pConcealChannelInfo,CConcealParams * pConcealCommonData,AACDEC_RENDER_MODE initRenderMode,int samplesPerFrame)317 void CConcealment_InitChannelData(CConcealmentInfo *pConcealChannelInfo,
318                                   CConcealParams *pConcealCommonData,
319                                   AACDEC_RENDER_MODE initRenderMode,
320                                   int samplesPerFrame) {
321   int i;
322   pConcealChannelInfo->TDNoiseSeed = 0;
323   FDKmemclear(pConcealChannelInfo->TDNoiseStates,
324               sizeof(pConcealChannelInfo->TDNoiseStates));
325   pConcealChannelInfo->TDNoiseCoef[0] = FL2FXCONST_SGL(0.05f);
326   pConcealChannelInfo->TDNoiseCoef[1] = FL2FXCONST_SGL(0.5f);
327   pConcealChannelInfo->TDNoiseCoef[2] = FL2FXCONST_SGL(0.45f);
328 
329   pConcealChannelInfo->pConcealParams = pConcealCommonData;
330 
331   pConcealChannelInfo->lastRenderMode = initRenderMode;
332 
333   pConcealChannelInfo->windowShape = CONCEAL_NOT_DEFINED;
334   pConcealChannelInfo->windowSequence = BLOCK_LONG; /* default type */
335   pConcealChannelInfo->lastWinGrpLen = 1;
336 
337   pConcealChannelInfo->concealState = ConcealState_Ok;
338 
339   FDKmemclear(pConcealChannelInfo->spectralCoefficient,
340               1024 * sizeof(FIXP_CNCL));
341 
342   for (i = 0; i < 8; i++) {
343     pConcealChannelInfo->specScale[i] = 0;
344   }
345 
346   pConcealChannelInfo->iRandomPhase = 0;
347 
348   pConcealChannelInfo->prevFrameOk[0] = 1;
349   pConcealChannelInfo->prevFrameOk[1] = 1;
350 
351   pConcealChannelInfo->cntFadeFrames = 0;
352   pConcealChannelInfo->cntValidFrames = 0;
353   pConcealChannelInfo->fade_old = (FIXP_DBL)MAXVAL_DBL;
354   pConcealChannelInfo->winGrpOffset[0] = 0;
355   pConcealChannelInfo->winGrpOffset[1] = 0;
356   pConcealChannelInfo->attGrpOffset[0] = 0;
357   pConcealChannelInfo->attGrpOffset[1] = 0;
358 }
359 
360 /*!
361   \brief Set error concealment parameters
362 
363   \param concealParams
364   \param method
365   \param fadeOutSlope
366   \param fadeInSlope
367   \param muteRelease
368   \param comfNoiseLevel
369 */
370 AAC_DECODER_ERROR
CConcealment_SetParams(CConcealParams * concealParams,int method,int fadeOutSlope,int fadeInSlope,int muteRelease,FIXP_DBL comfNoiseLevel)371 CConcealment_SetParams(CConcealParams *concealParams, int method,
372                        int fadeOutSlope, int fadeInSlope, int muteRelease,
373                        FIXP_DBL comfNoiseLevel) {
374   /* set concealment technique */
375   if (method != AACDEC_CONCEAL_PARAM_NOT_SPECIFIED) {
376     switch ((CConcealmentMethod)method) {
377       case ConcealMethodMute:
378       case ConcealMethodNoise:
379       case ConcealMethodInter:
380         /* Be sure to enable delay adjustment of SBR decoder! */
381         if (concealParams == NULL) {
382           return AAC_DEC_INVALID_HANDLE;
383         } else {
384           /* set param */
385           concealParams->method = (CConcealmentMethod)method;
386         }
387         break;
388 
389       default:
390         return AAC_DEC_SET_PARAM_FAIL;
391     }
392   }
393 
394   /* set number of frames for fade-out slope */
395   if (fadeOutSlope != AACDEC_CONCEAL_PARAM_NOT_SPECIFIED) {
396     if ((fadeOutSlope < CONCEAL_MAX_NUM_FADE_FACTORS) && (fadeOutSlope >= 0)) {
397       if (concealParams == NULL) {
398         return AAC_DEC_INVALID_HANDLE;
399       } else {
400         /* set param */
401         concealParams->numFadeOutFrames = fadeOutSlope;
402       }
403     } else {
404       return AAC_DEC_SET_PARAM_FAIL;
405     }
406   }
407 
408   /* set number of frames for fade-in slope */
409   if (fadeInSlope != AACDEC_CONCEAL_PARAM_NOT_SPECIFIED) {
410     if ((fadeInSlope < CONCEAL_MAX_NUM_FADE_FACTORS) && (fadeInSlope >= 0)) {
411       if (concealParams == NULL) {
412         return AAC_DEC_INVALID_HANDLE;
413       } else {
414         /* set param */
415         concealParams->numFadeInFrames = fadeInSlope;
416       }
417     } else {
418       return AAC_DEC_SET_PARAM_FAIL;
419     }
420   }
421 
422   /* set number of error-free frames after which the muting will be released */
423   if (muteRelease != AACDEC_CONCEAL_PARAM_NOT_SPECIFIED) {
424     if ((muteRelease < (CONCEAL_MAX_NUM_FADE_FACTORS << 1)) &&
425         (muteRelease >= 0)) {
426       if (concealParams == NULL) {
427         return AAC_DEC_INVALID_HANDLE;
428       } else {
429         /* set param */
430         concealParams->numMuteReleaseFrames = muteRelease;
431       }
432     } else {
433       return AAC_DEC_SET_PARAM_FAIL;
434     }
435   }
436 
437   /* set confort noise level which will be inserted while in state 'muting' */
438   if (comfNoiseLevel != (FIXP_DBL)AACDEC_CONCEAL_PARAM_NOT_SPECIFIED) {
439     if ((comfNoiseLevel < (FIXP_DBL)0) ||
440         (comfNoiseLevel > (FIXP_DBL)MAXVAL_DBL)) {
441       return AAC_DEC_SET_PARAM_FAIL;
442     }
443     if (concealParams == NULL) {
444       return AAC_DEC_INVALID_HANDLE;
445     } else {
446       concealParams->comfortNoiseLevel = (FIXP_DBL)comfNoiseLevel;
447     }
448   }
449 
450   return (AAC_DEC_OK);
451 }
452 
453 /*!
454   \brief Set fade-out/in attenuation factor vectors
455 
456   \param concealParams
457   \param fadeOutAttenuationVector
458   \param fadeInAttenuationVector
459 
460   \return 0 if OK all other values indicate errors
461 */
462 AAC_DECODER_ERROR
CConcealment_SetAttenuation(CConcealParams * concealParams,const SHORT * fadeOutAttenuationVector,const SHORT * fadeInAttenuationVector)463 CConcealment_SetAttenuation(CConcealParams *concealParams,
464                             const SHORT *fadeOutAttenuationVector,
465                             const SHORT *fadeInAttenuationVector) {
466   if ((fadeOutAttenuationVector == NULL) && (fadeInAttenuationVector == NULL)) {
467     return AAC_DEC_SET_PARAM_FAIL;
468   }
469 
470   /* Fade-out factors */
471   if (fadeOutAttenuationVector != NULL) {
472     int i;
473 
474     /* check quantized factors first */
475     for (i = 0; i < CONCEAL_MAX_NUM_FADE_FACTORS; i++) {
476       if ((fadeOutAttenuationVector[i] < 0) ||
477           (fadeOutAttenuationVector[i] > CONCEAL_MAX_QUANT_FACTOR)) {
478         return AAC_DEC_SET_PARAM_FAIL;
479       }
480     }
481     if (concealParams == NULL) {
482       return AAC_DEC_INVALID_HANDLE;
483     }
484 
485     /* now dequantize factors */
486     for (i = 0; i < CONCEAL_MAX_NUM_FADE_FACTORS; i++) {
487       concealParams->fadeOutFactor[i] =
488           FX_DBL2FX_SGL(fLdPow(CONCEAL_MIN_ATTENUATION_FACTOR_025_LD, 0,
489                                (FIXP_DBL)((INT)(FL2FXCONST_DBL(1.0 / 2.0) >>
490                                                 (CONCEAL_PARAMETER_BITS - 1)) *
491                                           (INT)fadeOutAttenuationVector[i]),
492                                CONCEAL_PARAMETER_BITS));
493     }
494   }
495 
496   /* Fade-in factors */
497   if (fadeInAttenuationVector != NULL) {
498     int i;
499 
500     /* check quantized factors first */
501     for (i = 0; i < CONCEAL_MAX_NUM_FADE_FACTORS; i++) {
502       if ((fadeInAttenuationVector[i] < 0) ||
503           (fadeInAttenuationVector[i] > CONCEAL_MAX_QUANT_FACTOR)) {
504         return AAC_DEC_SET_PARAM_FAIL;
505       }
506     }
507     if (concealParams == NULL) {
508       return AAC_DEC_INVALID_HANDLE;
509     }
510 
511     /* now dequantize factors */
512     for (i = 0; i < CONCEAL_MAX_NUM_FADE_FACTORS; i++) {
513       concealParams->fadeInFactor[i] = FX_DBL2FX_SGL(
514           fLdPow(CONCEAL_MIN_ATTENUATION_FACTOR_025_LD, 0,
515                  (FIXP_DBL)((INT)(FIXP_ONE >> CONCEAL_PARAMETER_BITS) *
516                             (INT)fadeInAttenuationVector[i]),
517                  CONCEAL_PARAMETER_BITS));
518     }
519   }
520 
521   return (AAC_DEC_OK);
522 }
523 
524 /*!
525   \brief Get state of concealment module.
526 
527   \param pConcealChannelInfo
528 
529   \return Concealment state.
530 */
CConcealment_GetState(CConcealmentInfo * pConcealChannelInfo)531 CConcealmentState CConcealment_GetState(CConcealmentInfo *pConcealChannelInfo) {
532   CConcealmentState state = ConcealState_Ok;
533 
534   if (pConcealChannelInfo != NULL) {
535     state = pConcealChannelInfo->concealState;
536   }
537 
538   return (state);
539 }
540 
541 /*!
542   \brief Store data for concealment techniques applied later
543 
544   Interface function to store data for different concealment strategies
545  */
CConcealment_Store(CConcealmentInfo * hConcealmentInfo,CAacDecoderChannelInfo * pAacDecoderChannelInfo,CAacDecoderStaticChannelInfo * pAacDecoderStaticChannelInfo)546 void CConcealment_Store(
547     CConcealmentInfo *hConcealmentInfo,
548     CAacDecoderChannelInfo *pAacDecoderChannelInfo,
549     CAacDecoderStaticChannelInfo *pAacDecoderStaticChannelInfo) {
550   UCHAR nbDiv = NB_DIV;
551 
552   if (!(pAacDecoderChannelInfo->renderMode == AACDEC_RENDER_LPD &&
553         pAacDecoderChannelInfo->data.usac.mod[nbDiv - 1] == 0))
554 
555   {
556     FIXP_DBL *pSpectralCoefficient =
557         SPEC_LONG(pAacDecoderChannelInfo->pSpectralCoefficient);
558     SHORT *pSpecScale = pAacDecoderChannelInfo->specScale;
559     CIcsInfo *pIcsInfo = &pAacDecoderChannelInfo->icsInfo;
560 
561     SHORT tSpecScale[8];
562     UCHAR tWindowShape;
563     BLOCK_TYPE tWindowSequence;
564 
565     /* store old window infos for swapping */
566     tWindowSequence = hConcealmentInfo->windowSequence;
567     tWindowShape = hConcealmentInfo->windowShape;
568 
569     /* store old scale factors for swapping */
570     FDKmemcpy(tSpecScale, hConcealmentInfo->specScale, 8 * sizeof(SHORT));
571 
572     /* store new window infos */
573     hConcealmentInfo->windowSequence = GetWindowSequence(pIcsInfo);
574     hConcealmentInfo->windowShape = GetWindowShape(pIcsInfo);
575     hConcealmentInfo->lastWinGrpLen =
576         *(GetWindowGroupLengthTable(pIcsInfo) + GetWindowGroups(pIcsInfo) - 1);
577 
578     /* store new scale factors */
579     FDKmemcpy(hConcealmentInfo->specScale, pSpecScale, 8 * sizeof(SHORT));
580 
581     if (hConcealmentInfo->pConcealParams->method < ConcealMethodInter) {
582     /* store new spectral bins */
583 #if (CNCL_FRACT_BITS == DFRACT_BITS)
584       FDKmemcpy(hConcealmentInfo->spectralCoefficient, pSpectralCoefficient,
585                 1024 * sizeof(FIXP_CNCL));
586 #else
587       FIXP_CNCL *RESTRICT pCncl =
588           &hConcealmentInfo->spectralCoefficient[1024 - 1];
589       FIXP_DBL *RESTRICT pSpec = &pSpectralCoefficient[1024 - 1];
590       int i;
591       for (i = 1024; i != 0; i--) {
592         *pCncl-- = FX_DBL2FX_CNCL(*pSpec--);
593       }
594 #endif
595     } else {
596     /* swap spectral data */
597 #if (FIXP_CNCL == FIXP_DBL)
598       C_ALLOC_SCRATCH_START(pSpecTmp, FIXP_DBL, 1024);
599       FDKmemcpy(pSpecTmp, pSpectralCoefficient, 1024 * sizeof(FIXP_DBL));
600       FDKmemcpy(pSpectralCoefficient, hConcealmentInfo->spectralCoefficient,
601                 1024 * sizeof(FIXP_DBL));
602       FDKmemcpy(hConcealmentInfo->spectralCoefficient, pSpecTmp,
603                 1024 * sizeof(FIXP_DBL));
604       C_ALLOC_SCRATCH_END(pSpecTmp, FIXP_DBL, 1024);
605 #else
606       FIXP_CNCL *RESTRICT pCncl =
607           &hConcealmentInfo->spectralCoefficient[1024 - 1];
608       FIXP_DBL *RESTRICT pSpec = &pSpectralCoefficient[1024 - 1];
609       FIXP_DBL tSpec;
610 
611       for (int i = 1024; i != 0; i--) {
612         tSpec = *pSpec;
613         *pSpec-- = FX_CNCL2FX_DBL(*pCncl);
614         *pCncl-- = FX_DBL2FX_CNCL(tSpec);
615       }
616 #endif
617 
618       /* complete swapping of window infos */
619       pIcsInfo->WindowSequence = tWindowSequence;
620       pIcsInfo->WindowShape = tWindowShape;
621 
622       /* complete swapping of scale factors */
623       FDKmemcpy(pSpecScale, tSpecScale, 8 * sizeof(SHORT));
624     }
625   }
626 
627   if (pAacDecoderChannelInfo->renderMode == AACDEC_RENDER_LPD) {
628     /* Store LSF4 */
629     FDKmemcpy(hConcealmentInfo->lsf4, pAacDecoderStaticChannelInfo->lpc4_lsf,
630               sizeof(hConcealmentInfo->lsf4));
631     /* Store TCX gain */
632     hConcealmentInfo->last_tcx_gain =
633         pAacDecoderStaticChannelInfo->last_tcx_gain;
634     hConcealmentInfo->last_tcx_gain_e =
635         pAacDecoderStaticChannelInfo->last_tcx_gain_e;
636   }
637 }
638 
639 /*!
640   \brief Apply concealment
641 
642   Interface function to different concealment strategies
643  */
CConcealment_Apply(CConcealmentInfo * hConcealmentInfo,CAacDecoderChannelInfo * pAacDecoderChannelInfo,CAacDecoderStaticChannelInfo * pAacDecoderStaticChannelInfo,const SamplingRateInfo * pSamplingRateInfo,const int samplesPerFrame,const UCHAR lastLpdMode,const int frameOk,const UINT flags)644 int CConcealment_Apply(
645     CConcealmentInfo *hConcealmentInfo,
646     CAacDecoderChannelInfo *pAacDecoderChannelInfo,
647     CAacDecoderStaticChannelInfo *pAacDecoderStaticChannelInfo,
648     const SamplingRateInfo *pSamplingRateInfo, const int samplesPerFrame,
649     const UCHAR lastLpdMode, const int frameOk, const UINT flags) {
650   int appliedProcessing = 0;
651   const int mute_release_active =
652       frameOk && (hConcealmentInfo->concealState >= ConcealState_Mute) &&
653       (hConcealmentInfo->cntValidFrames + 1 <=
654        hConcealmentInfo->pConcealParams->numMuteReleaseFrames);
655 
656   if (hConcealmentInfo->windowShape == CONCEAL_NOT_DEFINED) {
657     /* Initialize window_shape with same value as in the current (parsed) frame.
658        Because section 4.6.11.3.2 (Windowing and block switching) of ISO/IEC
659        14496-3:2009 says: For the first raw_data_block() to be decoded the
660        window_shape of the left and right half of the window are identical. */
661     hConcealmentInfo->windowShape = pAacDecoderChannelInfo->icsInfo.WindowShape;
662   }
663 
664   if (frameOk && !mute_release_active) {
665     /* Update render mode if frameOk except for ongoing mute release state. */
666     hConcealmentInfo->lastRenderMode =
667         (SCHAR)pAacDecoderChannelInfo->renderMode;
668 
669     /* Rescue current data for concealment in future frames */
670     CConcealment_Store(hConcealmentInfo, pAacDecoderChannelInfo,
671                        pAacDecoderStaticChannelInfo);
672     /* Reset index to random sign vector to make sign calculation frame agnostic
673        (only depends on number of subsequently concealed spectral blocks) */
674     hConcealmentInfo->iRandomPhase = 0;
675   } else {
676     if (hConcealmentInfo->lastRenderMode == AACDEC_RENDER_INVALID) {
677       hConcealmentInfo->lastRenderMode = AACDEC_RENDER_IMDCT;
678     }
679     pAacDecoderChannelInfo->renderMode =
680         (AACDEC_RENDER_MODE)hConcealmentInfo->lastRenderMode;
681   }
682 
683   /* hand current frame status to the state machine */
684   CConcealment_UpdateState(hConcealmentInfo, frameOk,
685                            pAacDecoderStaticChannelInfo, samplesPerFrame,
686                            pAacDecoderChannelInfo);
687 
688   {
689     if (!frameOk && pAacDecoderChannelInfo->renderMode == AACDEC_RENDER_IMDCT) {
690       /* LPC extrapolation */
691       CLpc_Conceal(pAacDecoderChannelInfo->data.usac.lsp_coeff,
692                    pAacDecoderStaticChannelInfo->lpc4_lsf,
693                    pAacDecoderStaticChannelInfo->lsf_adaptive_mean,
694                    hConcealmentInfo->lastRenderMode == AACDEC_RENDER_IMDCT);
695       FDKmemcpy(hConcealmentInfo->lsf4, pAacDecoderStaticChannelInfo->lpc4_lsf,
696                 sizeof(pAacDecoderStaticChannelInfo->lpc4_lsf));
697     }
698 
699     /* Create data for signal rendering according to the selected concealment
700      * method and decoder operating mode. */
701 
702     if ((!frameOk || mute_release_active) &&
703         (pAacDecoderChannelInfo->renderMode == AACDEC_RENDER_LPD)) {
704       /* Restore old LSF4 */
705       FDKmemcpy(pAacDecoderStaticChannelInfo->lpc4_lsf, hConcealmentInfo->lsf4,
706                 sizeof(pAacDecoderStaticChannelInfo->lpc4_lsf));
707       /* Restore old TCX gain */
708       pAacDecoderStaticChannelInfo->last_tcx_gain =
709           hConcealmentInfo->last_tcx_gain;
710       pAacDecoderStaticChannelInfo->last_tcx_gain_e =
711           hConcealmentInfo->last_tcx_gain_e;
712     }
713 
714     if (!(pAacDecoderChannelInfo->renderMode == AACDEC_RENDER_LPD &&
715           pAacDecoderStaticChannelInfo->last_lpd_mode == 0)) {
716       switch (hConcealmentInfo->pConcealParams->method) {
717         default:
718         case ConcealMethodMute:
719           if (!frameOk) {
720             /* Mute spectral data in case of errors */
721             FDKmemclear(pAacDecoderChannelInfo->pSpectralCoefficient,
722                         samplesPerFrame * sizeof(FIXP_DBL));
723             /* Set last window shape */
724             pAacDecoderChannelInfo->icsInfo.WindowShape =
725                 hConcealmentInfo->windowShape;
726             appliedProcessing = 1;
727           }
728           break;
729 
730         case ConcealMethodNoise:
731           /* Noise substitution error concealment technique */
732           appliedProcessing = CConcealment_ApplyNoise(
733               hConcealmentInfo, pAacDecoderChannelInfo,
734               pAacDecoderStaticChannelInfo, pSamplingRateInfo, samplesPerFrame,
735               flags);
736           break;
737 
738         case ConcealMethodInter:
739           /* Energy interpolation concealment based on 3GPP */
740           appliedProcessing = CConcealment_ApplyInter(
741               hConcealmentInfo, pAacDecoderChannelInfo, pSamplingRateInfo,
742               samplesPerFrame, 0, /* don't use tonal improvement */
743               frameOk, mute_release_active);
744           break;
745       }
746     } else if (!frameOk || mute_release_active) {
747       /* simply restore the buffer */
748       FIXP_DBL *pSpectralCoefficient =
749           SPEC_LONG(pAacDecoderChannelInfo->pSpectralCoefficient);
750       SHORT *pSpecScale = pAacDecoderChannelInfo->specScale;
751       CIcsInfo *pIcsInfo = &pAacDecoderChannelInfo->icsInfo;
752 #if (CNCL_FRACT_BITS != DFRACT_BITS)
753       FIXP_CNCL *RESTRICT pCncl =
754           &hConcealmentInfo->spectralCoefficient[1024 - 1];
755       FIXP_DBL *RESTRICT pSpec = &pSpectralCoefficient[1024 - 1];
756       int i;
757 #endif
758 
759       /* restore window infos (gri) do we need that? */
760       pIcsInfo->WindowSequence = hConcealmentInfo->windowSequence;
761       pIcsInfo->WindowShape = hConcealmentInfo->windowShape;
762 
763       if (hConcealmentInfo->concealState != ConcealState_Mute) {
764         /* restore scale factors */
765         FDKmemcpy(pSpecScale, hConcealmentInfo->specScale, 8 * sizeof(SHORT));
766 
767         /* restore spectral bins */
768 #if (CNCL_FRACT_BITS == DFRACT_BITS)
769         FDKmemcpy(pSpectralCoefficient, hConcealmentInfo->spectralCoefficient,
770                   1024 * sizeof(FIXP_DBL));
771 #else
772         for (i = 1024; i != 0; i--) {
773           *pSpec-- = FX_CNCL2FX_DBL(*pCncl--);
774         }
775 #endif
776       } else {
777         /* clear scale factors */
778         FDKmemclear(pSpecScale, 8 * sizeof(SHORT));
779 
780         /* clear buffer */
781         FDKmemclear(pSpectralCoefficient, 1024 * sizeof(FIXP_CNCL));
782       }
783     }
784   }
785   /* update history */
786   hConcealmentInfo->prevFrameOk[0] = hConcealmentInfo->prevFrameOk[1];
787   hConcealmentInfo->prevFrameOk[1] = frameOk;
788 
789   return mute_release_active ? -1 : appliedProcessing;
790 }
791 
792 /*!
793 \brief Apply concealment noise substitution
794 
795   In case of frame lost this function produces a noisy frame with respect to the
796   energies values of past frame.
797  */
CConcealment_ApplyNoise(CConcealmentInfo * pConcealmentInfo,CAacDecoderChannelInfo * pAacDecoderChannelInfo,CAacDecoderStaticChannelInfo * pAacDecoderStaticChannelInfo,const SamplingRateInfo * pSamplingRateInfo,const int samplesPerFrame,const UINT flags)798 static int CConcealment_ApplyNoise(
799     CConcealmentInfo *pConcealmentInfo,
800     CAacDecoderChannelInfo *pAacDecoderChannelInfo,
801     CAacDecoderStaticChannelInfo *pAacDecoderStaticChannelInfo,
802     const SamplingRateInfo *pSamplingRateInfo, const int samplesPerFrame,
803     const UINT flags) {
804   FIXP_DBL *pSpectralCoefficient =
805       SPEC_LONG(pAacDecoderChannelInfo->pSpectralCoefficient);
806   CIcsInfo *pIcsInfo = &pAacDecoderChannelInfo->icsInfo;
807 
808   int appliedProcessing = 0;
809 
810   FDK_ASSERT(pConcealmentInfo != NULL);
811   FDK_ASSERT((samplesPerFrame >= 120) && (samplesPerFrame <= 1024));
812 
813   switch (pConcealmentInfo->concealState) {
814     case ConcealState_Ok:
815       /* Nothing to do here! */
816       break;
817 
818     case ConcealState_Single:
819     case ConcealState_FadeOut:
820       appliedProcessing = CConcealment_ApplyFadeOut(
821           /*mode =*/1, pConcealmentInfo, pAacDecoderStaticChannelInfo,
822           samplesPerFrame, pAacDecoderChannelInfo);
823       break;
824 
825     case ConcealState_Mute: {
826       /* set dummy window parameters */
827       pIcsInfo->Valid = 0; /* Trigger the generation of a consitent IcsInfo */
828       pIcsInfo->WindowShape =
829           pConcealmentInfo->windowShape; /* Prevent an invalid WindowShape
830                                             (required for F/T transform) */
831       pIcsInfo->WindowSequence =
832           CConcealment_GetWinSeq(pConcealmentInfo->windowSequence);
833       pConcealmentInfo->windowSequence =
834           pIcsInfo->WindowSequence; /* Store for next frame
835                                        (spectrum in concealment
836                                        buffer can't be used at
837                                        all) */
838 
839       /* mute spectral data */
840       FDKmemclear(pSpectralCoefficient, samplesPerFrame * sizeof(FIXP_DBL));
841       FDKmemclear(pConcealmentInfo->spectralCoefficient,
842                   samplesPerFrame * sizeof(FIXP_DBL));
843 
844       appliedProcessing = 1;
845     } break;
846 
847     case ConcealState_FadeIn: {
848       /* TimeDomainFading:                                        */
849       /* Attenuation of signal is done in CConcealment_TDFading() */
850 
851       appliedProcessing = 1;
852     } break;
853 
854     default:
855       /* we shouldn't come here anyway */
856       FDK_ASSERT(0);
857       break;
858   }
859 
860   return appliedProcessing;
861 }
862 
863 /*!
864   \brief Apply concealment interpolation
865 
866   The function swaps the data from the current and the previous frame. If an
867   error has occured, frame interpolation is performed to restore the missing
868   frame. In case of multiple faulty frames, fade-in and fade-out is applied.
869 */
CConcealment_ApplyInter(CConcealmentInfo * pConcealmentInfo,CAacDecoderChannelInfo * pAacDecoderChannelInfo,const SamplingRateInfo * pSamplingRateInfo,const int samplesPerFrame,const int improveTonal,const int frameOk,const int mute_release_active)870 static int CConcealment_ApplyInter(
871     CConcealmentInfo *pConcealmentInfo,
872     CAacDecoderChannelInfo *pAacDecoderChannelInfo,
873     const SamplingRateInfo *pSamplingRateInfo, const int samplesPerFrame,
874     const int improveTonal, const int frameOk, const int mute_release_active) {
875 #if defined(FDK_ASSERT_ENABLE)
876   CConcealParams *pConcealCommonData = pConcealmentInfo->pConcealParams;
877 #endif
878 
879   FIXP_DBL *pSpectralCoefficient =
880       SPEC_LONG(pAacDecoderChannelInfo->pSpectralCoefficient);
881   CIcsInfo *pIcsInfo = &pAacDecoderChannelInfo->icsInfo;
882   SHORT *pSpecScale = pAacDecoderChannelInfo->specScale;
883 
884   int sfbEnergyPrev[64];
885   int sfbEnergyAct[64];
886 
887   int i, appliedProcessing = 0;
888 
889   /* clear/init */
890   FDKmemclear(sfbEnergyPrev, 64 * sizeof(int));
891   FDKmemclear(sfbEnergyAct, 64 * sizeof(int));
892 
893   if (!frameOk || mute_release_active) {
894     /* Restore last frame from concealment buffer */
895     pIcsInfo->WindowShape = pConcealmentInfo->windowShape;
896     pIcsInfo->WindowSequence = pConcealmentInfo->windowSequence;
897 
898     /* Restore spectral data */
899     for (i = 0; i < samplesPerFrame; i++) {
900       pSpectralCoefficient[i] =
901           FX_CNCL2FX_DBL(pConcealmentInfo->spectralCoefficient[i]);
902     }
903 
904     /* Restore scale factors */
905     FDKmemcpy(pSpecScale, pConcealmentInfo->specScale, 8 * sizeof(SHORT));
906   }
907 
908   /* if previous frame was not ok */
909   if (!pConcealmentInfo->prevFrameOk[1] || mute_release_active) {
910     /* if current frame (f_n) is ok and the last but one frame (f_(n-2))
911        was ok, too, then interpolate both frames in order to generate
912        the current output frame (f_(n-1)). Otherwise, use the last stored
913        frame (f_(n-2) or f_(n-3) or ...). */
914     if (frameOk && pConcealmentInfo->prevFrameOk[0] && !mute_release_active) {
915       appliedProcessing = 1;
916 
917       /* Interpolate both frames in order to generate the current output frame
918        * (f_(n-1)). */
919       if (pIcsInfo->WindowSequence == BLOCK_SHORT) {
920         /* f_(n-2) == BLOCK_SHORT */
921         /* short--??????--short, short--??????--long interpolation */
922         /* short--short---short, short---long---long interpolation */
923 
924         int wnd;
925 
926         if (pConcealmentInfo->windowSequence ==
927             BLOCK_SHORT) { /* f_n == BLOCK_SHORT */
928           /* short--short---short interpolation */
929 
930           int scaleFactorBandsTotal =
931               pSamplingRateInfo->NumberOfScaleFactorBands_Short;
932           const SHORT *pSfbOffset = pSamplingRateInfo->ScaleFactorBands_Short;
933           pIcsInfo->WindowShape = (samplesPerFrame <= 512) ? 2 : 1;
934           pIcsInfo->WindowSequence = BLOCK_SHORT;
935 
936           for (wnd = 0; wnd < 8; wnd++) {
937             CConcealment_CalcBandEnergy(
938                 &pSpectralCoefficient[wnd *
939                                       (samplesPerFrame / 8)], /* spec_(n-2) */
940                 pSamplingRateInfo, BLOCK_SHORT, CConcealment_NoExpand,
941                 sfbEnergyPrev);
942 
943             CConcealment_CalcBandEnergy(
944                 &pConcealmentInfo->spectralCoefficient[wnd * (samplesPerFrame /
945                                                               8)], /* spec_n */
946                 pSamplingRateInfo, BLOCK_SHORT, CConcealment_NoExpand,
947                 sfbEnergyAct);
948 
949             CConcealment_InterpolateBuffer(
950                 &pSpectralCoefficient[wnd *
951                                       (samplesPerFrame / 8)], /* spec_(n-1) */
952                 &pSpecScale[wnd], &pConcealmentInfo->specScale[wnd],
953                 &pSpecScale[wnd], sfbEnergyPrev, sfbEnergyAct,
954                 scaleFactorBandsTotal, pSfbOffset);
955           }
956         } else { /* f_n != BLOCK_SHORT */
957           /* short---long---long interpolation */
958 
959           int scaleFactorBandsTotal =
960               pSamplingRateInfo->NumberOfScaleFactorBands_Long;
961           const SHORT *pSfbOffset = pSamplingRateInfo->ScaleFactorBands_Long;
962           SHORT specScaleOut;
963 
964           CConcealment_CalcBandEnergy(
965               &pSpectralCoefficient[samplesPerFrame -
966                                     (samplesPerFrame /
967                                      8)], /* [wnd] spec_(n-2) */
968               pSamplingRateInfo, BLOCK_SHORT, CConcealment_Expand,
969               sfbEnergyAct);
970 
971           CConcealment_CalcBandEnergy(
972               pConcealmentInfo->spectralCoefficient, /* spec_n */
973               pSamplingRateInfo, BLOCK_LONG, CConcealment_NoExpand,
974               sfbEnergyPrev);
975 
976           pIcsInfo->WindowShape = 0;
977           pIcsInfo->WindowSequence = BLOCK_STOP;
978 
979           for (i = 0; i < samplesPerFrame; i++) {
980             pSpectralCoefficient[i] =
981                 pConcealmentInfo->spectralCoefficient[i]; /* spec_n */
982           }
983 
984           for (i = 0; i < 8; i++) { /* search for max(specScale) */
985             if (pSpecScale[i] > pSpecScale[0]) {
986               pSpecScale[0] = pSpecScale[i];
987             }
988           }
989 
990           CConcealment_InterpolateBuffer(
991               pSpectralCoefficient, /* spec_(n-1) */
992               &pConcealmentInfo->specScale[0], &pSpecScale[0], &specScaleOut,
993               sfbEnergyPrev, sfbEnergyAct, scaleFactorBandsTotal, pSfbOffset);
994 
995           pSpecScale[0] = specScaleOut;
996         }
997       } else {
998         /* long--??????--short, long--??????--long interpolation */
999         /* long---long---short, long---long---long interpolation */
1000 
1001         int scaleFactorBandsTotal =
1002             pSamplingRateInfo->NumberOfScaleFactorBands_Long;
1003         const SHORT *pSfbOffset = pSamplingRateInfo->ScaleFactorBands_Long;
1004         SHORT specScaleAct = pConcealmentInfo->specScale[0];
1005 
1006         CConcealment_CalcBandEnergy(pSpectralCoefficient, /* spec_(n-2) */
1007                                     pSamplingRateInfo, BLOCK_LONG,
1008                                     CConcealment_NoExpand, sfbEnergyPrev);
1009 
1010         if (pConcealmentInfo->windowSequence ==
1011             BLOCK_SHORT) { /* f_n == BLOCK_SHORT */
1012           /* long---long---short interpolation */
1013 
1014           pIcsInfo->WindowShape = (samplesPerFrame <= 512) ? 2 : 1;
1015           pIcsInfo->WindowSequence = BLOCK_START;
1016 
1017           for (i = 1; i < 8; i++) { /* search for max(specScale) */
1018             if (pConcealmentInfo->specScale[i] > specScaleAct) {
1019               specScaleAct = pConcealmentInfo->specScale[i];
1020             }
1021           }
1022 
1023           /* Expand first short spectrum */
1024           CConcealment_CalcBandEnergy(
1025               pConcealmentInfo->spectralCoefficient,               /* spec_n */
1026               pSamplingRateInfo, BLOCK_SHORT, CConcealment_Expand, /* !!! */
1027               sfbEnergyAct);
1028         } else {
1029           /* long---long---long interpolation */
1030 
1031           pIcsInfo->WindowShape = 0;
1032           pIcsInfo->WindowSequence = BLOCK_LONG;
1033 
1034           CConcealment_CalcBandEnergy(
1035               pConcealmentInfo->spectralCoefficient, /* spec_n */
1036               pSamplingRateInfo, BLOCK_LONG, CConcealment_NoExpand,
1037               sfbEnergyAct);
1038         }
1039 
1040         CConcealment_InterpolateBuffer(
1041             pSpectralCoefficient, /* spec_(n-1) */
1042             &pSpecScale[0], &specScaleAct, &pSpecScale[0], sfbEnergyPrev,
1043             sfbEnergyAct, scaleFactorBandsTotal, pSfbOffset);
1044       }
1045     }
1046 
1047     /* Noise substitution of sign of the output spectral coefficients */
1048     CConcealment_ApplyRandomSign(pConcealmentInfo->iRandomPhase,
1049                                  pSpectralCoefficient, samplesPerFrame);
1050     /* Increment random phase index to avoid repetition artifacts. */
1051     pConcealmentInfo->iRandomPhase =
1052         (pConcealmentInfo->iRandomPhase + 1) & (AAC_NF_NO_RANDOM_VAL - 1);
1053   }
1054 
1055   /* scale spectrum according to concealment state */
1056   switch (pConcealmentInfo->concealState) {
1057     case ConcealState_Single:
1058       appliedProcessing = 1;
1059       break;
1060 
1061     case ConcealState_FadeOut: {
1062       FDK_ASSERT(pConcealmentInfo->cntFadeFrames >= 0);
1063       FDK_ASSERT(pConcealmentInfo->cntFadeFrames <
1064                  CONCEAL_MAX_NUM_FADE_FACTORS);
1065       FDK_ASSERT(pConcealmentInfo->cntFadeFrames <
1066                  pConcealCommonData->numFadeOutFrames);
1067 
1068       /* TimeDomainFading:                                        */
1069       /* Attenuation of signal is done in CConcealment_TDFading() */
1070 
1071       appliedProcessing = 1;
1072     } break;
1073 
1074     case ConcealState_FadeIn: {
1075       FDK_ASSERT(pConcealmentInfo->cntFadeFrames >= 0);
1076       FDK_ASSERT(pConcealmentInfo->cntFadeFrames <
1077                  CONCEAL_MAX_NUM_FADE_FACTORS);
1078       FDK_ASSERT(pConcealmentInfo->cntFadeFrames <
1079                  pConcealCommonData->numFadeInFrames);
1080 
1081       /* TimeDomainFading:                                        */
1082       /* Attenuation of signal is done in CConcealment_TDFading() */
1083 
1084       appliedProcessing = 1;
1085     } break;
1086 
1087     case ConcealState_Mute: {
1088       /* set dummy window parameters */
1089       pIcsInfo->Valid = 0; /* Trigger the generation of a consitent IcsInfo */
1090       pIcsInfo->WindowShape =
1091           pConcealmentInfo->windowShape; /* Prevent an invalid WindowShape
1092                                             (required for F/T transform) */
1093       pIcsInfo->WindowSequence =
1094           CConcealment_GetWinSeq(pConcealmentInfo->windowSequence);
1095       pConcealmentInfo->windowSequence =
1096           pIcsInfo->WindowSequence; /* Store for next frame
1097                                        (spectrum in concealment
1098                                        buffer can't be used at
1099                                        all) */
1100 
1101       /* mute spectral data */
1102       FDKmemclear(pSpectralCoefficient, samplesPerFrame * sizeof(FIXP_DBL));
1103 
1104       appliedProcessing = 1;
1105     } break;
1106 
1107     default:
1108       /* nothing to do here */
1109       break;
1110   }
1111 
1112   return appliedProcessing;
1113 }
1114 
1115 /*!
1116   \brief Calculate the spectral energy
1117 
1118   The function calculates band-wise the spectral energy. This is used for
1119   frame interpolation.
1120 */
CConcealment_CalcBandEnergy(FIXP_DBL * spectrum,const SamplingRateInfo * pSamplingRateInfo,const int blockType,CConcealmentExpandType expandType,int * sfbEnergy)1121 static void CConcealment_CalcBandEnergy(
1122     FIXP_DBL *spectrum, const SamplingRateInfo *pSamplingRateInfo,
1123     const int blockType, CConcealmentExpandType expandType, int *sfbEnergy) {
1124   const SHORT *pSfbOffset;
1125   int line, sfb, scaleFactorBandsTotal = 0;
1126 
1127   /* In the following calculations, enAccu is initialized with LSB-value in
1128    * order to avoid zero energy-level */
1129 
1130   line = 0;
1131 
1132   switch (blockType) {
1133     case BLOCK_LONG:
1134     case BLOCK_START:
1135     case BLOCK_STOP:
1136 
1137       if (expandType == CConcealment_NoExpand) {
1138         /* standard long calculation */
1139         scaleFactorBandsTotal =
1140             pSamplingRateInfo->NumberOfScaleFactorBands_Long;
1141         pSfbOffset = pSamplingRateInfo->ScaleFactorBands_Long;
1142 
1143         for (sfb = 0; sfb < scaleFactorBandsTotal; sfb++) {
1144           FIXP_DBL enAccu = (FIXP_DBL)(LONG)1;
1145           int sfbScale =
1146               (sizeof(LONG) << 3) -
1147               CntLeadingZeros(pSfbOffset[sfb + 1] - pSfbOffset[sfb]) - 1;
1148           /* scaling depends on sfb width. */
1149           for (; line < pSfbOffset[sfb + 1]; line++) {
1150             enAccu += fPow2Div2(*(spectrum + line)) >> sfbScale;
1151           }
1152           *(sfbEnergy + sfb) = CntLeadingZeros(enAccu) - 1;
1153         }
1154       } else {
1155         /* compress long to short */
1156         scaleFactorBandsTotal =
1157             pSamplingRateInfo->NumberOfScaleFactorBands_Short;
1158         pSfbOffset = pSamplingRateInfo->ScaleFactorBands_Short;
1159 
1160         for (sfb = 0; sfb < scaleFactorBandsTotal; sfb++) {
1161           FIXP_DBL enAccu = (FIXP_DBL)(LONG)1;
1162           int sfbScale =
1163               (sizeof(LONG) << 3) -
1164               CntLeadingZeros(pSfbOffset[sfb + 1] - pSfbOffset[sfb]) - 1;
1165           /* scaling depends on sfb width. */
1166           for (; line < pSfbOffset[sfb + 1] << 3; line++) {
1167             enAccu +=
1168                 (enAccu + (fPow2Div2(*(spectrum + line)) >> sfbScale)) >> 3;
1169           }
1170           *(sfbEnergy + sfb) = CntLeadingZeros(enAccu) - 1;
1171         }
1172       }
1173       break;
1174 
1175     case BLOCK_SHORT:
1176 
1177       if (expandType == CConcealment_NoExpand) {
1178         /*   standard short calculation */
1179         scaleFactorBandsTotal =
1180             pSamplingRateInfo->NumberOfScaleFactorBands_Short;
1181         pSfbOffset = pSamplingRateInfo->ScaleFactorBands_Short;
1182 
1183         for (sfb = 0; sfb < scaleFactorBandsTotal; sfb++) {
1184           FIXP_DBL enAccu = (FIXP_DBL)(LONG)1;
1185           int sfbScale =
1186               (sizeof(LONG) << 3) -
1187               CntLeadingZeros(pSfbOffset[sfb + 1] - pSfbOffset[sfb]) - 1;
1188           /* scaling depends on sfb width. */
1189           for (; line < pSfbOffset[sfb + 1]; line++) {
1190             enAccu += fPow2Div2(*(spectrum + line)) >> sfbScale;
1191           }
1192           *(sfbEnergy + sfb) = CntLeadingZeros(enAccu) - 1;
1193         }
1194       } else {
1195         /*  expand short to long spectrum */
1196         scaleFactorBandsTotal =
1197             pSamplingRateInfo->NumberOfScaleFactorBands_Long;
1198         pSfbOffset = pSamplingRateInfo->ScaleFactorBands_Long;
1199 
1200         for (sfb = 0; sfb < scaleFactorBandsTotal; sfb++) {
1201           FIXP_DBL enAccu = (FIXP_DBL)(LONG)1;
1202           int sfbScale =
1203               (sizeof(LONG) << 3) -
1204               CntLeadingZeros(pSfbOffset[sfb + 1] - pSfbOffset[sfb]) - 1;
1205           /* scaling depends on sfb width. */
1206           for (; line < pSfbOffset[sfb + 1]; line++) {
1207             enAccu += fPow2Div2(*(spectrum + (line >> 3))) >> sfbScale;
1208           }
1209           *(sfbEnergy + sfb) = CntLeadingZeros(enAccu) - 1;
1210         }
1211       }
1212       break;
1213   }
1214 }
1215 
1216 /*!
1217   \brief Interpolate buffer
1218 
1219   The function creates the interpolated spectral data according to the
1220   energy of the last good frame and the current (good) frame.
1221 */
CConcealment_InterpolateBuffer(FIXP_DBL * spectrum,SHORT * pSpecScalePrv,SHORT * pSpecScaleAct,SHORT * pSpecScaleOut,int * enPrv,int * enAct,int sfbCnt,const SHORT * pSfbOffset)1222 static void CConcealment_InterpolateBuffer(FIXP_DBL *spectrum,
1223                                            SHORT *pSpecScalePrv,
1224                                            SHORT *pSpecScaleAct,
1225                                            SHORT *pSpecScaleOut, int *enPrv,
1226                                            int *enAct, int sfbCnt,
1227                                            const SHORT *pSfbOffset) {
1228   int sfb, line = 0;
1229   int fac_shift;
1230   int fac_mod;
1231   FIXP_DBL accu;
1232 
1233   for (sfb = 0; sfb < sfbCnt; sfb++) {
1234     fac_shift =
1235         enPrv[sfb] - enAct[sfb] + ((*pSpecScaleAct - *pSpecScalePrv) << 1);
1236     fac_mod = fac_shift & 3;
1237     fac_shift = (fac_shift >> 2) + 1;
1238     fac_shift += *pSpecScalePrv - fixMax(*pSpecScalePrv, *pSpecScaleAct);
1239 
1240     for (; line < pSfbOffset[sfb + 1]; line++) {
1241       accu = fMult(*(spectrum + line), facMod4Table[fac_mod]);
1242       if (fac_shift < 0) {
1243         accu >>= -fac_shift;
1244       } else {
1245         accu <<= fac_shift;
1246       }
1247       *(spectrum + line) = accu;
1248     }
1249   }
1250   *pSpecScaleOut = fixMax(*pSpecScalePrv, *pSpecScaleAct);
1251 }
1252 
1253 /*!
1254   \brief Find next fading frame in case of changing fading direction
1255 
1256   \param pConcealCommonData Pointer to the concealment common data structure.
1257   \param actFadeIndex Last index used for fading
1258   \param direction Direction of change: 0 : change from FADE-OUT to FADE-IN,  1
1259   : change from FADE-IN to FADE-OUT
1260 
1261   This function determines the next fading index to be used for the fading
1262   direction to be changed to.
1263 */
1264 
findEquiFadeFrame(CConcealParams * pConcealCommonData,INT actFadeIndex,int direction)1265 static INT findEquiFadeFrame(CConcealParams *pConcealCommonData,
1266                              INT actFadeIndex, int direction) {
1267   FIXP_SGL *pFactor;
1268   FIXP_SGL referenceVal;
1269   FIXP_SGL minDiff = (FIXP_SGL)MAXVAL_SGL;
1270 
1271   INT nextFadeIndex = 0;
1272 
1273   int i;
1274 
1275   /* init depending on direction */
1276   if (direction == 0) { /* FADE-OUT => FADE-IN */
1277     if (actFadeIndex < 0) {
1278       referenceVal = (FIXP_SGL)MAXVAL_SGL;
1279     } else {
1280       referenceVal = pConcealCommonData->fadeOutFactor[actFadeIndex] >> 1;
1281     }
1282     pFactor = pConcealCommonData->fadeInFactor;
1283   } else { /* FADE-IN => FADE-OUT */
1284     if (actFadeIndex < 0) {
1285       referenceVal = (FIXP_SGL)MAXVAL_SGL;
1286     } else {
1287       referenceVal = pConcealCommonData->fadeInFactor[actFadeIndex] >> 1;
1288     }
1289     pFactor = pConcealCommonData->fadeOutFactor;
1290   }
1291 
1292   /* search for minimum difference */
1293   for (i = 0; i < CONCEAL_MAX_NUM_FADE_FACTORS; i++) {
1294     FIXP_SGL diff = fixp_abs((pFactor[i] >> 1) - referenceVal);
1295     if (diff < minDiff) {
1296       minDiff = diff;
1297       nextFadeIndex = i;
1298     }
1299   }
1300 
1301   /* check and adjust depending on direction */
1302   if (direction == 0) { /* FADE-OUT => FADE-IN */
1303     if (nextFadeIndex > pConcealCommonData->numFadeInFrames) {
1304       nextFadeIndex = fMax(pConcealCommonData->numFadeInFrames - 1, 0);
1305     }
1306     if (((pFactor[nextFadeIndex] >> 1) <= referenceVal) &&
1307         (nextFadeIndex > 0)) {
1308       nextFadeIndex -= 1;
1309     }
1310   } else { /* FADE-IN => FADE-OUT */
1311     if (((pFactor[nextFadeIndex] >> 1) >= referenceVal) &&
1312         (nextFadeIndex < CONCEAL_MAX_NUM_FADE_FACTORS - 1)) {
1313       nextFadeIndex += 1;
1314     }
1315   }
1316 
1317   return (nextFadeIndex);
1318 }
1319 
1320 /*!
1321   \brief Update the concealment state
1322 
1323   The function updates the state of the concealment state-machine. The
1324   states are: mute, fade-in, fade-out, interpolate and frame-ok.
1325 */
CConcealment_UpdateState(CConcealmentInfo * pConcealmentInfo,int frameOk,CAacDecoderStaticChannelInfo * pAacDecoderStaticChannelInfo,const int samplesPerFrame,CAacDecoderChannelInfo * pAacDecoderChannelInfo)1326 static void CConcealment_UpdateState(
1327     CConcealmentInfo *pConcealmentInfo, int frameOk,
1328     CAacDecoderStaticChannelInfo *pAacDecoderStaticChannelInfo,
1329     const int samplesPerFrame, CAacDecoderChannelInfo *pAacDecoderChannelInfo) {
1330   CConcealParams *pConcealCommonData = pConcealmentInfo->pConcealParams;
1331 
1332   switch (pConcealCommonData->method) {
1333     case ConcealMethodNoise: {
1334       if (pConcealmentInfo->concealState != ConcealState_Ok) {
1335         /* count the valid frames during concealment process */
1336         if (frameOk) {
1337           pConcealmentInfo->cntValidFrames += 1;
1338         } else {
1339           pConcealmentInfo->cntValidFrames = 0;
1340         }
1341       }
1342 
1343       /* -- STATE MACHINE for Noise Substitution -- */
1344       switch (pConcealmentInfo->concealState) {
1345         case ConcealState_Ok:
1346           if (!frameOk) {
1347             pConcealmentInfo->cntFadeFrames = 0;
1348             pConcealmentInfo->cntValidFrames = 0;
1349             pConcealmentInfo->attGrpOffset[0] = 0;
1350             pConcealmentInfo->attGrpOffset[1] = 0;
1351             pConcealmentInfo->winGrpOffset[0] = 0;
1352             pConcealmentInfo->winGrpOffset[1] = 0;
1353             if (pConcealCommonData->numFadeOutFrames > 0) {
1354               /* change to state SINGLE-FRAME-LOSS */
1355               pConcealmentInfo->concealState = ConcealState_Single;
1356               /* mode 0 just updates the Fading counter */
1357               CConcealment_ApplyFadeOut(
1358                   /*mode =*/0, pConcealmentInfo, pAacDecoderStaticChannelInfo,
1359                   samplesPerFrame, pAacDecoderChannelInfo);
1360 
1361             } else {
1362               /* change to state MUTE */
1363               pConcealmentInfo->concealState = ConcealState_Mute;
1364             }
1365           }
1366           break;
1367 
1368         case ConcealState_Single: /* Just a pre-stage before fade-out begins.
1369                                      Stay here only one frame! */
1370           if (frameOk) {
1371             /* change to state OK */
1372             pConcealmentInfo->concealState = ConcealState_Ok;
1373           } else {
1374             if (pConcealmentInfo->cntFadeFrames >=
1375                 pConcealCommonData->numFadeOutFrames) {
1376               /* change to state MUTE */
1377               pConcealmentInfo->concealState = ConcealState_Mute;
1378             } else {
1379               /* change to state FADE-OUT */
1380               pConcealmentInfo->concealState = ConcealState_FadeOut;
1381               /* mode 0 just updates the Fading counter */
1382               CConcealment_ApplyFadeOut(
1383                   /*mode =*/0, pConcealmentInfo, pAacDecoderStaticChannelInfo,
1384                   samplesPerFrame, pAacDecoderChannelInfo);
1385             }
1386           }
1387           break;
1388 
1389         case ConcealState_FadeOut:
1390           if (pConcealmentInfo->cntValidFrames >
1391               pConcealCommonData->numMuteReleaseFrames) {
1392             if (pConcealCommonData->numFadeInFrames > 0) {
1393               /* change to state FADE-IN */
1394               pConcealmentInfo->concealState = ConcealState_FadeIn;
1395               pConcealmentInfo->cntFadeFrames = findEquiFadeFrame(
1396                   pConcealCommonData, pConcealmentInfo->cntFadeFrames,
1397                   0 /* FadeOut -> FadeIn */);
1398             } else {
1399               /* change to state OK */
1400               pConcealmentInfo->concealState = ConcealState_Ok;
1401             }
1402           } else {
1403             if (frameOk) {
1404               /* we have good frame information but stay fully in concealment -
1405                * reset winGrpOffset/attGrpOffset */
1406               pConcealmentInfo->winGrpOffset[0] = 0;
1407               pConcealmentInfo->winGrpOffset[1] = 0;
1408               pConcealmentInfo->attGrpOffset[0] = 0;
1409               pConcealmentInfo->attGrpOffset[1] = 0;
1410             }
1411             if (pConcealmentInfo->cntFadeFrames >=
1412                 pConcealCommonData->numFadeOutFrames) {
1413               /* change to state MUTE */
1414               pConcealmentInfo->concealState = ConcealState_Mute;
1415             } else /* Stay in FADE-OUT */
1416             {
1417               /* mode 0 just updates the Fading counter */
1418               CConcealment_ApplyFadeOut(
1419                   /*mode =*/0, pConcealmentInfo, pAacDecoderStaticChannelInfo,
1420                   samplesPerFrame, pAacDecoderChannelInfo);
1421             }
1422           }
1423           break;
1424 
1425         case ConcealState_Mute:
1426           if (pConcealmentInfo->cntValidFrames >
1427               pConcealCommonData->numMuteReleaseFrames) {
1428             if (pConcealCommonData->numFadeInFrames > 0) {
1429               /* change to state FADE-IN */
1430               pConcealmentInfo->concealState = ConcealState_FadeIn;
1431               pConcealmentInfo->cntFadeFrames =
1432                   pConcealCommonData->numFadeInFrames - 1;
1433             } else {
1434               /* change to state OK */
1435               pConcealmentInfo->concealState = ConcealState_Ok;
1436             }
1437           } else {
1438             if (frameOk) {
1439               /* we have good frame information but stay fully in concealment -
1440                * reset winGrpOffset/attGrpOffset */
1441               pConcealmentInfo->winGrpOffset[0] = 0;
1442               pConcealmentInfo->winGrpOffset[1] = 0;
1443               pConcealmentInfo->attGrpOffset[0] = 0;
1444               pConcealmentInfo->attGrpOffset[1] = 0;
1445             }
1446           }
1447           break;
1448 
1449         case ConcealState_FadeIn:
1450           pConcealmentInfo->cntFadeFrames -= 1;
1451           if (frameOk) {
1452             if (pConcealmentInfo->cntFadeFrames < 0) {
1453               /* change to state OK */
1454               pConcealmentInfo->concealState = ConcealState_Ok;
1455             }
1456           } else {
1457             if (pConcealCommonData->numFadeOutFrames > 0) {
1458               /* change to state FADE-OUT */
1459               pConcealmentInfo->concealState = ConcealState_FadeOut;
1460               pConcealmentInfo->cntFadeFrames = findEquiFadeFrame(
1461                   pConcealCommonData, pConcealmentInfo->cntFadeFrames + 1,
1462                   1 /* FadeIn -> FadeOut */);
1463               pConcealmentInfo->winGrpOffset[0] = 0;
1464               pConcealmentInfo->winGrpOffset[1] = 0;
1465               pConcealmentInfo->attGrpOffset[0] = 0;
1466               pConcealmentInfo->attGrpOffset[1] = 0;
1467 
1468               pConcealmentInfo
1469                   ->cntFadeFrames--; /* decrease because
1470                                         CConcealment_ApplyFadeOut() will
1471                                         increase, accordingly */
1472               /* mode 0 just updates the Fading counter */
1473               CConcealment_ApplyFadeOut(
1474                   /*mode =*/0, pConcealmentInfo, pAacDecoderStaticChannelInfo,
1475                   samplesPerFrame, pAacDecoderChannelInfo);
1476             } else {
1477               /* change to state MUTE */
1478               pConcealmentInfo->concealState = ConcealState_Mute;
1479             }
1480           }
1481           break;
1482 
1483         default:
1484           FDK_ASSERT(0);
1485           break;
1486       }
1487     } break;
1488 
1489     case ConcealMethodInter:
1490     case ConcealMethodTonal: {
1491       if (pConcealmentInfo->concealState != ConcealState_Ok) {
1492         /* count the valid frames during concealment process */
1493         if (pConcealmentInfo->prevFrameOk[1] ||
1494             (pConcealmentInfo->prevFrameOk[0] &&
1495              !pConcealmentInfo->prevFrameOk[1] && frameOk)) {
1496           /* The frame is OK even if it can be estimated by the energy
1497            * interpolation algorithm */
1498           pConcealmentInfo->cntValidFrames += 1;
1499         } else {
1500           pConcealmentInfo->cntValidFrames = 0;
1501         }
1502       }
1503 
1504       /* -- STATE MACHINE for energy interpolation -- */
1505       switch (pConcealmentInfo->concealState) {
1506         case ConcealState_Ok:
1507           if (!(pConcealmentInfo->prevFrameOk[1] ||
1508                 (pConcealmentInfo->prevFrameOk[0] &&
1509                  !pConcealmentInfo->prevFrameOk[1] && frameOk))) {
1510             if (pConcealCommonData->numFadeOutFrames > 0) {
1511               /* Fade out only if the energy interpolation algorithm can not be
1512                * applied! */
1513               pConcealmentInfo->concealState = ConcealState_FadeOut;
1514             } else {
1515               /* change to state MUTE */
1516               pConcealmentInfo->concealState = ConcealState_Mute;
1517             }
1518             pConcealmentInfo->cntFadeFrames = 0;
1519             pConcealmentInfo->cntValidFrames = 0;
1520           }
1521           break;
1522 
1523         case ConcealState_Single:
1524           pConcealmentInfo->concealState = ConcealState_Ok;
1525           break;
1526 
1527         case ConcealState_FadeOut:
1528           pConcealmentInfo->cntFadeFrames += 1;
1529 
1530           if (pConcealmentInfo->cntValidFrames >
1531               pConcealCommonData->numMuteReleaseFrames) {
1532             if (pConcealCommonData->numFadeInFrames > 0) {
1533               /* change to state FADE-IN */
1534               pConcealmentInfo->concealState = ConcealState_FadeIn;
1535               pConcealmentInfo->cntFadeFrames = findEquiFadeFrame(
1536                   pConcealCommonData, pConcealmentInfo->cntFadeFrames - 1,
1537                   0 /* FadeOut -> FadeIn */);
1538             } else {
1539               /* change to state OK */
1540               pConcealmentInfo->concealState = ConcealState_Ok;
1541             }
1542           } else {
1543             if (pConcealmentInfo->cntFadeFrames >=
1544                 pConcealCommonData->numFadeOutFrames) {
1545               /* change to state MUTE */
1546               pConcealmentInfo->concealState = ConcealState_Mute;
1547             }
1548           }
1549           break;
1550 
1551         case ConcealState_Mute:
1552           if (pConcealmentInfo->cntValidFrames >
1553               pConcealCommonData->numMuteReleaseFrames) {
1554             if (pConcealCommonData->numFadeInFrames > 0) {
1555               /* change to state FADE-IN */
1556               pConcealmentInfo->concealState = ConcealState_FadeIn;
1557               pConcealmentInfo->cntFadeFrames =
1558                   pConcealCommonData->numFadeInFrames - 1;
1559             } else {
1560               /* change to state OK */
1561               pConcealmentInfo->concealState = ConcealState_Ok;
1562             }
1563           }
1564           break;
1565 
1566         case ConcealState_FadeIn:
1567           pConcealmentInfo->cntFadeFrames -=
1568               1; /* used to address the fade-in factors */
1569 
1570           if (frameOk || pConcealmentInfo->prevFrameOk[1]) {
1571             if (pConcealmentInfo->cntFadeFrames < 0) {
1572               /* change to state OK */
1573               pConcealmentInfo->concealState = ConcealState_Ok;
1574             }
1575           } else {
1576             if (pConcealCommonData->numFadeOutFrames > 0) {
1577               /* change to state FADE-OUT */
1578               pConcealmentInfo->concealState = ConcealState_FadeOut;
1579               pConcealmentInfo->cntFadeFrames = findEquiFadeFrame(
1580                   pConcealCommonData, pConcealmentInfo->cntFadeFrames + 1,
1581                   1 /* FadeIn -> FadeOut */);
1582             } else {
1583               /* change to state MUTE */
1584               pConcealmentInfo->concealState = ConcealState_Mute;
1585             }
1586           }
1587           break;
1588       } /* End switch(pConcealmentInfo->concealState) */
1589     } break;
1590 
1591     default:
1592       /* Don't need a state machine for other concealment methods. */
1593       break;
1594   }
1595 }
1596 
1597 /*!
1598 \brief Randomizes the sign of the spectral data
1599 
1600   The function toggles the sign of the spectral data randomly. This is
1601   useful to ensure the quality of the concealed frames.
1602  */
CConcealment_ApplyRandomSign(int randomPhase,FIXP_DBL * spec,int samplesPerFrame)1603 static void CConcealment_ApplyRandomSign(int randomPhase, FIXP_DBL *spec,
1604                                          int samplesPerFrame) {
1605   int i;
1606   USHORT packedSign = 0;
1607 
1608   /* random table 512x16bit has been reduced to 512 packed sign bits = 32x16 bit
1609    */
1610 
1611   /* read current packed sign word */
1612   packedSign = AacDec_randomSign[randomPhase >> 4];
1613   packedSign >>= (randomPhase & 0xf);
1614 
1615   for (i = 0; i < samplesPerFrame; i++) {
1616     if ((randomPhase & 0xf) == 0) {
1617       packedSign = AacDec_randomSign[randomPhase >> 4];
1618     }
1619 
1620     if (packedSign & 0x1) {
1621       spec[i] = -spec[i];
1622     }
1623     packedSign >>= 1;
1624 
1625     randomPhase = (randomPhase + 1) & (AAC_NF_NO_RANDOM_VAL - 1);
1626   }
1627 }
1628 
1629 /*!
1630   \brief Get fadeing factor for current concealment state.
1631 
1632   The function returns the state (ok or not) of the previous frame.
1633   If called before the function CConcealment_Apply() set the fBeforeApply
1634   flag to get the correct value.
1635 
1636   \return Frame OK flag of previous frame.
1637  */
CConcealment_GetLastFrameOk(CConcealmentInfo * hConcealmentInfo,const int fBeforeApply)1638 int CConcealment_GetLastFrameOk(CConcealmentInfo *hConcealmentInfo,
1639                                 const int fBeforeApply) {
1640   int prevFrameOk = 1;
1641 
1642   if (hConcealmentInfo != NULL) {
1643     prevFrameOk = hConcealmentInfo->prevFrameOk[fBeforeApply & 0x1];
1644   }
1645 
1646   return prevFrameOk;
1647 }
1648 
1649 /*!
1650   \brief Get the number of delay frames introduced by concealment technique.
1651 
1652   \return Number of delay frames.
1653  */
CConcealment_GetDelay(CConcealParams * pConcealCommonData)1654 UINT CConcealment_GetDelay(CConcealParams *pConcealCommonData) {
1655   UINT frameDelay = 0;
1656 
1657   if (pConcealCommonData != NULL) {
1658     switch (pConcealCommonData->method) {
1659       case ConcealMethodTonal:
1660       case ConcealMethodInter:
1661         frameDelay = 1;
1662         break;
1663       default:
1664         break;
1665     }
1666   }
1667 
1668   return frameDelay;
1669 }
1670 
CConcealment_ApplyFadeOut(int mode,CConcealmentInfo * pConcealmentInfo,CAacDecoderStaticChannelInfo * pAacDecoderStaticChannelInfo,const int samplesPerFrame,CAacDecoderChannelInfo * pAacDecoderChannelInfo)1671 static int CConcealment_ApplyFadeOut(
1672     int mode, CConcealmentInfo *pConcealmentInfo,
1673     CAacDecoderStaticChannelInfo *pAacDecoderStaticChannelInfo,
1674     const int samplesPerFrame, CAacDecoderChannelInfo *pAacDecoderChannelInfo) {
1675   /* mode 1 = apply RandomSign and mute spectral coefficients if necessary,  *
1676    * mode 0 = Update cntFadeFrames                                            */
1677 
1678   /* restore frequency coefficients from buffer with a specific muting */
1679   int srcWin, dstWin, numWindows = 1;
1680   int windowLen = samplesPerFrame;
1681   int srcGrpStart = 0;
1682   int winIdxStride = 1;
1683   int numWinGrpPerFac, attIdx, attIdxStride;
1684   int i;
1685   int appliedProcessing = 0;
1686 
1687   CIcsInfo *pIcsInfo = &pAacDecoderChannelInfo->icsInfo;
1688   FIXP_DBL *pSpectralCoefficient =
1689       SPEC_LONG(pAacDecoderChannelInfo->pSpectralCoefficient);
1690   SHORT *pSpecScale = pAacDecoderChannelInfo->specScale;
1691 
1692   /* set old window parameters */
1693   if (pConcealmentInfo->lastRenderMode == AACDEC_RENDER_LPD) {
1694     switch (pAacDecoderStaticChannelInfo->last_lpd_mode) {
1695       case 1:
1696         numWindows = 4;
1697         srcGrpStart = 3;
1698         windowLen = samplesPerFrame >> 2;
1699         break;
1700       case 2:
1701         numWindows = 2;
1702         srcGrpStart = 1;
1703         windowLen = samplesPerFrame >> 1;
1704         winIdxStride = 2;
1705         break;
1706       case 3:
1707         numWindows = 1;
1708         srcGrpStart = 0;
1709         windowLen = samplesPerFrame;
1710         winIdxStride = 4;
1711         break;
1712     }
1713     pConcealmentInfo->lastWinGrpLen = 1;
1714   } else {
1715     pIcsInfo->WindowShape = pConcealmentInfo->windowShape;
1716     pIcsInfo->WindowSequence = pConcealmentInfo->windowSequence;
1717 
1718     if (pConcealmentInfo->windowSequence == BLOCK_SHORT) {
1719       /* short block handling */
1720       numWindows = 8;
1721       windowLen = samplesPerFrame >> 3;
1722       srcGrpStart = numWindows - pConcealmentInfo->lastWinGrpLen;
1723     }
1724   }
1725 
1726   attIdxStride =
1727       fMax(1, (int)(numWindows / (pConcealmentInfo->lastWinGrpLen + 1)));
1728 
1729   /* load last state */
1730   attIdx = pConcealmentInfo->cntFadeFrames;
1731   numWinGrpPerFac = pConcealmentInfo->attGrpOffset[mode];
1732   srcWin = srcGrpStart + pConcealmentInfo->winGrpOffset[mode];
1733 
1734   FDK_ASSERT((srcGrpStart * windowLen + windowLen) <= samplesPerFrame);
1735   FDK_ASSERT((srcWin * windowLen + windowLen) <= 1024);
1736 
1737   for (dstWin = 0; dstWin < numWindows; dstWin += 1) {
1738     FIXP_CNCL *pCncl =
1739         pConcealmentInfo->spectralCoefficient + (srcWin * windowLen);
1740     FIXP_DBL *pOut = pSpectralCoefficient + (dstWin * windowLen);
1741 
1742     if (mode == 1) {
1743       /* mute if attIdx gets large enaugh */
1744       if (attIdx > pConcealmentInfo->pConcealParams->numFadeOutFrames) {
1745         FDKmemclear(pCncl, sizeof(FIXP_DBL) * windowLen);
1746       }
1747 
1748       /* restore frequency coefficients from buffer - attenuation is done later
1749        */
1750       for (i = 0; i < windowLen; i++) {
1751         pOut[i] = pCncl[i];
1752       }
1753 
1754       /* apply random change of sign for spectral coefficients */
1755       CConcealment_ApplyRandomSign(pConcealmentInfo->iRandomPhase, pOut,
1756                                    windowLen);
1757 
1758       /* Increment random phase index to avoid repetition artifacts. */
1759       pConcealmentInfo->iRandomPhase =
1760           (pConcealmentInfo->iRandomPhase + 1) & (AAC_NF_NO_RANDOM_VAL - 1);
1761 
1762       /* set old scale factors */
1763       pSpecScale[dstWin * winIdxStride] =
1764           pConcealmentInfo->specScale[srcWin * winIdxStride];
1765     }
1766 
1767     srcWin += 1;
1768 
1769     if (srcWin >= numWindows) {
1770       /* end of sequence -> rewind to first window of group */
1771       srcWin = srcGrpStart;
1772       numWinGrpPerFac += 1;
1773       if (numWinGrpPerFac >= attIdxStride) {
1774         numWinGrpPerFac = 0;
1775         attIdx += 1;
1776       }
1777     }
1778   }
1779 
1780   /* store current state */
1781 
1782   pConcealmentInfo->winGrpOffset[mode] = srcWin - srcGrpStart;
1783   FDK_ASSERT((pConcealmentInfo->winGrpOffset[mode] >= 0) &&
1784              (pConcealmentInfo->winGrpOffset[mode] < 8));
1785   pConcealmentInfo->attGrpOffset[mode] = numWinGrpPerFac;
1786   FDK_ASSERT((pConcealmentInfo->attGrpOffset[mode] >= 0) &&
1787              (pConcealmentInfo->attGrpOffset[mode] < attIdxStride));
1788 
1789   if (mode == 0) {
1790     pConcealmentInfo->cntFadeFrames = attIdx;
1791   }
1792 
1793   appliedProcessing = 1;
1794 
1795   return appliedProcessing;
1796 }
1797 
1798 /*!
1799   \brief Do Time domain fading (TDFading) in concealment case
1800 
1801   In case of concealment, this function takes care of the fading, after time
1802 domain signal has been rendered by the respective signal rendering functions.
1803   The fading out in case of ACELP decoding is not done by this function but by
1804 the ACELP decoder for the first concealed frame if CONCEAL_CORE_IGNORANT_FADE is
1805 not set.
1806 
1807   TimeDomain fading never creates jumps in energy / discontinuities, it always
1808 does a continuous fading. To achieve this, fading is always done from a starting
1809 point to a target point, while the starting point is always determined to be the
1810 last target point. By varying the target point of a fading, the fading slope can
1811 be controlled.
1812 
1813   This principle is applied to the fading within a frame and the fading from
1814 frame to frame.
1815 
1816   One frame is divided into 8 subframes to obtain 8 parts of fading slopes
1817 within a frame, each maybe with its own gradient.
1818 
1819   Workflow:
1820   1.) Determine Fading behavior and end-of-frame target fading level, based on
1821 concealmentState (determined by CConcealment_UpdateState()) and the core mode.
1822         - By _DEFAULT_,
1823           The target fading level is determined by fadeOutFactor[cntFadeFrames]
1824 in case of fadeOut, or fadeInFactor[cntFadeFrames] in case of fadeIn.
1825           --> fading type is FADE_TIMEDOMAIN in this case. Target fading level
1826 is determined by fading index cntFadeFrames.
1827 
1828         - If concealmentState is signalling a _MUTED SIGNAL_,
1829           TDFading decays to 0 within 1/8th of a frame if numFadeOutFrames == 0.
1830           --> fading type is FADE_TIMEDOMAIN_TOSPECTRALMUTE in this case.
1831 
1832         - If concealmentState is signalling the _END OF MUTING_,
1833           TDFading fades to target fading level within 1/8th of a frame if
1834 numFadeInFrames == 0.
1835           --> fading type is FADE_TIMEDOMAIN_FROMSPECTRALMUTE in this case.
1836 Target fading level is determined by fading index cntFadeFrames.
1837 
1838 #ifndef CONCEAL_CORE_IGNORANT_FADE
1839         - In case of an _ACELP FADEOUT_,
1840           TDFading leaves fading control to ACELP decoder for 1/2 frame.
1841           --> fading type is FADE_ACELPDOMAIN in this case.
1842 #endif
1843 
1844   2.) Render fading levels within current frame and do the final fading:
1845       Map Fading slopes to fading levels and apply to time domain signal.
1846 
1847 
1848 */
1849 
CConcealment_TDFading(int len,CAacDecoderStaticChannelInfo ** ppAacDecoderStaticChannelInfo,FIXP_PCM * pcmdata,FIXP_PCM * pcmdata_1)1850 INT CConcealment_TDFading(
1851     int len, CAacDecoderStaticChannelInfo **ppAacDecoderStaticChannelInfo,
1852     FIXP_PCM *pcmdata, FIXP_PCM *pcmdata_1) {
1853   /*
1854   Do the fading in Time domain based on concealment states and core mode
1855   */
1856   FIXP_DBL fadeStop, attMute = (FIXP_DBL)0;
1857   int idx = 0, ii;
1858   CAacDecoderStaticChannelInfo *pAacDecoderStaticChannelInfo =
1859       *ppAacDecoderStaticChannelInfo;
1860   CConcealmentInfo *pConcealmentInfo =
1861       &pAacDecoderStaticChannelInfo->concealmentInfo;
1862   CConcealParams *pConcealParams = pConcealmentInfo->pConcealParams;
1863   const CConcealmentState concealState = pConcealmentInfo->concealState;
1864   TDfadingType fadingType;
1865   FIXP_DBL fadingStations[9] = {0};
1866   int fadingSteps[8] = {0};
1867   const FIXP_DBL fadeStart =
1868       pConcealmentInfo
1869           ->fade_old; /* start fading at last end-of-frame attenuation */
1870   FIXP_SGL *fadeFactor = pConcealParams->fadeOutFactor;
1871   const INT cntFadeFrames = pConcealmentInfo->cntFadeFrames;
1872   int TDFadeOutStopBeforeMute = 1;
1873   int TDFadeInStopBeforeFullLevel = 1;
1874 
1875   /*
1876   determine Fading behaviour (end-of-frame attenuation and fading type) (1.)
1877   */
1878 
1879   switch (concealState) {
1880     case ConcealState_Single:
1881     case ConcealState_Mute:
1882     case ConcealState_FadeOut:
1883       idx = (pConcealParams->method == ConcealMethodNoise) ? cntFadeFrames - 1
1884                                                            : cntFadeFrames;
1885       fadingType = FADE_TIMEDOMAIN;
1886 
1887       if (concealState == ConcealState_Mute ||
1888           (cntFadeFrames + TDFadeOutStopBeforeMute) >
1889               pConcealmentInfo->pConcealParams->numFadeOutFrames) {
1890         fadingType = FADE_TIMEDOMAIN_TOSPECTRALMUTE;
1891       }
1892 
1893       break;
1894     case ConcealState_FadeIn:
1895       idx = cntFadeFrames;
1896       idx -= TDFadeInStopBeforeFullLevel;
1897     case ConcealState_Ok:
1898       fadeFactor = pConcealParams->fadeInFactor;
1899       idx = (concealState == ConcealState_Ok) ? -1 : idx;
1900       fadingType = (pConcealmentInfo->concealState_old == ConcealState_Mute)
1901                        ? FADE_TIMEDOMAIN_FROMSPECTRALMUTE
1902                        : FADE_TIMEDOMAIN;
1903       break;
1904     default:
1905       FDK_ASSERT(0);
1906       fadingType = FADE_TIMEDOMAIN_TOSPECTRALMUTE;
1907       break;
1908   }
1909 
1910   /* determine Target end-of-frame fading level and fading slope */
1911   switch (fadingType) {
1912     case FADE_TIMEDOMAIN_FROMSPECTRALMUTE:
1913       fadeStop =
1914           (idx < 0) ? (FIXP_DBL)MAXVAL_DBL : FX_SGL2FX_DBL(fadeFactor[idx]);
1915       if (pConcealmentInfo->pConcealParams->numFadeInFrames == 0) {
1916         /* do step as fast as possible */
1917         fadingSteps[0] = 1;
1918         break;
1919       }
1920       CConcealment_TDFading_doLinearFadingSteps(&fadingSteps[0]);
1921       break;
1922     case FADE_TIMEDOMAIN:
1923       fadeStop =
1924           (idx < 0) ? (FIXP_DBL)MAXVAL_DBL : FX_SGL2FX_DBL(fadeFactor[idx]);
1925       CConcealment_TDFading_doLinearFadingSteps(&fadingSteps[0]);
1926       break;
1927     case FADE_TIMEDOMAIN_TOSPECTRALMUTE:
1928       fadeStop = attMute;
1929       if (pConcealmentInfo->pConcealParams->numFadeOutFrames == 0) {
1930         /* do step as fast as possible */
1931         fadingSteps[0] = 1;
1932         break;
1933       }
1934       CConcealment_TDFading_doLinearFadingSteps(&fadingSteps[0]);
1935       break;
1936   }
1937 
1938   /*
1939   Render fading levels within current frame and do the final fading (2.)
1940   */
1941 
1942   len >>= 3;
1943   CConcealment_TDFadeFillFadingStations(fadingStations, fadingSteps, fadeStop,
1944                                         fadeStart, fadingType);
1945 
1946   if ((fadingStations[8] != (FIXP_DBL)MAXVAL_DBL) ||
1947       (fadingStations[7] != (FIXP_DBL)MAXVAL_DBL) ||
1948       (fadingStations[6] != (FIXP_DBL)MAXVAL_DBL) ||
1949       (fadingStations[5] != (FIXP_DBL)MAXVAL_DBL) ||
1950       (fadingStations[4] != (FIXP_DBL)MAXVAL_DBL) ||
1951       (fadingStations[3] != (FIXP_DBL)MAXVAL_DBL) ||
1952       (fadingStations[2] != (FIXP_DBL)MAXVAL_DBL) ||
1953       (fadingStations[1] != (FIXP_DBL)MAXVAL_DBL) ||
1954       (fadingStations[0] !=
1955        (FIXP_DBL)MAXVAL_DBL)) /* if there's something to fade */
1956   {
1957     int start = 0;
1958     for (ii = 0; ii < 8; ii++) {
1959       CConcealment_TDFadePcmAtt(start, len, fadingStations[ii],
1960                                 fadingStations[ii + 1], pcmdata);
1961       start += len;
1962     }
1963   }
1964   CConcealment_TDNoise_Apply(pConcealmentInfo, len, pcmdata);
1965 
1966   /* Save end-of-frame attenuation and fading type */
1967   pConcealmentInfo->lastFadingType = fadingType;
1968   pConcealmentInfo->fade_old = fadeStop;
1969   pConcealmentInfo->concealState_old = concealState;
1970 
1971   return 1;
1972 }
1973 
1974 /* attenuate pcmdata in Time Domain Fading process */
CConcealment_TDFadePcmAtt(int start,int len,FIXP_DBL fadeStart,FIXP_DBL fadeStop,FIXP_PCM * pcmdata)1975 static void CConcealment_TDFadePcmAtt(int start, int len, FIXP_DBL fadeStart,
1976                                       FIXP_DBL fadeStop, FIXP_PCM *pcmdata) {
1977   int i;
1978   FIXP_DBL dStep;
1979   FIXP_DBL dGain;
1980   FIXP_DBL dGain_apply;
1981   int bitshift = (DFRACT_BITS - SAMPLE_BITS);
1982 
1983   /* set start energy */
1984   dGain = fadeStart;
1985   /* determine energy steps from sample to sample */
1986   dStep = (FIXP_DBL)((int)((fadeStart >> 1) - (fadeStop >> 1)) / len) << 1;
1987 
1988   for (i = start; i < (start + len); i++) {
1989     dGain -= dStep;
1990     /* prevent gain from getting negative due to possible fixpoint inaccuracies
1991      */
1992     dGain_apply = fMax((FIXP_DBL)0, dGain);
1993     /* finally, attenuate samples */
1994     pcmdata[i] = (FIXP_PCM)((fMult(pcmdata[i], (dGain_apply))) >> bitshift);
1995   }
1996 }
1997 
1998 /*
1999 \brief Fill FadingStations
2000 
2001 The fadingstations are the attenuation factors, being applied to its dedicated
2002 portions of pcm data. They are calculated using the fadingsteps. One fadingstep
2003 is the weighted contribution to the fading slope within its dedicated portion of
2004 pcm data.
2005 
2006 *Fadingsteps  :      0  0  0  1  0  1  2  0
2007 
2008                   |<-  1 Frame pcm data ->|
2009       fadeStart-->|__________             |
2010                   ^  ^  ^  ^ \____        |
2011  Attenuation  :   |  |  |  |  ^  ^\__     |
2012                   |  |  |  |  |  |  ^\    |
2013                   |  |  |  |  |  |  | \___|<-- fadeStop
2014                   |  |  |  |  |  |  |  ^  ^
2015                   |  |  |  |  |  |  |  |  |
2016 Fadingstations:  [0][1][2][3][4][5][6][7][8]
2017 
2018 (Fadingstations "[0]" is "[8] from previous frame", therefore its not meaningful
2019 to be edited)
2020 
2021 */
CConcealment_TDFadeFillFadingStations(FIXP_DBL * fadingStations,int * fadingSteps,FIXP_DBL fadeStop,FIXP_DBL fadeStart,TDfadingType fadingType)2022 static void CConcealment_TDFadeFillFadingStations(FIXP_DBL *fadingStations,
2023                                                   int *fadingSteps,
2024                                                   FIXP_DBL fadeStop,
2025                                                   FIXP_DBL fadeStart,
2026                                                   TDfadingType fadingType) {
2027   int i;
2028   INT fadingSteps_sum = 0;
2029   INT fadeDiff;
2030 
2031   fadingSteps_sum = fadingSteps[0] + fadingSteps[1] + fadingSteps[2] +
2032                     fadingSteps[3] + fadingSteps[4] + fadingSteps[5] +
2033                     fadingSteps[6] + fadingSteps[7];
2034   fadeDiff = ((INT)(fadeStop - fadeStart) / fMax(fadingSteps_sum, (INT)1));
2035   fadingStations[0] = fadeStart;
2036   for (i = 1; i < 8; i++) {
2037     fadingStations[i] =
2038         fadingStations[i - 1] + (FIXP_DBL)(fadeDiff * fadingSteps[i - 1]);
2039   }
2040   fadingStations[8] = fadeStop;
2041 }
2042 
CConcealment_TDFading_doLinearFadingSteps(int * fadingSteps)2043 static void CConcealment_TDFading_doLinearFadingSteps(int *fadingSteps) {
2044   fadingSteps[0] = fadingSteps[1] = fadingSteps[2] = fadingSteps[3] =
2045       fadingSteps[4] = fadingSteps[5] = fadingSteps[6] = fadingSteps[7] = 1;
2046 }
2047 
2048 /* end of TimeDomainFading functions */
2049 
2050 /* derived from int UsacRandomSign() */
CConcealment_TDNoise_Random(ULONG * seed)2051 static int CConcealment_TDNoise_Random(ULONG *seed) {
2052   *seed = (ULONG)(((UINT64)(*seed) * 69069) + 5);
2053   return (int)(*seed);
2054 }
2055 
CConcealment_TDNoise_Apply(CConcealmentInfo * const pConcealmentInfo,const int len,FIXP_PCM * const pcmdata)2056 static void CConcealment_TDNoise_Apply(CConcealmentInfo *const pConcealmentInfo,
2057                                        const int len, FIXP_PCM *const pcmdata) {
2058   FIXP_PCM *states = pConcealmentInfo->TDNoiseStates;
2059   FIXP_PCM noiseVal;
2060   FIXP_DBL noiseValLong;
2061   FIXP_SGL *coef = pConcealmentInfo->TDNoiseCoef;
2062   FIXP_DBL TDNoiseAtt;
2063   ULONG seed = pConcealmentInfo->TDNoiseSeed =
2064       (ULONG)CConcealment_TDNoise_Random(&pConcealmentInfo->TDNoiseSeed) + 1;
2065 
2066   TDNoiseAtt = pConcealmentInfo->pConcealParams->comfortNoiseLevel;
2067 
2068   int ii;
2069 
2070   if ((pConcealmentInfo->concealState != ConcealState_Ok ||
2071        pConcealmentInfo->concealState_old != ConcealState_Ok) &&
2072       TDNoiseAtt != (FIXP_DBL)0) {
2073     for (ii = 0; ii < (len << 3); ii++) {
2074       /* create filtered noise */
2075       states[2] = states[1];
2076       states[1] = states[0];
2077       states[0] = ((FIXP_PCM)CConcealment_TDNoise_Random(&seed));
2078       noiseValLong = fMult(states[0], coef[0]) + fMult(states[1], coef[1]) +
2079                      fMult(states[2], coef[2]);
2080       noiseVal = FX_DBL2FX_PCM(fMult(noiseValLong, TDNoiseAtt));
2081 
2082       /* add filtered noise - check for clipping, before */
2083       if (pcmdata[ii] > (FIXP_PCM)MAXVAL_FIXP_PCM - noiseVal &&
2084           noiseVal > (FIXP_PCM)0) {
2085         noiseVal = noiseVal * (FIXP_PCM)-1;
2086       } else if (pcmdata[ii] < (FIXP_PCM)MINVAL_FIXP_PCM - noiseVal &&
2087                  noiseVal < (FIXP_PCM)0) {
2088         noiseVal = noiseVal * (FIXP_PCM)-1;
2089       }
2090 
2091       pcmdata[ii] += noiseVal;
2092     }
2093   }
2094 }
2095