• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 
2 /* -----------------------------------------------------------------------------------------------------------
3 Software License for The Fraunhofer FDK AAC Codec Library for Android
4 
5 � Copyright  1995 - 2012 Fraunhofer-Gesellschaft zur F�rderung der angewandten Forschung e.V.
6   All rights reserved.
7 
8  1.    INTRODUCTION
9 The Fraunhofer FDK AAC Codec Library for Android ("FDK AAC Codec") is software that implements
10 the MPEG Advanced Audio Coding ("AAC") encoding and decoding scheme for digital audio.
11 This FDK AAC Codec software is intended to be used on a wide variety of Android devices.
12 
13 AAC's HE-AAC and HE-AAC v2 versions are regarded as today's most efficient general perceptual
14 audio codecs. AAC-ELD is considered the best-performing full-bandwidth communications codec by
15 independent studies and is widely deployed. AAC has been standardized by ISO and IEC as part
16 of the MPEG specifications.
17 
18 Patent licenses for necessary patent claims for the FDK AAC Codec (including those of Fraunhofer)
19 may be obtained through Via Licensing (www.vialicensing.com) or through the respective patent owners
20 individually for the purpose of encoding or decoding bit streams in products that are compliant with
21 the ISO/IEC MPEG audio standards. Please note that most manufacturers of Android devices already license
22 these patent claims through Via Licensing or directly from the patent owners, and therefore FDK AAC Codec
23 software may already be covered under those patent licenses when it is used for those licensed purposes only.
24 
25 Commercially-licensed AAC software libraries, including floating-point versions with enhanced sound quality,
26 are also available from Fraunhofer. Users are encouraged to check the Fraunhofer website for additional
27 applications information and documentation.
28 
29 2.    COPYRIGHT LICENSE
30 
31 Redistribution and use in source and binary forms, with or without modification, are permitted without
32 payment of copyright license fees provided that you satisfy the following conditions:
33 
34 You must retain the complete text of this software license in redistributions of the FDK AAC Codec or
35 your modifications thereto in source code form.
36 
37 You must retain the complete text of this software license in the documentation and/or other materials
38 provided with redistributions of the FDK AAC Codec or your modifications thereto in binary form.
39 You must make available free of charge copies of the complete source code of the FDK AAC Codec and your
40 modifications thereto to recipients of copies in binary form.
41 
42 The name of Fraunhofer may not be used to endorse or promote products derived from this library without
43 prior written permission.
44 
45 You may not charge copyright license fees for anyone to use, copy or distribute the FDK AAC Codec
46 software or your modifications thereto.
47 
48 Your modified versions of the FDK AAC Codec must carry prominent notices stating that you changed the software
49 and the date of any change. For modified versions of the FDK AAC Codec, the term
50 "Fraunhofer FDK AAC Codec Library for Android" must be replaced by the term
51 "Third-Party Modified Version of the Fraunhofer FDK AAC Codec Library for Android."
52 
53 3.    NO PATENT LICENSE
54 
55 NO EXPRESS OR IMPLIED LICENSES TO ANY PATENT CLAIMS, including without limitation the patents of Fraunhofer,
56 ARE GRANTED BY THIS SOFTWARE LICENSE. Fraunhofer provides no warranty of patent non-infringement with
57 respect to this software.
58 
59 You may use this FDK AAC Codec software or modifications thereto only for purposes that are authorized
60 by appropriate patent licenses.
61 
62 4.    DISCLAIMER
63 
64 This FDK AAC Codec software is provided by Fraunhofer on behalf of the copyright holders and contributors
65 "AS IS" and WITHOUT ANY EXPRESS OR IMPLIED WARRANTIES, including but not limited to the implied warranties
66 of merchantability and fitness for a particular purpose. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
67 CONTRIBUTORS BE LIABLE for any direct, indirect, incidental, special, exemplary, or consequential damages,
68 including but not limited to procurement of substitute goods or services; loss of use, data, or profits,
69 or business interruption, however caused and on any theory of liability, whether in contract, strict
70 liability, or tort (including negligence), arising in any way out of the use of this software, even if
71 advised of the possibility of such damage.
72 
73 5.    CONTACT INFORMATION
74 
75 Fraunhofer Institute for Integrated Circuits IIS
76 Attention: Audio and Multimedia Departments - FDK AAC LL
77 Am Wolfsmantel 33
78 91058 Erlangen, Germany
79 
80 www.iis.fraunhofer.de/amm
81 amm-info@iis.fraunhofer.de
82 ----------------------------------------------------------------------------------------------------------- */
83 
84 /*****************************  MPEG Audio Encoder  ***************************
85 
86    Initial Authors:      M. Neuendorf, N. Rettelbach, M. Multrus
87    Contents/Description: PS parameter extraction, encoding
88 
89 ******************************************************************************/
90 /*!
91   \file
92   \brief  PS parameter extraction, encoding functions
93 */
94 
95 #include "ps_main.h"
96 
97 
98 #include "sbr_ram.h"
99 #include "ps_encode.h"
100 
101 #include "qmf.h"
102 
103 #include "ps_const.h"
104 #include "sbr_misc.h"
105 
106 #include "genericStds.h"
107 
FDKsbrEnc_addFIXP_DBL(const FIXP_DBL * X,const FIXP_DBL * Y,FIXP_DBL * Z,INT n)108 inline void FDKsbrEnc_addFIXP_DBL(const FIXP_DBL *X, const FIXP_DBL *Y, FIXP_DBL *Z, INT n)
109 {
110   for (INT i=0; i<n; i++)
111     Z[i] = (X[i]>>1) + (Y[i]>>1);
112 }
113 
114 #define LOG10_2_10             3.01029995664f /* 10.0f*log10(2.f) */
115 
116 static const INT iidGroupBordersLoRes[QMF_GROUPS_LO_RES + SUBQMF_GROUPS_LO_RES + 1] =
117 {
118   0, 1, 2, 3, 4, 5,    /* 6 subqmf subbands - 0th qmf subband */
119   6, 7,                /* 2 subqmf subbands - 1st qmf subband */
120   8, 9,                /* 2 subqmf subbands - 2nd qmf subband */
121   10, 11, 12, 13, 14, 15, 16, 18, 21, 25, 30, 42, 71
122 };
123 
124 static const UCHAR iidGroupWidthLdLoRes[QMF_GROUPS_LO_RES + SUBQMF_GROUPS_LO_RES] =
125 {
126   0, 0, 0, 0, 0, 0,
127   0, 0,
128   0, 0,
129   0, 0, 0, 0, 0, 0, 1, 2, 2, 3, 4, 5
130 };
131 
132 
133 static const INT subband2parameter20[QMF_GROUPS_LO_RES + SUBQMF_GROUPS_LO_RES] =
134 {
135   1, 0, 0, 1, 2, 3,   /* 6 subqmf subbands - 0th qmf subband */
136   4, 5,               /* 2 subqmf subbands - 1st qmf subband */
137   6, 7,               /* 2 subqmf subbands - 2nd qmf subband */
138   8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19
139 };
140 
141 
142 typedef enum {
143   MAX_TIME_DIFF_FRAMES = 20,
144   MAX_PS_NOHEADER_CNT  = 10,
145   MAX_NOENV_CNT        = 10,
146   DO_NOT_USE_THIS_MODE = 0x7FFFFF
147 } __PS_CONSTANTS;
148 
149 
150 
151 static const FIXP_DBL iidQuant_fx[15] = {
152   0xce000000, 0xdc000000, 0xe4000000, 0xec000000, 0xf2000000, 0xf8000000, 0xfc000000, 0x00000000,
153   0x04000000, 0x08000000, 0x0e000000, 0x14000000, 0x1c000000, 0x24000000, 0x32000000
154 };
155 
156 static const FIXP_DBL iidQuantFine_fx[31] = {
157   0x9c000001, 0xa6000001, 0xb0000001, 0xba000001, 0xc4000000, 0xce000000, 0xd4000000, 0xda000000,
158   0xe0000000, 0xe6000000, 0xec000000, 0xf0000000, 0xf4000000, 0xf8000000, 0xfc000000, 0x00000000,
159   0x04000000, 0x08000000, 0x0c000000, 0x10000000, 0x14000000, 0x1a000000, 0x20000000, 0x26000000,
160   0x2c000000, 0x32000000, 0x3c000000, 0x45ffffff, 0x4fffffff, 0x59ffffff, 0x63ffffff
161 };
162 
163 
164 
165 static const FIXP_DBL iccQuant[8] = {
166   0x7fffffff, 0x77ef9d7f, 0x6babc97f, 0x4ceaf27f, 0x2f0ed3c0, 0x00000000, 0xb49ba601, 0x80000000
167 };
168 
InitPSData(HANDLE_PS_DATA hPsData)169 static FDK_PSENC_ERROR InitPSData(
170         HANDLE_PS_DATA            hPsData
171         )
172 {
173   FDK_PSENC_ERROR error = PSENC_OK;
174 
175   if(hPsData == NULL) {
176     error = PSENC_INVALID_HANDLE;
177   }
178   else {
179     int i, env;
180     FDKmemclear(hPsData,sizeof(PS_DATA));
181 
182     for (i=0; i<PS_MAX_BANDS; i++) {
183       hPsData->iidIdxLast[i] = 0;
184       hPsData->iccIdxLast[i] = 0;
185     }
186 
187     hPsData->iidEnable    = hPsData->iidEnableLast = 0;
188     hPsData->iccEnable    = hPsData->iccEnableLast = 0;
189     hPsData->iidQuantMode = hPsData->iidQuantModeLast = PS_IID_RES_COARSE;
190     hPsData->iccQuantMode = hPsData->iccQuantModeLast = PS_ICC_ROT_A;
191 
192     for(env=0; env<PS_MAX_ENVELOPES; env++) {
193       hPsData->iccDiffMode[env] = PS_DELTA_FREQ;
194       hPsData->iccDiffMode[env] = PS_DELTA_FREQ;
195 
196       for (i=0; i<PS_MAX_BANDS; i++) {
197         hPsData->iidIdx[env][i] = 0;
198         hPsData->iccIdx[env][i] = 0;
199       }
200     }
201 
202     hPsData->nEnvelopesLast = 0;
203 
204     hPsData->headerCnt  = MAX_PS_NOHEADER_CNT;
205     hPsData->iidTimeCnt = MAX_TIME_DIFF_FRAMES;
206     hPsData->iccTimeCnt = MAX_TIME_DIFF_FRAMES;
207     hPsData->noEnvCnt   = MAX_NOENV_CNT;
208   }
209 
210   return error;
211 }
212 
quantizeCoef(const FIXP_DBL * RESTRICT input,const INT nBands,const FIXP_DBL * RESTRICT quantTable,const INT idxOffset,const INT nQuantSteps,INT * RESTRICT quantOut)213 static FIXP_DBL quantizeCoef( const FIXP_DBL *RESTRICT input,
214                               const INT       nBands,
215                               const FIXP_DBL *RESTRICT quantTable,
216                               const INT       idxOffset,
217                               const INT       nQuantSteps,
218                               INT            *RESTRICT quantOut)
219 {
220   INT idx, band;
221   FIXP_DBL quantErr = FL2FXCONST_DBL(0.f);
222 
223   for (band=0; band<nBands;band++) {
224     for(idx=0; idx<nQuantSteps-1; idx++){
225       if( fixp_abs((input[band]>>1)-(quantTable[idx+1]>>1)) >
226           fixp_abs((input[band]>>1)-(quantTable[idx]>>1)) )
227       {
228         break;
229       }
230     }
231     quantErr      += (fixp_abs(input[band]-quantTable[idx])>>PS_QUANT_SCALE);   /* don't scale before subtraction; diff smaller (64-25)/64 */
232     quantOut[band] = idx - idxOffset;
233   }
234 
235   return quantErr;
236 }
237 
getICCMode(const INT nBands,const INT rotType)238 static INT getICCMode(const INT nBands,
239                       const INT rotType)
240 {
241   INT mode = 0;
242 
243   switch(nBands) {
244   case PS_BANDS_COARSE:
245     mode = PS_RES_COARSE;
246     break;
247   case PS_BANDS_MID:
248     mode = PS_RES_MID;
249     break;
250   default:
251     mode = 0;
252   }
253   if(rotType==PS_ICC_ROT_B){
254     mode += 3;
255   }
256 
257   return mode;
258 }
259 
260 
getIIDMode(const INT nBands,const INT iidRes)261 static INT getIIDMode(const INT nBands,
262                       const INT iidRes)
263 {
264   INT mode = 0;
265 
266   switch(nBands) {
267   case PS_BANDS_COARSE:
268     mode = PS_RES_COARSE;
269     break;
270   case PS_BANDS_MID:
271     mode = PS_RES_MID;
272     break;
273   default:
274     mode = 0;
275     break;
276   }
277 
278   if(iidRes == PS_IID_RES_FINE){
279     mode += 3;
280   }
281 
282   return mode;
283 }
284 
285 
envelopeReducible(FIXP_DBL iid[PS_MAX_ENVELOPES][PS_MAX_BANDS],FIXP_DBL icc[PS_MAX_ENVELOPES][PS_MAX_BANDS],INT psBands,INT nEnvelopes)286 static INT envelopeReducible(FIXP_DBL iid[PS_MAX_ENVELOPES][PS_MAX_BANDS],
287                              FIXP_DBL icc[PS_MAX_ENVELOPES][PS_MAX_BANDS],
288                              INT psBands,
289                              INT nEnvelopes)
290 {
291   #define THRESH_SCALE     7
292 
293   INT reducible = 1; /* true */
294   INT e = 0, b = 0;
295   FIXP_DBL dIid = FL2FXCONST_DBL(0.f);
296   FIXP_DBL dIcc = FL2FXCONST_DBL(0.f);
297 
298   FIXP_DBL iidErrThreshold, iccErrThreshold;
299   FIXP_DBL iidMeanError, iccMeanError;
300 
301   /* square values to prevent sqrt,
302      multiply bands to prevent division; bands shifted DFRACT_BITS instead (DFRACT_BITS-1) because fMultDiv2 used*/
303   iidErrThreshold = fMultDiv2 ( FL2FXCONST_DBL(6.5f*6.5f/(IID_SCALE_FT*IID_SCALE_FT)), (FIXP_DBL)(psBands<<((DFRACT_BITS)-THRESH_SCALE)) );
304   iccErrThreshold = fMultDiv2 ( FL2FXCONST_DBL(0.75f*0.75f),                           (FIXP_DBL)(psBands<<((DFRACT_BITS)-THRESH_SCALE)) );
305 
306   if (nEnvelopes <= 1) {
307     reducible = 0;
308   } else {
309 
310     /* mean error criterion */
311     for (e=0; (e < nEnvelopes/2) && (reducible!=0 ) ; e++) {
312       iidMeanError = iccMeanError = FL2FXCONST_DBL(0.f);
313       for(b=0; b<psBands; b++) {
314         dIid = (iid[2*e][b]>>1) - (iid[2*e+1][b]>>1);   /* scale 1 bit; squared -> 2 bit */
315         dIcc = (icc[2*e][b]>>1) - (icc[2*e+1][b]>>1);
316         iidMeanError += fPow2Div2(dIid)>>(5-1);    /* + (bands=20) scale = 5 */
317         iccMeanError += fPow2Div2(dIcc)>>(5-1);
318       }                                                 /* --> scaling = 7 bit = THRESH_SCALE !! */
319 
320       /* instead sqrt values are squared!
321          instead of division, multiply threshold with psBands
322          scaling necessary!! */
323 
324       /* quit as soon as threshold is reached */
325       if ( (iidMeanError > (iidErrThreshold)) ||
326            (iccMeanError > (iccErrThreshold)) ) {
327         reducible = 0;
328       }
329     }
330   } /* nEnvelopes != 1 */
331 
332   return reducible;
333 }
334 
335 
processIidData(PS_DATA * psData,FIXP_DBL iid[PS_MAX_ENVELOPES][PS_MAX_BANDS],const INT psBands,const INT nEnvelopes,const FIXP_DBL quantErrorThreshold)336 static void processIidData(PS_DATA       *psData,
337                            FIXP_DBL       iid[PS_MAX_ENVELOPES][PS_MAX_BANDS],
338                            const INT      psBands,
339                            const INT      nEnvelopes,
340                            const FIXP_DBL quantErrorThreshold)
341 {
342   INT iidIdxFine  [PS_MAX_ENVELOPES][PS_MAX_BANDS];
343   INT iidIdxCoarse[PS_MAX_ENVELOPES][PS_MAX_BANDS];
344 
345   FIXP_DBL errIID = FL2FXCONST_DBL(0.f);
346   FIXP_DBL errIIDFine = FL2FXCONST_DBL(0.f);
347   INT   bitsIidFreq = 0;
348   INT   bitsIidTime = 0;
349   INT   bitsFineTot = 0;
350   INT   bitsCoarseTot = 0;
351   INT   error = 0;
352   INT   env, band;
353   INT   diffMode[PS_MAX_ENVELOPES], diffModeFine[PS_MAX_ENVELOPES];
354   INT loudnDiff = 0;
355   INT iidTransmit = 0;
356 
357   bitsIidFreq = bitsIidTime = 0;
358 
359   /* Quantize IID coefficients */
360   for(env=0;env<nEnvelopes; env++) {
361     errIID     += quantizeCoef(iid[env], psBands, iidQuant_fx,      7, 15, iidIdxCoarse[env]);
362     errIIDFine += quantizeCoef(iid[env], psBands, iidQuantFine_fx, 15, 31, iidIdxFine[env]);
363   }
364 
365   /* normalize error to number of envelopes, ps bands
366      errIID /= psBands*nEnvelopes;
367      errIIDFine /= psBands*nEnvelopes; */
368 
369 
370   /* Check if IID coefficients should be used in this frame */
371   psData->iidEnable = 0;
372   for(env=0;env<nEnvelopes; env++) {
373     for(band=0;band<psBands;band++) {
374       loudnDiff   += fixp_abs(iidIdxCoarse[env][band]);
375       iidTransmit ++;
376     }
377   }
378 
379   if(loudnDiff > fMultI(FL2FXCONST_DBL(0.7f),iidTransmit)){    /* 0.7f empiric value */
380     psData->iidEnable = 1;
381   }
382 
383   /* if iid not active -> RESET data */
384   if(psData->iidEnable==0) {
385     psData->iidTimeCnt = MAX_TIME_DIFF_FRAMES;
386     for(env=0;env<nEnvelopes; env++) {
387       psData->iidDiffMode[env] = PS_DELTA_FREQ;
388       FDKmemclear(psData->iidIdx[env], sizeof(INT)*psBands);
389     }
390     return;
391   }
392 
393   /* count COARSE quantization bits for first envelope*/
394   bitsIidFreq = FDKsbrEnc_EncodeIid(NULL, iidIdxCoarse[0], NULL, psBands, PS_IID_RES_COARSE, PS_DELTA_FREQ, &error);
395 
396   if( (psData->iidTimeCnt>=MAX_TIME_DIFF_FRAMES) || (psData->iidQuantModeLast==PS_IID_RES_FINE) ) {
397     bitsIidTime     = DO_NOT_USE_THIS_MODE;
398   }
399   else {
400     bitsIidTime     = FDKsbrEnc_EncodeIid(NULL, iidIdxCoarse[0], psData->iidIdxLast, psBands, PS_IID_RES_COARSE, PS_DELTA_TIME, &error);
401   }
402 
403   /* decision DELTA_FREQ vs DELTA_TIME */
404   if(bitsIidTime>bitsIidFreq) {
405     diffMode[0]   = PS_DELTA_FREQ;
406     bitsCoarseTot = bitsIidFreq;
407   }
408   else {
409     diffMode[0]   = PS_DELTA_TIME;
410     bitsCoarseTot = bitsIidTime;
411   }
412 
413   /* count COARSE quantization bits for following envelopes*/
414   for(env=1;env<nEnvelopes; env++) {
415     bitsIidFreq  = FDKsbrEnc_EncodeIid(NULL, iidIdxCoarse[env], NULL,                psBands, PS_IID_RES_COARSE, PS_DELTA_FREQ, &error);
416     bitsIidTime  = FDKsbrEnc_EncodeIid(NULL, iidIdxCoarse[env], iidIdxCoarse[env-1], psBands, PS_IID_RES_COARSE, PS_DELTA_TIME, &error);
417 
418     /* decision DELTA_FREQ vs DELTA_TIME */
419     if(bitsIidTime>bitsIidFreq) {
420       diffMode[env]  = PS_DELTA_FREQ;
421       bitsCoarseTot += bitsIidFreq;
422     }
423     else {
424       diffMode[env]  = PS_DELTA_TIME;
425       bitsCoarseTot += bitsIidTime;
426     }
427   }
428 
429 
430   /* count FINE quantization bits for first envelope*/
431   bitsIidFreq = FDKsbrEnc_EncodeIid(NULL, iidIdxFine[0],   NULL, psBands, PS_IID_RES_FINE,   PS_DELTA_FREQ, &error);
432 
433   if( (psData->iidTimeCnt>=MAX_TIME_DIFF_FRAMES) || (psData->iidQuantModeLast==PS_IID_RES_COARSE) ) {
434     bitsIidTime = DO_NOT_USE_THIS_MODE;
435   }
436   else {
437     bitsIidTime = FDKsbrEnc_EncodeIid(NULL, iidIdxFine[0],  psData->iidIdxLast, psBands, PS_IID_RES_FINE, PS_DELTA_TIME, &error);
438   }
439 
440   /* decision DELTA_FREQ vs DELTA_TIME */
441   if(bitsIidTime>bitsIidFreq) {
442     diffModeFine[0]   = PS_DELTA_FREQ;
443     bitsFineTot       = bitsIidFreq;
444   }
445   else {
446     diffModeFine[0]   = PS_DELTA_TIME;
447     bitsFineTot       = bitsIidTime;
448   }
449 
450   /* count FINE quantization bits for following envelopes*/
451   for(env=1;env<nEnvelopes; env++) {
452     bitsIidFreq = FDKsbrEnc_EncodeIid(NULL, iidIdxFine[env],   NULL,              psBands, PS_IID_RES_FINE, PS_DELTA_FREQ, &error);
453     bitsIidTime = FDKsbrEnc_EncodeIid(NULL, iidIdxFine[env],   iidIdxFine[env-1], psBands, PS_IID_RES_FINE, PS_DELTA_TIME, &error);
454 
455     /* decision DELTA_FREQ vs DELTA_TIME */
456     if(bitsIidTime>bitsIidFreq) {
457       diffModeFine[env]  = PS_DELTA_FREQ;
458       bitsFineTot += bitsIidFreq;
459     }
460     else {
461       diffModeFine[env]  = PS_DELTA_TIME;
462       bitsFineTot       += bitsIidTime;
463     }
464   }
465 
466   if(bitsFineTot == bitsCoarseTot){
467     /* if same number of bits is needed, use the quantization with lower error */
468     if(errIIDFine < errIID){
469       bitsCoarseTot = DO_NOT_USE_THIS_MODE;
470     } else {
471       bitsFineTot = DO_NOT_USE_THIS_MODE;
472     }
473   } else {
474     /* const FIXP_DBL minThreshold = FL2FXCONST_DBL(0.2f/(IID_SCALE_FT*PS_QUANT_SCALE_FT)*(psBands*nEnvelopes)); */
475     const FIXP_DBL minThreshold = (FIXP_DBL)((LONG)0x00019999 * (psBands*nEnvelopes));
476 
477     /* decision RES_FINE vs RES_COARSE                 */
478     /* test if errIIDFine*quantErrorThreshold < errIID */
479     /* shiftVal 2 comes from scaling of quantErrorThreshold */
480     if(fixMax(((errIIDFine>>1)+(minThreshold>>1))>>1, fMult(quantErrorThreshold,errIIDFine)) < (errIID>>2) ) {
481       bitsCoarseTot = DO_NOT_USE_THIS_MODE;
482     }
483     else if(fixMax(((errIID>>1)+(minThreshold>>1))>>1, fMult(quantErrorThreshold,errIID)) < (errIIDFine>>2) ) {
484       bitsFineTot = DO_NOT_USE_THIS_MODE;
485     }
486   }
487 
488   /* decision RES_FINE vs RES_COARSE */
489   if(bitsFineTot<bitsCoarseTot) {
490     psData->iidQuantMode = PS_IID_RES_FINE;
491     for(env=0;env<nEnvelopes; env++) {
492       psData->iidDiffMode[env] = diffModeFine[env];
493       FDKmemcpy(psData->iidIdx[env], iidIdxFine[env], psBands*sizeof(INT));
494     }
495   }
496   else {
497     psData->iidQuantMode = PS_IID_RES_COARSE;
498     for(env=0;env<nEnvelopes; env++) {
499       psData->iidDiffMode[env] = diffMode[env];
500       FDKmemcpy(psData->iidIdx[env], iidIdxCoarse[env], psBands*sizeof(INT));
501     }
502   }
503 
504   /* Count DELTA_TIME encoding streaks */
505   for(env=0;env<nEnvelopes; env++) {
506     if(psData->iidDiffMode[env]==PS_DELTA_TIME)
507       psData->iidTimeCnt++;
508     else
509       psData->iidTimeCnt=0;
510   }
511 }
512 
513 
similarIid(PS_DATA * psData,const INT psBands,const INT nEnvelopes)514 static INT similarIid(PS_DATA   *psData,
515                       const INT  psBands,
516                       const INT  nEnvelopes)
517 {
518   const INT diffThr = (psData->iidQuantMode == PS_IID_RES_COARSE) ? 2 : 3;
519   const INT sumDiffThr = diffThr * psBands/4;
520   INT similar = 0;
521   INT diff    = 0;
522   INT sumDiff = 0;
523   INT env = 0;
524   INT b   = 0;
525   if ((nEnvelopes == psData->nEnvelopesLast) && (nEnvelopes==1)) {
526     similar = 1;
527     for (env=0; env<nEnvelopes; env++) {
528       sumDiff = 0;
529       b = 0;
530       do {
531         diff = fixp_abs(psData->iidIdx[env][b] - psData->iidIdxLast[b]);
532         sumDiff += diff;
533         if ( (diff > diffThr) /* more than x quantization steps in any band */
534              || (sumDiff > sumDiffThr) ) {  /* more than x quantisations steps overall difference */
535           similar = 0;
536         }
537         b++;
538       } while ((b<psBands) && (similar>0));
539     }
540   } /* nEnvelopes==1  */
541 
542   return similar;
543 }
544 
545 
similarIcc(PS_DATA * psData,const INT psBands,const INT nEnvelopes)546 static INT similarIcc(PS_DATA *psData,
547                       const INT    psBands,
548                       const INT    nEnvelopes)
549 {
550   const INT diffThr = 2;
551   const INT sumDiffThr = diffThr * psBands/4;
552   INT similar = 0;
553   INT diff    = 0;
554   INT sumDiff = 0;
555   INT env = 0;
556   INT b   = 0;
557   if ((nEnvelopes == psData->nEnvelopesLast) && (nEnvelopes==1)) {
558     similar = 1;
559     for (env=0; env<nEnvelopes; env++) {
560       sumDiff = 0;
561       b = 0;
562       do {
563         diff = fixp_abs(psData->iccIdx[env][b] - psData->iccIdxLast[b]);
564         sumDiff += diff;
565         if ( (diff > diffThr) /* more than x quantisation step in any band */
566              || (sumDiff > sumDiffThr) ) {  /* more than x quantisations steps overall difference */
567           similar = 0;
568         }
569         b++;
570       } while ((b<psBands) && (similar>0));
571     }
572   } /* nEnvelopes==1  */
573 
574   return similar;
575 }
576 
processIccData(PS_DATA * psData,FIXP_DBL icc[PS_MAX_ENVELOPES][PS_MAX_BANDS],const INT psBands,const INT nEnvelopes)577 static void processIccData(PS_DATA   *psData,
578                            FIXP_DBL   icc[PS_MAX_ENVELOPES][PS_MAX_BANDS], /* const input values: unable to declare as const, since it does not poINT to const memory */
579                            const INT  psBands,
580                            const INT  nEnvelopes)
581 {
582   FIXP_DBL errICC = FL2FXCONST_DBL(0.f);
583   INT   env, band;
584   INT   bitsIccFreq, bitsIccTime;
585   INT   error = 0;
586   INT   inCoherence=0, iccTransmit=0;
587   INT  *iccIdxLast;
588 
589   iccIdxLast = psData->iccIdxLast;
590 
591   /* Quantize ICC coefficients */
592   for(env=0;env<nEnvelopes; env++) {
593     errICC += quantizeCoef(icc[env], psBands, iccQuant, 0, 8, psData->iccIdx[env]);
594   }
595 
596   /* Check if ICC coefficients should be used */
597   psData->iccEnable = 0;
598   for(env=0;env<nEnvelopes; env++) {
599     for(band=0;band<psBands;band++) {
600       inCoherence += psData->iccIdx[env][band];
601       iccTransmit ++;
602     }
603   }
604   if(inCoherence > fMultI(FL2FXCONST_DBL(0.5f),iccTransmit)){   /* 0.5f empiric value */
605     psData->iccEnable = 1;
606   }
607 
608   if(psData->iccEnable==0) {
609     psData->iccTimeCnt = MAX_TIME_DIFF_FRAMES;
610     for(env=0;env<nEnvelopes; env++) {
611       psData->iccDiffMode[env] = PS_DELTA_FREQ;
612       FDKmemclear(psData->iccIdx[env], sizeof(INT)*psBands);
613     }
614     return;
615   }
616 
617   for(env=0;env<nEnvelopes; env++) {
618     bitsIccFreq  = FDKsbrEnc_EncodeIcc(NULL, psData->iccIdx[env],   NULL,       psBands, PS_DELTA_FREQ, &error);
619 
620     if(psData->iccTimeCnt<MAX_TIME_DIFF_FRAMES) {
621       bitsIccTime  = FDKsbrEnc_EncodeIcc(NULL, psData->iccIdx[env], iccIdxLast, psBands, PS_DELTA_TIME, &error);
622     }
623     else {
624       bitsIccTime  = DO_NOT_USE_THIS_MODE;
625     }
626 
627     if(bitsIccFreq>bitsIccTime) {
628       psData->iccDiffMode[env] = PS_DELTA_TIME;
629       psData->iccTimeCnt++;
630     }
631     else {
632       psData->iccDiffMode[env] = PS_DELTA_FREQ;
633       psData->iccTimeCnt=0;
634     }
635     iccIdxLast = psData->iccIdx[env];
636   }
637 }
638 
calculateIID(FIXP_DBL ldPwrL[PS_MAX_ENVELOPES][PS_MAX_BANDS],FIXP_DBL ldPwrR[PS_MAX_ENVELOPES][PS_MAX_BANDS],FIXP_DBL iid[PS_MAX_ENVELOPES][PS_MAX_BANDS],INT nEnvelopes,INT psBands)639 static void calculateIID(FIXP_DBL ldPwrL[PS_MAX_ENVELOPES][PS_MAX_BANDS],
640                          FIXP_DBL ldPwrR[PS_MAX_ENVELOPES][PS_MAX_BANDS],
641                          FIXP_DBL iid[PS_MAX_ENVELOPES][PS_MAX_BANDS],
642                          INT   nEnvelopes,
643                          INT   psBands)
644 {
645   INT i=0;
646   INT env=0;
647   for(env=0; env<nEnvelopes;env++) {
648     for (i=0; i<psBands; i++) {
649 
650       /* iid[env][i] = 10.0f*(float)log10(pwrL[env][i]/pwrR[env][i]);
651       */
652       FIXP_DBL IID = fMultDiv2( FL2FXCONST_DBL(LOG10_2_10/IID_SCALE_FT), (ldPwrL[env][i]-ldPwrR[env][i]) );
653 
654       IID = fixMin( IID, (FIXP_DBL)(MAXVAL_DBL>>(LD_DATA_SHIFT+1)) );
655       IID = fixMax( IID, (FIXP_DBL)(MINVAL_DBL>>(LD_DATA_SHIFT+1)) );
656       iid[env][i] = IID << (LD_DATA_SHIFT+1);
657     }
658   }
659 }
660 
calculateICC(FIXP_DBL ldPwrL[PS_MAX_ENVELOPES][PS_MAX_BANDS],FIXP_DBL ldPwrR[PS_MAX_ENVELOPES][PS_MAX_BANDS],FIXP_DBL pwrCr[PS_MAX_ENVELOPES][PS_MAX_BANDS],FIXP_DBL pwrCi[PS_MAX_ENVELOPES][PS_MAX_BANDS],FIXP_DBL icc[PS_MAX_ENVELOPES][PS_MAX_BANDS],INT nEnvelopes,INT psBands)661 static void calculateICC(FIXP_DBL ldPwrL[PS_MAX_ENVELOPES][PS_MAX_BANDS],
662                          FIXP_DBL ldPwrR[PS_MAX_ENVELOPES][PS_MAX_BANDS],
663                          FIXP_DBL pwrCr[PS_MAX_ENVELOPES][PS_MAX_BANDS],
664                          FIXP_DBL pwrCi[PS_MAX_ENVELOPES][PS_MAX_BANDS],
665                          FIXP_DBL icc[PS_MAX_ENVELOPES][PS_MAX_BANDS],
666                          INT   nEnvelopes,
667                          INT   psBands)
668 {
669   INT i = 0;
670   INT env = 0;
671   INT border = psBands;
672 
673   switch (psBands) {
674   case PS_BANDS_COARSE:
675     border = 5;
676     break;
677   case PS_BANDS_MID:
678     border = 11;
679     break;
680   default:
681     break;
682   }
683 
684   for(env=0; env<nEnvelopes;env++) {
685     for (i=0; i<border; i++) {
686 
687       /* icc[env][i] = min( pwrCr[env][i] / (float) sqrt(pwrL[env][i] * pwrR[env][i]) , 1.f);
688       */
689       FIXP_DBL ICC, invNrg = CalcInvLdData ( -((ldPwrL[env][i]>>1) + (ldPwrR[env][i]>>1) + (FIXP_DBL)1) );
690       INT      scale, invScale = CountLeadingBits(invNrg);
691 
692       scale = (DFRACT_BITS-1) - invScale;
693       ICC = fMult(pwrCr[env][i], invNrg<<invScale) ;
694       icc[env][i] = SATURATE_LEFT_SHIFT(ICC, scale, DFRACT_BITS);
695     }
696 
697     for (; i<psBands; i++) {
698       INT sc1, sc2;
699       FIXP_DBL cNrgR, cNrgI, ICC;
700 
701       sc1 = CountLeadingBits( fixMax(fixp_abs(pwrCr[env][i]),fixp_abs(pwrCi[env][i])) ) ;
702       cNrgR = fPow2Div2((pwrCr[env][i]<<sc1));       /* squared nrg's expect explicit scaling */
703       cNrgI = fPow2Div2((pwrCi[env][i]<<sc1));
704 
705       ICC = CalcInvLdData( (CalcLdData((cNrgR + cNrgI)>>1)>>1) - (FIXP_DBL)((sc1-1)<<(DFRACT_BITS-1-LD_DATA_SHIFT)) );
706 
707       FIXP_DBL invNrg = CalcInvLdData ( -((ldPwrL[env][i]>>1) + (ldPwrR[env][i]>>1) + (FIXP_DBL)1) );
708       sc1 = CountLeadingBits(invNrg);
709       invNrg <<= sc1;
710 
711       sc2 = CountLeadingBits(ICC);
712       ICC = fMult(ICC<<sc2,invNrg);
713 
714       sc1 = ( (DFRACT_BITS-1) - sc1 - sc2 );
715       if (sc1 < 0) {
716           ICC >>= -sc1;
717       }
718       else {
719           if (ICC >= ((FIXP_DBL)MAXVAL_DBL>>sc1) )
720               ICC = (FIXP_DBL)MAXVAL_DBL;
721           else
722               ICC <<= sc1;
723       }
724 
725       icc[env][i] = ICC;
726     }
727   }
728 }
729 
FDKsbrEnc_initPsBandNrgScale(HANDLE_PS_ENCODE hPsEncode)730 void FDKsbrEnc_initPsBandNrgScale(HANDLE_PS_ENCODE hPsEncode)
731 {
732   INT group, bin;
733   INT nIidGroups   = hPsEncode->nQmfIidGroups + hPsEncode->nSubQmfIidGroups;
734 
735   FDKmemclear(hPsEncode->psBandNrgScale, PS_MAX_BANDS*sizeof(SCHAR));
736 
737   for (group=0; group < nIidGroups; group++) {
738     /* Translate group to bin */
739     bin = hPsEncode->subband2parameterIndex[group];
740 
741     /* Translate from 20 bins to 10 bins */
742     if (hPsEncode->psEncMode == PS_BANDS_COARSE) {
743       bin = bin>>1;
744     }
745 
746     hPsEncode->psBandNrgScale[bin] = (hPsEncode->psBandNrgScale[bin]==0)
747                           ? (hPsEncode->iidGroupWidthLd[group] + 5)
748                           : (fixMax(hPsEncode->iidGroupWidthLd[group],hPsEncode->psBandNrgScale[bin]) + 1) ;
749 
750   }
751 }
752 
FDKsbrEnc_CreatePSEncode(HANDLE_PS_ENCODE * phPsEncode)753 FDK_PSENC_ERROR FDKsbrEnc_CreatePSEncode(
754         HANDLE_PS_ENCODE         *phPsEncode
755         )
756 {
757   FDK_PSENC_ERROR error = PSENC_OK;
758 
759   if (phPsEncode==NULL) {
760     error = PSENC_INVALID_HANDLE;
761   }
762   else {
763     HANDLE_PS_ENCODE hPsEncode = NULL;
764     if (NULL==(hPsEncode = GetRam_PsEncode())) {
765       error = PSENC_MEMORY_ERROR;
766       goto bail;
767     }
768     FDKmemclear(hPsEncode,sizeof(PS_ENCODE));
769     *phPsEncode = hPsEncode; /* return allocated handle */
770   }
771 bail:
772   return error;
773 }
774 
FDKsbrEnc_InitPSEncode(HANDLE_PS_ENCODE hPsEncode,const PS_BANDS psEncMode,const FIXP_DBL iidQuantErrorThreshold)775 FDK_PSENC_ERROR FDKsbrEnc_InitPSEncode(
776         HANDLE_PS_ENCODE          hPsEncode,
777         const PS_BANDS            psEncMode,
778         const FIXP_DBL            iidQuantErrorThreshold
779         )
780 {
781   FDK_PSENC_ERROR error = PSENC_OK;
782 
783   if (NULL==hPsEncode) {
784     error = PSENC_INVALID_HANDLE;
785   }
786   else {
787     if (PSENC_OK != (InitPSData(&hPsEncode->psData))) {
788       goto bail;
789     }
790 
791     switch(psEncMode){
792       case PS_BANDS_COARSE:
793       case PS_BANDS_MID:
794         hPsEncode->nQmfIidGroups    = QMF_GROUPS_LO_RES;
795         hPsEncode->nSubQmfIidGroups = SUBQMF_GROUPS_LO_RES;
796         FDKmemcpy(hPsEncode->iidGroupBorders,        iidGroupBordersLoRes, (hPsEncode->nQmfIidGroups + hPsEncode->nSubQmfIidGroups + 1)*sizeof(INT));
797         FDKmemcpy(hPsEncode->subband2parameterIndex, subband2parameter20,  (hPsEncode->nQmfIidGroups + hPsEncode->nSubQmfIidGroups)    *sizeof(INT));
798         FDKmemcpy(hPsEncode->iidGroupWidthLd,        iidGroupWidthLdLoRes, (hPsEncode->nQmfIidGroups + hPsEncode->nSubQmfIidGroups)    *sizeof(UCHAR));
799         break;
800       default:
801         error = PSENC_INIT_ERROR;
802         goto bail;
803     }
804 
805     hPsEncode->psEncMode = psEncMode;
806     hPsEncode->iidQuantErrorThreshold = iidQuantErrorThreshold;
807     FDKsbrEnc_initPsBandNrgScale(hPsEncode);
808   }
809 bail:
810   return error;
811 }
812 
813 
FDKsbrEnc_DestroyPSEncode(HANDLE_PS_ENCODE * phPsEncode)814 FDK_PSENC_ERROR FDKsbrEnc_DestroyPSEncode(
815         HANDLE_PS_ENCODE         *phPsEncode
816         )
817 {
818   FDK_PSENC_ERROR error = PSENC_OK;
819 
820   if (NULL !=phPsEncode) {
821     FreeRam_PsEncode(phPsEncode);
822   }
823 
824   return error;
825 }
826 
827 typedef struct {
828   FIXP_DBL pwrL[PS_MAX_ENVELOPES][PS_MAX_BANDS];
829   FIXP_DBL pwrR[PS_MAX_ENVELOPES][PS_MAX_BANDS];
830   FIXP_DBL ldPwrL[PS_MAX_ENVELOPES][PS_MAX_BANDS];
831   FIXP_DBL ldPwrR[PS_MAX_ENVELOPES][PS_MAX_BANDS];
832   FIXP_DBL pwrCr[PS_MAX_ENVELOPES][PS_MAX_BANDS];
833   FIXP_DBL pwrCi[PS_MAX_ENVELOPES][PS_MAX_BANDS];
834 
835 } PS_PWR_DATA;
836 
837 
FDKsbrEnc_PSEncode(HANDLE_PS_ENCODE hPsEncode,HANDLE_PS_OUT hPsOut,UCHAR * dynBandScale,UINT maxEnvelopes,FIXP_DBL * hybridData[HYBRID_FRAMESIZE][MAX_PS_CHANNELS][2],const INT frameSize,const INT sendHeader)838 FDK_PSENC_ERROR FDKsbrEnc_PSEncode(
839         HANDLE_PS_ENCODE          hPsEncode,
840         HANDLE_PS_OUT             hPsOut,
841         UCHAR                    *dynBandScale,
842         UINT                      maxEnvelopes,
843         FIXP_DBL                 *hybridData[HYBRID_FRAMESIZE][MAX_PS_CHANNELS][2],
844         const INT                 frameSize,
845         const INT                 sendHeader
846         )
847 {
848   FDK_PSENC_ERROR error = PSENC_OK;
849 
850   HANDLE_PS_DATA hPsData = &hPsEncode->psData;
851   FIXP_DBL iid [PS_MAX_ENVELOPES][PS_MAX_BANDS];
852   FIXP_DBL icc [PS_MAX_ENVELOPES][PS_MAX_BANDS];
853   int envBorder[PS_MAX_ENVELOPES+1];
854 
855   int group, bin, col, subband, band;
856   int i = 0;
857 
858   int env = 0;
859   int psBands      = (int) hPsEncode->psEncMode;
860   int nIidGroups   = hPsEncode->nQmfIidGroups + hPsEncode->nSubQmfIidGroups;
861   int nEnvelopes   = fixMin(maxEnvelopes, (UINT)PS_MAX_ENVELOPES);
862 
863   C_ALLOC_SCRATCH_START(pwrData, PS_PWR_DATA, 1);
864 
865   for(env=0; env<nEnvelopes+1;env++) {
866     envBorder[env] = fMultI(GetInvInt(nEnvelopes),frameSize*env);
867   }
868 
869   for(env=0; env<nEnvelopes;env++) {
870 
871     /* clear energy array */
872     for (band=0; band<psBands; band++) {
873       pwrData->pwrL[env][band] = pwrData->pwrR[env][band] = pwrData->pwrCr[env][band] = pwrData->pwrCi[env][band] = FIXP_DBL(1);
874     }
875 
876     /**** calculate energies and correlation ****/
877 
878     /* start with hybrid data */
879     for (group=0; group < nIidGroups; group++) {
880       /* Translate group to bin */
881       bin = hPsEncode->subband2parameterIndex[group];
882 
883       /* Translate from 20 bins to 10 bins */
884       if (hPsEncode->psEncMode == PS_BANDS_COARSE) {
885         bin >>= 1;
886       }
887 
888       /* determine group border */
889       int bScale = hPsEncode->psBandNrgScale[bin];
890 
891       FIXP_DBL pwrL_env_bin = pwrData->pwrL[env][bin];
892       FIXP_DBL pwrR_env_bin = pwrData->pwrR[env][bin];
893       FIXP_DBL pwrCr_env_bin = pwrData->pwrCr[env][bin];
894       FIXP_DBL pwrCi_env_bin = pwrData->pwrCi[env][bin];
895 
896       int scale = (int)dynBandScale[bin];
897       for (col=envBorder[env]; col<envBorder[env+1]; col++) {
898         for (subband = hPsEncode->iidGroupBorders[group]; subband < hPsEncode->iidGroupBorders[group+1]; subband++) {
899           FIXP_QMF l_real = (hybridData[col][0][0][subband]) << scale;
900           FIXP_QMF l_imag = (hybridData[col][0][1][subband]) << scale;
901           FIXP_QMF r_real = (hybridData[col][1][0][subband]) << scale;
902           FIXP_QMF r_imag = (hybridData[col][1][1][subband]) << scale;
903 
904           pwrL_env_bin  += (fPow2Div2(l_real) + fPow2Div2(l_imag)) >> bScale;
905           pwrR_env_bin  += (fPow2Div2(r_real) + fPow2Div2(r_imag)) >> bScale;
906           pwrCr_env_bin += (fMultDiv2(l_real, r_real) + fMultDiv2(l_imag, r_imag)) >> bScale;
907           pwrCi_env_bin += (fMultDiv2(r_real, l_imag) - fMultDiv2(l_real, r_imag)) >> bScale;
908         }
909       }
910       /* assure, nrg's of left and right channel are not negative; necessary on 16 bit multiply units */
911       pwrData->pwrL[env][bin] = fixMax((FIXP_DBL)0,pwrL_env_bin);
912       pwrData->pwrR[env][bin] = fixMax((FIXP_DBL)0,pwrR_env_bin);
913 
914       pwrData->pwrCr[env][bin] = pwrCr_env_bin;
915       pwrData->pwrCi[env][bin] = pwrCi_env_bin;
916 
917     } /* nIidGroups */
918 
919     /* calc logarithmic energy */
920     LdDataVector(pwrData->pwrL[env], pwrData->ldPwrL[env], psBands);
921     LdDataVector(pwrData->pwrR[env], pwrData->ldPwrR[env], psBands);
922 
923   } /* nEnvelopes */
924 
925   /* calculate iid and icc */
926   calculateIID(pwrData->ldPwrL, pwrData->ldPwrR, iid, nEnvelopes, psBands);
927   calculateICC(pwrData->ldPwrL, pwrData->ldPwrR, pwrData->pwrCr, pwrData->pwrCi, icc, nEnvelopes, psBands);
928 
929   /*** Envelope Reduction ***/
930   while (envelopeReducible(iid,icc,psBands,nEnvelopes)) {
931     int e=0;
932     /* sum energies of two neighboring envelopes */
933     nEnvelopes >>= 1;
934     for (e=0; e<nEnvelopes; e++) {
935       FDKsbrEnc_addFIXP_DBL(pwrData->pwrL[2*e], pwrData->pwrL[2*e+1], pwrData->pwrL[e], psBands);
936       FDKsbrEnc_addFIXP_DBL(pwrData->pwrR[2*e], pwrData->pwrR[2*e+1], pwrData->pwrR[e], psBands);
937       FDKsbrEnc_addFIXP_DBL(pwrData->pwrCr[2*e],pwrData->pwrCr[2*e+1],pwrData->pwrCr[e],psBands);
938       FDKsbrEnc_addFIXP_DBL(pwrData->pwrCi[2*e],pwrData->pwrCi[2*e+1],pwrData->pwrCi[e],psBands);
939 
940       /* calc logarithmic energy */
941       LdDataVector(pwrData->pwrL[e], pwrData->ldPwrL[e], psBands);
942       LdDataVector(pwrData->pwrR[e], pwrData->ldPwrR[e], psBands);
943 
944       /* reduce number of envelopes and adjust borders */
945       envBorder[e] = envBorder[2*e];
946     }
947     envBorder[nEnvelopes] = envBorder[2*nEnvelopes];
948 
949     /* re-calculate iid and icc */
950     calculateIID(pwrData->ldPwrL, pwrData->ldPwrR, iid, nEnvelopes, psBands);
951     calculateICC(pwrData->ldPwrL, pwrData->ldPwrR, pwrData->pwrCr, pwrData->pwrCi, icc, nEnvelopes, psBands);
952   }
953 
954 
955   /*  */
956   if(sendHeader) {
957     hPsData->headerCnt  = MAX_PS_NOHEADER_CNT;
958     hPsData->iidTimeCnt = MAX_TIME_DIFF_FRAMES;
959     hPsData->iccTimeCnt = MAX_TIME_DIFF_FRAMES;
960     hPsData->noEnvCnt   = MAX_NOENV_CNT;
961   }
962 
963   /*** Parameter processing, quantisation etc ***/
964   processIidData(hPsData, iid, psBands, nEnvelopes, hPsEncode->iidQuantErrorThreshold);
965   processIccData(hPsData, icc, psBands, nEnvelopes);
966 
967 
968   /*** Initialize output struct ***/
969 
970   /* PS Header on/off ? */
971   if( (hPsData->headerCnt<MAX_PS_NOHEADER_CNT)
972        && ( (hPsData->iidQuantMode == hPsData->iidQuantModeLast) && (hPsData->iccQuantMode == hPsData->iccQuantModeLast) )
973        && ( (hPsData->iidEnable    == hPsData->iidEnableLast)    && (hPsData->iccEnable    == hPsData->iccEnableLast)  ) ) {
974     hPsOut->enablePSHeader = 0;
975   }
976   else {
977     hPsOut->enablePSHeader = 1;
978     hPsData->headerCnt = 0;
979   }
980 
981   /* nEnvelopes = 0 ? */
982   if ( (hPsData->noEnvCnt < MAX_NOENV_CNT)
983        && (similarIid(hPsData, psBands, nEnvelopes))
984        && (similarIcc(hPsData, psBands, nEnvelopes)) ) {
985     hPsOut->nEnvelopes = nEnvelopes = 0;
986     hPsData->noEnvCnt++;
987   } else {
988     hPsData->noEnvCnt = 0;
989   }
990 
991 
992   if (nEnvelopes>0) {
993 
994     hPsOut->enableIID      = hPsData->iidEnable;
995     hPsOut->iidMode        = getIIDMode(psBands, hPsData->iidQuantMode);
996 
997     hPsOut->enableICC      = hPsData->iccEnable;
998     hPsOut->iccMode        = getICCMode(psBands, hPsData->iccQuantMode);
999 
1000     hPsOut->enableIpdOpd  = 0;
1001     hPsOut->frameClass    = 0;
1002     hPsOut->nEnvelopes    = nEnvelopes;
1003 
1004     for(env=0; env<nEnvelopes; env++) {
1005       hPsOut->frameBorder[env] = envBorder[env+1];
1006     }
1007 
1008     for(env=0; env<hPsOut->nEnvelopes; env++) {
1009       hPsOut->deltaIID[env] = (PS_DELTA)hPsData->iidDiffMode[env];
1010 
1011       for(band=0; band<psBands; band++) {
1012         hPsOut->iid[env][band] = hPsData->iidIdx[env][band];
1013       }
1014     }
1015 
1016     for(env=0; env<hPsOut->nEnvelopes; env++) {
1017       hPsOut->deltaICC[env] = (PS_DELTA)hPsData->iccDiffMode[env];
1018       for(band=0; band<psBands; band++) {
1019         hPsOut->icc[env][band] = hPsData->iccIdx[env][band];
1020       }
1021     }
1022 
1023     /* IPD OPD not supported right now */
1024     FDKmemclear(hPsOut->ipd, PS_MAX_ENVELOPES*PS_MAX_BANDS*sizeof(PS_DELTA));
1025     for(env=0; env<PS_MAX_ENVELOPES; env++) {
1026       hPsOut->deltaIPD[env] = PS_DELTA_FREQ;
1027       hPsOut->deltaOPD[env] = PS_DELTA_FREQ;
1028     }
1029 
1030     FDKmemclear(hPsOut->ipdLast, PS_MAX_BANDS*sizeof(INT));
1031     FDKmemclear(hPsOut->opdLast, PS_MAX_BANDS*sizeof(INT));
1032 
1033     for(band=0; band<PS_MAX_BANDS; band++) {
1034       hPsOut->iidLast[band] = hPsData->iidIdxLast[band];
1035       hPsOut->iccLast[band] = hPsData->iccIdxLast[band];
1036     }
1037 
1038     /* save iids and iccs for differential time coding in the next frame */
1039     hPsData->nEnvelopesLast   = nEnvelopes;
1040     hPsData->iidEnableLast    = hPsData->iidEnable;
1041     hPsData->iccEnableLast    = hPsData->iccEnable;
1042     hPsData->iidQuantModeLast = hPsData->iidQuantMode;
1043     hPsData->iccQuantModeLast = hPsData->iccQuantMode;
1044     for (i=0; i<psBands; i++) {
1045       hPsData->iidIdxLast[i] = hPsData->iidIdx[nEnvelopes-1][i];
1046       hPsData->iccIdxLast[i] = hPsData->iccIdx[nEnvelopes-1][i];
1047     }
1048   } /* Envelope > 0 */
1049 
1050   C_ALLOC_SCRATCH_END(pwrData, PS_PWR_DATA, 1)
1051 
1052   return error;
1053 }
1054 
1055