1 /* -----------------------------------------------------------------------------
2 Software License for The Fraunhofer FDK AAC Codec Library for Android
3
4 © Copyright 1995 - 2019 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 /******************* Library for basic calculation routines ********************
96
97 Author(s): Markus Lohwasser
98
99 Description: FDK Tools Hybrid Filterbank
100
101 *******************************************************************************/
102
103 #include "FDK_hybrid.h"
104
105 #include "fft.h"
106
107 /*--------------- defines -----------------------------*/
108 #define FFT_IDX_R(a) (2 * a)
109 #define FFT_IDX_I(a) (2 * a + 1)
110
111 #define HYB_COEF8_0 (0.00746082949812f)
112 #define HYB_COEF8_1 (0.02270420949825f)
113 #define HYB_COEF8_2 (0.04546865930473f)
114 #define HYB_COEF8_3 (0.07266113929591f)
115 #define HYB_COEF8_4 (0.09885108575264f)
116 #define HYB_COEF8_5 (0.11793710567217f)
117 #define HYB_COEF8_6 (0.12500000000000f)
118 #define HYB_COEF8_7 (HYB_COEF8_5)
119 #define HYB_COEF8_8 (HYB_COEF8_4)
120 #define HYB_COEF8_9 (HYB_COEF8_3)
121 #define HYB_COEF8_10 (HYB_COEF8_2)
122 #define HYB_COEF8_11 (HYB_COEF8_1)
123 #define HYB_COEF8_12 (HYB_COEF8_0)
124
125 /*--------------- structure definitions ---------------*/
126
127 #if defined(ARCH_PREFER_MULT_32x16)
128 #define FIXP_HTB FIXP_SGL /* SGL data type. */
129 #define FIXP_HTP FIXP_SPK /* Packed SGL data type. */
130 #define HTC(a) (FX_DBL2FXCONST_SGL(a)) /* Cast to SGL */
131 #define FL2FXCONST_HTB FL2FXCONST_SGL
132 #else
133 #define FIXP_HTB FIXP_DBL /* SGL data type. */
134 #define FIXP_HTP FIXP_DPK /* Packed DBL data type. */
135 #define HTC(a) ((FIXP_DBL)(LONG)(a)) /* Cast to DBL */
136 #define FL2FXCONST_HTB FL2FXCONST_DBL
137 #endif
138
139 #define HTCP(real, imag) \
140 { \
141 { HTC(real), HTC(imag) } \
142 } /* How to arrange the packed values. */
143
144 struct FDK_HYBRID_SETUP {
145 UCHAR nrQmfBands; /*!< Number of QMF bands to be converted to hybrid. */
146 UCHAR nHybBands[3]; /*!< Number of Hybrid bands generated by nrQmfBands. */
147 UCHAR synHybScale[3]; /*!< Headroom needed in hybrid synthesis filterbank. */
148 SCHAR kHybrid[3]; /*!< Filter configuration of each QMF band. */
149 UCHAR protoLen; /*!< Prototype filter length. */
150 UCHAR filterDelay; /*!< Delay caused by hybrid filter. */
151 const INT
152 *pReadIdxTable; /*!< Helper table to access input data ringbuffer. */
153 };
154
155 /*--------------- constants ---------------------------*/
156 static const INT ringbuffIdxTab[2 * 13] = {0, 1, 2, 3, 4, 5, 6, 7, 8,
157 9, 10, 11, 12, 0, 1, 2, 3, 4,
158 5, 6, 7, 8, 9, 10, 11, 12};
159
160 static const FDK_HYBRID_SETUP setup_3_16 = {
161 3, {8, 4, 4}, {4, 3, 3}, {8, 4, 4}, 13, (13 - 1) / 2, ringbuffIdxTab};
162 static const FDK_HYBRID_SETUP setup_3_12 = {
163 3, {8, 2, 2}, {4, 2, 2}, {8, 2, 2}, 13, (13 - 1) / 2, ringbuffIdxTab};
164 static const FDK_HYBRID_SETUP setup_3_10 = {
165 3, {6, 2, 2}, {3, 2, 2}, {-8, -2, 2}, 13, (13 - 1) / 2, ringbuffIdxTab};
166
167 static const FIXP_HTP HybFilterCoef8[] = {
168 HTCP(0x10000000, 0x00000000), HTCP(0x0df26407, 0xfa391882),
169 HTCP(0xff532109, 0x00acdef7), HTCP(0x08f26d36, 0xf70d92ca),
170 HTCP(0xfee34b5f, 0x02af570f), HTCP(0x038f276e, 0xf7684793),
171 HTCP(0x00000000, 0x05d1eac2), HTCP(0x00000000, 0x05d1eac2),
172 HTCP(0x038f276e, 0x0897b86d), HTCP(0xfee34b5f, 0xfd50a8f1),
173 HTCP(0x08f26d36, 0x08f26d36), HTCP(0xff532109, 0xff532109),
174 HTCP(0x0df26407, 0x05c6e77e)};
175
176 static const FIXP_HTB HybFilterCoef2[3] = {FL2FXCONST_HTB(0.01899487526049f),
177 FL2FXCONST_HTB(-0.07293139167538f),
178 FL2FXCONST_HTB(0.30596630545168f)};
179
180 static const FIXP_HTB HybFilterCoef4[13] = {FL2FXCONST_HTB(-0.00305151927305f),
181 FL2FXCONST_HTB(-0.00794862316203f),
182 FL2FXCONST_HTB(0.0f),
183 FL2FXCONST_HTB(0.04318924038756f),
184 FL2FXCONST_HTB(0.12542448210445f),
185 FL2FXCONST_HTB(0.21227807049160f),
186 FL2FXCONST_HTB(0.25f),
187 FL2FXCONST_HTB(0.21227807049160f),
188 FL2FXCONST_HTB(0.12542448210445f),
189 FL2FXCONST_HTB(0.04318924038756f),
190 FL2FXCONST_HTB(0.0f),
191 FL2FXCONST_HTB(-0.00794862316203f),
192 FL2FXCONST_HTB(-0.00305151927305f)};
193
194 /*--------------- function declarations ---------------*/
195 static INT kChannelFiltering(const FIXP_DBL *const pQmfReal,
196 const FIXP_DBL *const pQmfImag,
197 const INT *const pReadIdx,
198 FIXP_DBL *const mHybridReal,
199 FIXP_DBL *const mHybridImag,
200 const SCHAR hybridConfig);
201
202 /*--------------- function definitions ----------------*/
203
FDKhybridAnalysisOpen(HANDLE_FDK_ANA_HYB_FILTER hAnalysisHybFilter,FIXP_DBL * const pLFmemory,const UINT LFmemorySize,FIXP_DBL * const pHFmemory,const UINT HFmemorySize)204 INT FDKhybridAnalysisOpen(HANDLE_FDK_ANA_HYB_FILTER hAnalysisHybFilter,
205 FIXP_DBL *const pLFmemory, const UINT LFmemorySize,
206 FIXP_DBL *const pHFmemory, const UINT HFmemorySize) {
207 INT err = 0;
208
209 /* Save pointer to extern memory. */
210 hAnalysisHybFilter->pLFmemory = pLFmemory;
211 hAnalysisHybFilter->LFmemorySize = LFmemorySize;
212
213 hAnalysisHybFilter->pHFmemory = pHFmemory;
214 hAnalysisHybFilter->HFmemorySize = HFmemorySize;
215
216 return err;
217 }
218
FDKhybridAnalysisInit(HANDLE_FDK_ANA_HYB_FILTER hAnalysisHybFilter,const FDK_HYBRID_MODE mode,const INT qmfBands,const INT cplxBands,const INT initStatesFlag)219 INT FDKhybridAnalysisInit(HANDLE_FDK_ANA_HYB_FILTER hAnalysisHybFilter,
220 const FDK_HYBRID_MODE mode, const INT qmfBands,
221 const INT cplxBands, const INT initStatesFlag) {
222 int k;
223 INT err = 0;
224 FIXP_DBL *pMem = NULL;
225 HANDLE_FDK_HYBRID_SETUP setup = NULL;
226
227 switch (mode) {
228 case THREE_TO_TEN:
229 setup = &setup_3_10;
230 break;
231 case THREE_TO_TWELVE:
232 setup = &setup_3_12;
233 break;
234 case THREE_TO_SIXTEEN:
235 setup = &setup_3_16;
236 break;
237 default:
238 err = -1;
239 goto bail;
240 }
241
242 /* Initialize handle. */
243 hAnalysisHybFilter->pSetup = setup;
244 if (initStatesFlag) {
245 hAnalysisHybFilter->bufferLFpos = setup->protoLen - 1;
246 hAnalysisHybFilter->bufferHFpos = 0;
247 }
248 hAnalysisHybFilter->nrBands = qmfBands;
249 hAnalysisHybFilter->cplxBands = cplxBands;
250 hAnalysisHybFilter->hfMode = 0;
251
252 /* Check available memory. */
253 if (((2 * setup->nrQmfBands * setup->protoLen * sizeof(FIXP_DBL)) >
254 hAnalysisHybFilter->LFmemorySize)) {
255 err = -2;
256 goto bail;
257 }
258 if (hAnalysisHybFilter->HFmemorySize != 0) {
259 if (((setup->filterDelay *
260 ((qmfBands - setup->nrQmfBands) + (cplxBands - setup->nrQmfBands)) *
261 sizeof(FIXP_DBL)) > hAnalysisHybFilter->HFmemorySize)) {
262 err = -3;
263 goto bail;
264 }
265 }
266
267 /* Distribute LF memory. */
268 pMem = hAnalysisHybFilter->pLFmemory;
269 for (k = 0; k < setup->nrQmfBands; k++) {
270 hAnalysisHybFilter->bufferLFReal[k] = pMem;
271 pMem += setup->protoLen;
272 hAnalysisHybFilter->bufferLFImag[k] = pMem;
273 pMem += setup->protoLen;
274 }
275
276 /* Distribute HF memory. */
277 if (hAnalysisHybFilter->HFmemorySize != 0) {
278 pMem = hAnalysisHybFilter->pHFmemory;
279 for (k = 0; k < setup->filterDelay; k++) {
280 hAnalysisHybFilter->bufferHFReal[k] = pMem;
281 pMem += (qmfBands - setup->nrQmfBands);
282 hAnalysisHybFilter->bufferHFImag[k] = pMem;
283 pMem += (cplxBands - setup->nrQmfBands);
284 }
285 }
286
287 if (initStatesFlag) {
288 /* Clear LF buffer */
289 for (k = 0; k < setup->nrQmfBands; k++) {
290 FDKmemclear(hAnalysisHybFilter->bufferLFReal[k],
291 setup->protoLen * sizeof(FIXP_DBL));
292 FDKmemclear(hAnalysisHybFilter->bufferLFImag[k],
293 setup->protoLen * sizeof(FIXP_DBL));
294 }
295
296 if (hAnalysisHybFilter->HFmemorySize != 0) {
297 if (qmfBands > setup->nrQmfBands) {
298 /* Clear HF buffer */
299 for (k = 0; k < setup->filterDelay; k++) {
300 FDKmemclear(hAnalysisHybFilter->bufferHFReal[k],
301 (qmfBands - setup->nrQmfBands) * sizeof(FIXP_DBL));
302 FDKmemclear(hAnalysisHybFilter->bufferHFImag[k],
303 (cplxBands - setup->nrQmfBands) * sizeof(FIXP_DBL));
304 }
305 }
306 }
307 }
308
309 bail:
310 return err;
311 }
312
FDKhybridAnalysisScaleStates(HANDLE_FDK_ANA_HYB_FILTER hAnalysisHybFilter,const INT scalingValue)313 INT FDKhybridAnalysisScaleStates(HANDLE_FDK_ANA_HYB_FILTER hAnalysisHybFilter,
314 const INT scalingValue) {
315 INT err = 0;
316
317 if (hAnalysisHybFilter == NULL) {
318 err = 1; /* invalid handle */
319 } else {
320 int k;
321 HANDLE_FDK_HYBRID_SETUP setup = hAnalysisHybFilter->pSetup;
322
323 /* Scale LF buffer */
324 for (k = 0; k < setup->nrQmfBands; k++) {
325 scaleValues(hAnalysisHybFilter->bufferLFReal[k], setup->protoLen,
326 scalingValue);
327 scaleValues(hAnalysisHybFilter->bufferLFImag[k], setup->protoLen,
328 scalingValue);
329 }
330 if (hAnalysisHybFilter->nrBands > setup->nrQmfBands) {
331 /* Scale HF buffer */
332 for (k = 0; k < setup->filterDelay; k++) {
333 scaleValues(hAnalysisHybFilter->bufferHFReal[k],
334 (hAnalysisHybFilter->nrBands - setup->nrQmfBands),
335 scalingValue);
336 scaleValues(hAnalysisHybFilter->bufferHFImag[k],
337 (hAnalysisHybFilter->cplxBands - setup->nrQmfBands),
338 scalingValue);
339 }
340 }
341 }
342 return err;
343 }
344
FDKhybridAnalysisApply(HANDLE_FDK_ANA_HYB_FILTER hAnalysisHybFilter,const FIXP_DBL * const pQmfReal,const FIXP_DBL * const pQmfImag,FIXP_DBL * const pHybridReal,FIXP_DBL * const pHybridImag)345 INT FDKhybridAnalysisApply(HANDLE_FDK_ANA_HYB_FILTER hAnalysisHybFilter,
346 const FIXP_DBL *const pQmfReal,
347 const FIXP_DBL *const pQmfImag,
348 FIXP_DBL *const pHybridReal,
349 FIXP_DBL *const pHybridImag) {
350 int k, hybOffset = 0;
351 INT err = 0;
352 const int nrQmfBandsLF =
353 hAnalysisHybFilter->pSetup
354 ->nrQmfBands; /* number of QMF bands to be converted to hybrid */
355
356 const int writIndex = hAnalysisHybFilter->bufferLFpos;
357 int readIndex = hAnalysisHybFilter->bufferLFpos;
358
359 if (++readIndex >= hAnalysisHybFilter->pSetup->protoLen) readIndex = 0;
360 const INT *pBufferLFreadIdx =
361 &hAnalysisHybFilter->pSetup->pReadIdxTable[readIndex];
362
363 /*
364 * LF buffer.
365 */
366 for (k = 0; k < nrQmfBandsLF; k++) {
367 /* New input sample. */
368 hAnalysisHybFilter->bufferLFReal[k][writIndex] = pQmfReal[k];
369 hAnalysisHybFilter->bufferLFImag[k][writIndex] = pQmfImag[k];
370
371 /* Perform hybrid filtering. */
372 err |=
373 kChannelFiltering(hAnalysisHybFilter->bufferLFReal[k],
374 hAnalysisHybFilter->bufferLFImag[k], pBufferLFreadIdx,
375 pHybridReal + hybOffset, pHybridImag + hybOffset,
376 hAnalysisHybFilter->pSetup->kHybrid[k]);
377
378 hybOffset += hAnalysisHybFilter->pSetup->nHybBands[k];
379 }
380
381 hAnalysisHybFilter->bufferLFpos =
382 readIndex; /* Index where to write next input sample. */
383
384 if (hAnalysisHybFilter->nrBands > nrQmfBandsLF) {
385 /*
386 * HF buffer.
387 */
388 if (hAnalysisHybFilter->hfMode != 0) {
389 /* HF delay compensation was applied outside. */
390 FDKmemcpy(
391 pHybridReal + hybOffset, &pQmfReal[nrQmfBandsLF],
392 (hAnalysisHybFilter->nrBands - nrQmfBandsLF) * sizeof(FIXP_DBL));
393 FDKmemcpy(
394 pHybridImag + hybOffset, &pQmfImag[nrQmfBandsLF],
395 (hAnalysisHybFilter->cplxBands - nrQmfBandsLF) * sizeof(FIXP_DBL));
396 } else {
397 FDK_ASSERT(hAnalysisHybFilter->HFmemorySize != 0);
398 /* HF delay compensation, filterlength/2. */
399 FDKmemcpy(
400 pHybridReal + hybOffset,
401 hAnalysisHybFilter->bufferHFReal[hAnalysisHybFilter->bufferHFpos],
402 (hAnalysisHybFilter->nrBands - nrQmfBandsLF) * sizeof(FIXP_DBL));
403 FDKmemcpy(
404 pHybridImag + hybOffset,
405 hAnalysisHybFilter->bufferHFImag[hAnalysisHybFilter->bufferHFpos],
406 (hAnalysisHybFilter->cplxBands - nrQmfBandsLF) * sizeof(FIXP_DBL));
407
408 FDKmemcpy(
409 hAnalysisHybFilter->bufferHFReal[hAnalysisHybFilter->bufferHFpos],
410 &pQmfReal[nrQmfBandsLF],
411 (hAnalysisHybFilter->nrBands - nrQmfBandsLF) * sizeof(FIXP_DBL));
412 FDKmemcpy(
413 hAnalysisHybFilter->bufferHFImag[hAnalysisHybFilter->bufferHFpos],
414 &pQmfImag[nrQmfBandsLF],
415 (hAnalysisHybFilter->cplxBands - nrQmfBandsLF) * sizeof(FIXP_DBL));
416
417 if (++hAnalysisHybFilter->bufferHFpos >=
418 hAnalysisHybFilter->pSetup->filterDelay)
419 hAnalysisHybFilter->bufferHFpos = 0;
420 }
421 } /* process HF part*/
422
423 return err;
424 }
425
FDKhybridAnalysisClose(HANDLE_FDK_ANA_HYB_FILTER hAnalysisHybFilter)426 INT FDKhybridAnalysisClose(HANDLE_FDK_ANA_HYB_FILTER hAnalysisHybFilter) {
427 INT err = 0;
428
429 if (hAnalysisHybFilter != NULL) {
430 hAnalysisHybFilter->pLFmemory = NULL;
431 hAnalysisHybFilter->pHFmemory = NULL;
432 hAnalysisHybFilter->LFmemorySize = 0;
433 hAnalysisHybFilter->HFmemorySize = 0;
434 }
435
436 return err;
437 }
438
FDKhybridSynthesisInit(HANDLE_FDK_SYN_HYB_FILTER hSynthesisHybFilter,const FDK_HYBRID_MODE mode,const INT qmfBands,const INT cplxBands)439 INT FDKhybridSynthesisInit(HANDLE_FDK_SYN_HYB_FILTER hSynthesisHybFilter,
440 const FDK_HYBRID_MODE mode, const INT qmfBands,
441 const INT cplxBands) {
442 INT err = 0;
443 HANDLE_FDK_HYBRID_SETUP setup = NULL;
444
445 switch (mode) {
446 case THREE_TO_TEN:
447 setup = &setup_3_10;
448 break;
449 case THREE_TO_TWELVE:
450 setup = &setup_3_12;
451 break;
452 case THREE_TO_SIXTEEN:
453 setup = &setup_3_16;
454 break;
455 default:
456 err = -1;
457 goto bail;
458 }
459
460 hSynthesisHybFilter->pSetup = setup;
461 hSynthesisHybFilter->nrBands = qmfBands;
462 hSynthesisHybFilter->cplxBands = cplxBands;
463
464 bail:
465 return err;
466 }
467
FDKhybridSynthesisApply(HANDLE_FDK_SYN_HYB_FILTER hSynthesisHybFilter,const FIXP_DBL * const pHybridReal,const FIXP_DBL * const pHybridImag,FIXP_DBL * const pQmfReal,FIXP_DBL * const pQmfImag)468 void FDKhybridSynthesisApply(HANDLE_FDK_SYN_HYB_FILTER hSynthesisHybFilter,
469 const FIXP_DBL *const pHybridReal,
470 const FIXP_DBL *const pHybridImag,
471 FIXP_DBL *const pQmfReal,
472 FIXP_DBL *const pQmfImag) {
473 int k, n, hybOffset = 0;
474 const INT nrQmfBandsLF = hSynthesisHybFilter->pSetup->nrQmfBands;
475
476 /*
477 * LF buffer.
478 */
479 for (k = 0; k < nrQmfBandsLF; k++) {
480 const int nHybBands = hSynthesisHybFilter->pSetup->nHybBands[k];
481 const int scale = hSynthesisHybFilter->pSetup->synHybScale[k];
482
483 FIXP_DBL accu1 = FL2FXCONST_DBL(0.f);
484 FIXP_DBL accu2 = FL2FXCONST_DBL(0.f);
485
486 /* Perform hybrid filtering. */
487 for (n = 0; n < nHybBands; n++) {
488 accu1 += pHybridReal[hybOffset + n] >> scale;
489 accu2 += pHybridImag[hybOffset + n] >> scale;
490 }
491 pQmfReal[k] = SATURATE_LEFT_SHIFT(accu1, scale, DFRACT_BITS);
492 pQmfImag[k] = SATURATE_LEFT_SHIFT(accu2, scale, DFRACT_BITS);
493
494 hybOffset += nHybBands;
495 }
496
497 if (hSynthesisHybFilter->nrBands > nrQmfBandsLF) {
498 /*
499 * HF buffer.
500 */
501 FDKmemcpy(&pQmfReal[nrQmfBandsLF], &pHybridReal[hybOffset],
502 (hSynthesisHybFilter->nrBands - nrQmfBandsLF) * sizeof(FIXP_DBL));
503 FDKmemcpy(
504 &pQmfImag[nrQmfBandsLF], &pHybridImag[hybOffset],
505 (hSynthesisHybFilter->cplxBands - nrQmfBandsLF) * sizeof(FIXP_DBL));
506 }
507
508 return;
509 }
510
dualChannelFiltering(const FIXP_DBL * const pQmfReal,const FIXP_DBL * const pQmfImag,const INT * const pReadIdx,FIXP_DBL * const mHybridReal,FIXP_DBL * const mHybridImag,const INT invert)511 static void dualChannelFiltering(const FIXP_DBL *const pQmfReal,
512 const FIXP_DBL *const pQmfImag,
513 const INT *const pReadIdx,
514 FIXP_DBL *const mHybridReal,
515 FIXP_DBL *const mHybridImag,
516 const INT invert) {
517 FIXP_DBL r1, r6;
518 FIXP_DBL i1, i6;
519
520 const FIXP_HTB f0 = HybFilterCoef2[0]; /* corresponds to p1 and p11 */
521 const FIXP_HTB f1 = HybFilterCoef2[1]; /* corresponds to p3 and p9 */
522 const FIXP_HTB f2 = HybFilterCoef2[2]; /* corresponds to p5 and p7 */
523
524 /* symmetric filter coefficients */
525 r1 = fMultDiv2(f0, pQmfReal[pReadIdx[1]]) +
526 fMultDiv2(f0, pQmfReal[pReadIdx[11]]);
527 i1 = fMultDiv2(f0, pQmfImag[pReadIdx[1]]) +
528 fMultDiv2(f0, pQmfImag[pReadIdx[11]]);
529 r1 += fMultDiv2(f1, pQmfReal[pReadIdx[3]]) +
530 fMultDiv2(f1, pQmfReal[pReadIdx[9]]);
531 i1 += fMultDiv2(f1, pQmfImag[pReadIdx[3]]) +
532 fMultDiv2(f1, pQmfImag[pReadIdx[9]]);
533 r1 += fMultDiv2(f2, pQmfReal[pReadIdx[5]]) +
534 fMultDiv2(f2, pQmfReal[pReadIdx[7]]);
535 i1 += fMultDiv2(f2, pQmfImag[pReadIdx[5]]) +
536 fMultDiv2(f2, pQmfImag[pReadIdx[7]]);
537
538 r6 = pQmfReal[pReadIdx[6]] >> 2;
539 i6 = pQmfImag[pReadIdx[6]] >> 2;
540
541 FDK_ASSERT((invert == 0) || (invert == 1));
542 mHybridReal[0 + invert] = (r6 + r1) << 1;
543 mHybridImag[0 + invert] = (i6 + i1) << 1;
544
545 mHybridReal[1 - invert] = (r6 - r1) << 1;
546 mHybridImag[1 - invert] = (i6 - i1) << 1;
547 }
548
fourChannelFiltering(const FIXP_DBL * const pQmfReal,const FIXP_DBL * const pQmfImag,const INT * const pReadIdx,FIXP_DBL * const mHybridReal,FIXP_DBL * const mHybridImag,const INT invert)549 static void fourChannelFiltering(const FIXP_DBL *const pQmfReal,
550 const FIXP_DBL *const pQmfImag,
551 const INT *const pReadIdx,
552 FIXP_DBL *const mHybridReal,
553 FIXP_DBL *const mHybridImag,
554 const INT invert) {
555 const FIXP_HTB *p = HybFilterCoef4;
556
557 FIXP_DBL fft[8];
558
559 static const FIXP_DBL cr[13] = {
560 FL2FXCONST_DBL(0.f), FL2FXCONST_DBL(-0.70710678118655f),
561 FL2FXCONST_DBL(-1.f), FL2FXCONST_DBL(-0.70710678118655f),
562 FL2FXCONST_DBL(0.f), FL2FXCONST_DBL(0.70710678118655f),
563 FL2FXCONST_DBL(1.f), FL2FXCONST_DBL(0.70710678118655f),
564 FL2FXCONST_DBL(0.f), FL2FXCONST_DBL(-0.70710678118655f),
565 FL2FXCONST_DBL(-1.f), FL2FXCONST_DBL(-0.70710678118655f),
566 FL2FXCONST_DBL(0.f)};
567 static const FIXP_DBL ci[13] = {
568 FL2FXCONST_DBL(-1.f), FL2FXCONST_DBL(-0.70710678118655f),
569 FL2FXCONST_DBL(0.f), FL2FXCONST_DBL(0.70710678118655f),
570 FL2FXCONST_DBL(1.f), FL2FXCONST_DBL(0.70710678118655f),
571 FL2FXCONST_DBL(0.f), FL2FXCONST_DBL(-0.70710678118655f),
572 FL2FXCONST_DBL(-1.f), FL2FXCONST_DBL(-0.70710678118655f),
573 FL2FXCONST_DBL(0.f), FL2FXCONST_DBL(0.70710678118655f),
574 FL2FXCONST_DBL(1.f)};
575
576 /* FIR filter. */
577 /* pre twiddeling with pre-twiddling coefficients c[n] */
578 /* multiplication with filter coefficients p[n] */
579 /* hint: (a + ib)*(c + id) = (a*c - b*d) + i(a*d + b*c) */
580 /* write to fft coefficient n' */
581 fft[FFT_IDX_R(0)] =
582 (fMult(p[10], (fMultSub(fMultDiv2(cr[2], pQmfReal[pReadIdx[2]]), ci[2],
583 pQmfImag[pReadIdx[2]]))) +
584 fMult(p[6], (fMultSub(fMultDiv2(cr[6], pQmfReal[pReadIdx[6]]), ci[6],
585 pQmfImag[pReadIdx[6]]))) +
586 fMult(p[2], (fMultSub(fMultDiv2(cr[10], pQmfReal[pReadIdx[10]]), ci[10],
587 pQmfImag[pReadIdx[10]]))));
588 fft[FFT_IDX_I(0)] =
589 (fMult(p[10], (fMultAdd(fMultDiv2(ci[2], pQmfReal[pReadIdx[2]]), cr[2],
590 pQmfImag[pReadIdx[2]]))) +
591 fMult(p[6], (fMultAdd(fMultDiv2(ci[6], pQmfReal[pReadIdx[6]]), cr[6],
592 pQmfImag[pReadIdx[6]]))) +
593 fMult(p[2], (fMultAdd(fMultDiv2(ci[10], pQmfReal[pReadIdx[10]]), cr[10],
594 pQmfImag[pReadIdx[10]]))));
595
596 /* twiddle dee dum */
597 fft[FFT_IDX_R(1)] =
598 (fMult(p[9], (fMultSub(fMultDiv2(cr[3], pQmfReal[pReadIdx[3]]), ci[3],
599 pQmfImag[pReadIdx[3]]))) +
600 fMult(p[5], (fMultSub(fMultDiv2(cr[7], pQmfReal[pReadIdx[7]]), ci[7],
601 pQmfImag[pReadIdx[7]]))) +
602 fMult(p[1], (fMultSub(fMultDiv2(cr[11], pQmfReal[pReadIdx[11]]), ci[11],
603 pQmfImag[pReadIdx[11]]))));
604 fft[FFT_IDX_I(1)] =
605 (fMult(p[9], (fMultAdd(fMultDiv2(ci[3], pQmfReal[pReadIdx[3]]), cr[3],
606 pQmfImag[pReadIdx[3]]))) +
607 fMult(p[5], (fMultAdd(fMultDiv2(ci[7], pQmfReal[pReadIdx[7]]), cr[7],
608 pQmfImag[pReadIdx[7]]))) +
609 fMult(p[1], (fMultAdd(fMultDiv2(ci[11], pQmfReal[pReadIdx[11]]), cr[11],
610 pQmfImag[pReadIdx[11]]))));
611
612 /* twiddle dee dee */
613 fft[FFT_IDX_R(2)] =
614 (fMult(p[12], (fMultSub(fMultDiv2(cr[0], pQmfReal[pReadIdx[0]]), ci[0],
615 pQmfImag[pReadIdx[0]]))) +
616 fMult(p[8], (fMultSub(fMultDiv2(cr[4], pQmfReal[pReadIdx[4]]), ci[4],
617 pQmfImag[pReadIdx[4]]))) +
618 fMult(p[4], (fMultSub(fMultDiv2(cr[8], pQmfReal[pReadIdx[8]]), ci[8],
619 pQmfImag[pReadIdx[8]]))) +
620 fMult(p[0], (fMultSub(fMultDiv2(cr[12], pQmfReal[pReadIdx[12]]), ci[12],
621 pQmfImag[pReadIdx[12]]))));
622 fft[FFT_IDX_I(2)] =
623 (fMult(p[12], (fMultAdd(fMultDiv2(ci[0], pQmfReal[pReadIdx[0]]), cr[0],
624 pQmfImag[pReadIdx[0]]))) +
625 fMult(p[8], (fMultAdd(fMultDiv2(ci[4], pQmfReal[pReadIdx[4]]), cr[4],
626 pQmfImag[pReadIdx[4]]))) +
627 fMult(p[4], (fMultAdd(fMultDiv2(ci[8], pQmfReal[pReadIdx[8]]), cr[8],
628 pQmfImag[pReadIdx[8]]))) +
629 fMult(p[0], (fMultAdd(fMultDiv2(ci[12], pQmfReal[pReadIdx[12]]), cr[12],
630 pQmfImag[pReadIdx[12]]))));
631
632 fft[FFT_IDX_R(3)] =
633 (fMult(p[11], (fMultSub(fMultDiv2(cr[1], pQmfReal[pReadIdx[1]]), ci[1],
634 pQmfImag[pReadIdx[1]]))) +
635 fMult(p[7], (fMultSub(fMultDiv2(cr[5], pQmfReal[pReadIdx[5]]), ci[5],
636 pQmfImag[pReadIdx[5]]))) +
637 fMult(p[3], (fMultSub(fMultDiv2(cr[9], pQmfReal[pReadIdx[9]]), ci[9],
638 pQmfImag[pReadIdx[9]]))));
639 fft[FFT_IDX_I(3)] =
640 (fMult(p[11], (fMultAdd(fMultDiv2(ci[1], pQmfReal[pReadIdx[1]]), cr[1],
641 pQmfImag[pReadIdx[1]]))) +
642 fMult(p[7], (fMultAdd(fMultDiv2(ci[5], pQmfReal[pReadIdx[5]]), cr[5],
643 pQmfImag[pReadIdx[5]]))) +
644 fMult(p[3], (fMultAdd(fMultDiv2(ci[9], pQmfReal[pReadIdx[9]]), cr[9],
645 pQmfImag[pReadIdx[9]]))));
646
647 /* fft modulation */
648 /* here: fast manual fft modulation for a fft of length M=4 */
649 /* fft_4{x[n]} = x[0]*exp(-i*2*pi/4*m*0) + x[1]*exp(-i*2*pi/4*m*1) +
650 x[2]*exp(-i*2*pi/4*m*2) + x[3]*exp(-i*2*pi/4*m*3) */
651
652 /*
653 fft bin m=0:
654 X[0, n] = x[0] + x[1] + x[2] + x[3]
655 */
656 mHybridReal[0] = fft[FFT_IDX_R(0)] + fft[FFT_IDX_R(1)] + fft[FFT_IDX_R(2)] +
657 fft[FFT_IDX_R(3)];
658 mHybridImag[0] = fft[FFT_IDX_I(0)] + fft[FFT_IDX_I(1)] + fft[FFT_IDX_I(2)] +
659 fft[FFT_IDX_I(3)];
660
661 /*
662 fft bin m=1:
663 X[1, n] = x[0] - i*x[1] - x[2] + i*x[3]
664 */
665 mHybridReal[1] = fft[FFT_IDX_R(0)] + fft[FFT_IDX_I(1)] - fft[FFT_IDX_R(2)] -
666 fft[FFT_IDX_I(3)];
667 mHybridImag[1] = fft[FFT_IDX_I(0)] - fft[FFT_IDX_R(1)] - fft[FFT_IDX_I(2)] +
668 fft[FFT_IDX_R(3)];
669
670 /*
671 fft bin m=2:
672 X[2, n] = x[0] - x[1] + x[2] - x[3]
673 */
674 mHybridReal[2] = fft[FFT_IDX_R(0)] - fft[FFT_IDX_R(1)] + fft[FFT_IDX_R(2)] -
675 fft[FFT_IDX_R(3)];
676 mHybridImag[2] = fft[FFT_IDX_I(0)] - fft[FFT_IDX_I(1)] + fft[FFT_IDX_I(2)] -
677 fft[FFT_IDX_I(3)];
678
679 /*
680 fft bin m=3:
681 X[3, n] = x[0] + j*x[1] - x[2] - j*x[3]
682 */
683 mHybridReal[3] = fft[FFT_IDX_R(0)] - fft[FFT_IDX_I(1)] - fft[FFT_IDX_R(2)] +
684 fft[FFT_IDX_I(3)];
685 mHybridImag[3] = fft[FFT_IDX_I(0)] + fft[FFT_IDX_R(1)] - fft[FFT_IDX_I(2)] -
686 fft[FFT_IDX_R(3)];
687 }
688
eightChannelFiltering(const FIXP_DBL * const pQmfReal,const FIXP_DBL * const pQmfImag,const INT * const pReadIdx,FIXP_DBL * const mHybridReal,FIXP_DBL * const mHybridImag,const INT invert)689 static void eightChannelFiltering(const FIXP_DBL *const pQmfReal,
690 const FIXP_DBL *const pQmfImag,
691 const INT *const pReadIdx,
692 FIXP_DBL *const mHybridReal,
693 FIXP_DBL *const mHybridImag,
694 const INT invert) {
695 const FIXP_HTP *p = HybFilterCoef8;
696 INT k, sc;
697
698 FIXP_DBL mfft[16 + ALIGNMENT_DEFAULT];
699 FIXP_DBL *pfft = (FIXP_DBL *)ALIGN_PTR(mfft);
700
701 FIXP_DBL accu1, accu2, accu3, accu4;
702
703 /* pre twiddeling */
704 pfft[FFT_IDX_R(0)] =
705 pQmfReal[pReadIdx[6]] >>
706 (3 + 1); /* fMultDiv2(p[0].v.re, pQmfReal[pReadIdx[6]]); */
707 pfft[FFT_IDX_I(0)] =
708 pQmfImag[pReadIdx[6]] >>
709 (3 + 1); /* fMultDiv2(p[0].v.re, pQmfImag[pReadIdx[6]]); */
710
711 cplxMultDiv2(&accu1, &accu2, pQmfReal[pReadIdx[7]], pQmfImag[pReadIdx[7]],
712 p[1]);
713 pfft[FFT_IDX_R(1)] = accu1;
714 pfft[FFT_IDX_I(1)] = accu2;
715
716 cplxMultDiv2(&accu1, &accu2, pQmfReal[pReadIdx[0]], pQmfImag[pReadIdx[0]],
717 p[2]);
718 cplxMultDiv2(&accu3, &accu4, pQmfReal[pReadIdx[8]], pQmfImag[pReadIdx[8]],
719 p[3]);
720 pfft[FFT_IDX_R(2)] = accu1 + accu3;
721 pfft[FFT_IDX_I(2)] = accu2 + accu4;
722
723 cplxMultDiv2(&accu1, &accu2, pQmfReal[pReadIdx[1]], pQmfImag[pReadIdx[1]],
724 p[4]);
725 cplxMultDiv2(&accu3, &accu4, pQmfReal[pReadIdx[9]], pQmfImag[pReadIdx[9]],
726 p[5]);
727 pfft[FFT_IDX_R(3)] = accu1 + accu3;
728 pfft[FFT_IDX_I(3)] = accu2 + accu4;
729
730 pfft[FFT_IDX_R(4)] = fMultDiv2(pQmfImag[pReadIdx[10]], p[7].v.im) -
731 fMultDiv2(pQmfImag[pReadIdx[2]], p[6].v.im);
732 pfft[FFT_IDX_I(4)] = fMultDiv2(pQmfReal[pReadIdx[2]], p[6].v.im) -
733 fMultDiv2(pQmfReal[pReadIdx[10]], p[7].v.im);
734
735 cplxMultDiv2(&accu1, &accu2, pQmfReal[pReadIdx[3]], pQmfImag[pReadIdx[3]],
736 p[8]);
737 cplxMultDiv2(&accu3, &accu4, pQmfReal[pReadIdx[11]], pQmfImag[pReadIdx[11]],
738 p[9]);
739 pfft[FFT_IDX_R(5)] = accu1 + accu3;
740 pfft[FFT_IDX_I(5)] = accu2 + accu4;
741
742 cplxMultDiv2(&accu1, &accu2, pQmfReal[pReadIdx[4]], pQmfImag[pReadIdx[4]],
743 p[10]);
744 cplxMultDiv2(&accu3, &accu4, pQmfReal[pReadIdx[12]], pQmfImag[pReadIdx[12]],
745 p[11]);
746 pfft[FFT_IDX_R(6)] = accu1 + accu3;
747 pfft[FFT_IDX_I(6)] = accu2 + accu4;
748
749 cplxMultDiv2(&accu1, &accu2, pQmfReal[pReadIdx[5]], pQmfImag[pReadIdx[5]],
750 p[12]);
751 pfft[FFT_IDX_R(7)] = accu1;
752 pfft[FFT_IDX_I(7)] = accu2;
753
754 /* fft modulation */
755 fft_8(pfft);
756 sc = 1 + 2;
757
758 if (invert) {
759 mHybridReal[0] = pfft[FFT_IDX_R(7)] << sc;
760 mHybridImag[0] = pfft[FFT_IDX_I(7)] << sc;
761 mHybridReal[1] = pfft[FFT_IDX_R(0)] << sc;
762 mHybridImag[1] = pfft[FFT_IDX_I(0)] << sc;
763
764 mHybridReal[2] = pfft[FFT_IDX_R(6)] << sc;
765 mHybridImag[2] = pfft[FFT_IDX_I(6)] << sc;
766 mHybridReal[3] = pfft[FFT_IDX_R(1)] << sc;
767 mHybridImag[3] = pfft[FFT_IDX_I(1)] << sc;
768
769 mHybridReal[4] = pfft[FFT_IDX_R(2)] << sc;
770 mHybridReal[4] += pfft[FFT_IDX_R(5)] << sc;
771 mHybridImag[4] = pfft[FFT_IDX_I(2)] << sc;
772 mHybridImag[4] += pfft[FFT_IDX_I(5)] << sc;
773
774 mHybridReal[5] = pfft[FFT_IDX_R(3)] << sc;
775 mHybridReal[5] += pfft[FFT_IDX_R(4)] << sc;
776 mHybridImag[5] = pfft[FFT_IDX_I(3)] << sc;
777 mHybridImag[5] += pfft[FFT_IDX_I(4)] << sc;
778 } else {
779 for (k = 0; k < 8; k++) {
780 mHybridReal[k] = pfft[FFT_IDX_R(k)] << sc;
781 mHybridImag[k] = pfft[FFT_IDX_I(k)] << sc;
782 }
783 }
784 }
785
kChannelFiltering(const FIXP_DBL * const pQmfReal,const FIXP_DBL * const pQmfImag,const INT * const pReadIdx,FIXP_DBL * const mHybridReal,FIXP_DBL * const mHybridImag,const SCHAR hybridConfig)786 static INT kChannelFiltering(const FIXP_DBL *const pQmfReal,
787 const FIXP_DBL *const pQmfImag,
788 const INT *const pReadIdx,
789 FIXP_DBL *const mHybridReal,
790 FIXP_DBL *const mHybridImag,
791 const SCHAR hybridConfig) {
792 INT err = 0;
793
794 switch (hybridConfig) {
795 case 2:
796 case -2:
797 dualChannelFiltering(pQmfReal, pQmfImag, pReadIdx, mHybridReal,
798 mHybridImag, (hybridConfig < 0) ? 1 : 0);
799 break;
800 case 4:
801 case -4:
802 fourChannelFiltering(pQmfReal, pQmfImag, pReadIdx, mHybridReal,
803 mHybridImag, (hybridConfig < 0) ? 1 : 0);
804 break;
805 case 8:
806 case -8:
807 eightChannelFiltering(pQmfReal, pQmfImag, pReadIdx, mHybridReal,
808 mHybridImag, (hybridConfig < 0) ? 1 : 0);
809 break;
810 default:
811 err = -1;
812 }
813
814 return err;
815 }
816