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 /**************************** FDK PCM utils module **************************
85
86 Author(s): Christian Griebel
87 Description: Defines functions to interface with the PCM post processing
88 module.
89
90 *******************************************************************************/
91
92 #include "pcmutils_lib.h"
93
94 #include "genericStds.h"
95 #include "fixpoint_math.h"
96
97 /* Decoder library info */
98 #define PCMDMX_LIB_VL0 2
99 #define PCMDMX_LIB_VL1 3
100 #define PCMDMX_LIB_VL2 1
101 #define PCMDMX_LIB_TITLE "PCM Downmix Lib"
102 #define PCMDMX_LIB_BUILD_DATE __DATE__
103 #define PCMDMX_LIB_BUILD_TIME __TIME__
104
105 /* Library settings */
106 #define PCM_DMX_MAX_DELAY_FRAMES ( 1 )
107 #define PCM_DMX_MAX_CHANNELS ( 8 )
108 #define PCM_DMX_MAX_CHANNEL_GROUPS ( 4 )
109 #define PCM_DMX_MAX_CHANNELS_PER_GROUP ( 3 ) /* The maximum over all groups */
110 #define PCMDMX_DFLT_EXPIRY_FRAME ( 50 ) /* At least 500ms (FL 960 @ 96kHz) */
111
112 /* Fixed and unique channel group indices.
113 * The last group index has to be smaller than PCM_DMX_MAX_CHANNEL_GROUPS. */
114 #define CH_GROUP_FRONT ( 0 )
115 #define CH_GROUP_SIDE ( 1 )
116 #define CH_GROUP_REAR ( 2 )
117 #define CH_GROUP_LFE ( 3 )
118
119 /* The ordering of the following fixed channel labels has to be in MPEG-4 style.
120 * From the center to the back with left and right channel interleaved (starting with left).
121 * The last channel label index has to be smaller than PCM_DMX_MAX_CHANNELS. */
122 #define CENTER_FRONT_CHANNEL ( 0 ) /* C */
123 #define LEFT_FRONT_CHANNEL ( 1 ) /* L */
124 #define RIGHT_FRONT_CHANNEL ( 2 ) /* R */
125 #define LEFT_OUTSIDE_CHANNEL ( 3 ) /* Lo */
126 #define RIGHT_OUTSIDE_CHANNEL ( 4 ) /* Ro */
127 #define LEFT_REAR_CHANNEL ( 5 ) /* Lr aka left back channel */
128 #define RIGHT_REAR_CHANNEL ( 6 ) /* Rr aka right back channel */
129 #define LOW_FREQUENCY_CHANNEL ( 7 ) /* Lf */
130
131 /* More constants */
132 #define ANC_DATA_SYNC_BYTE ( 0xBC ) /* ancillary data sync byte. */
133 #define ATTENUATION_FACTOR_1 ( FL2FXCONST_SGL(0.70710678f) )
134 #define TWO_CHANNEL ( 2 )
135
136 /* Sanity checks on library setting: */
137
138 /* List of packed channel modes */
139 typedef enum
140 { /* CH_MODE_<numFrontCh>_<numOutsideCh>_<numRearCh>_<numLfCh> */
141 CH_MODE_UNDEFINED = 0x0000,
142 /* 1 channel */
143 CH_MODE_1_0_0_0 = 0x0001, /* chCfg 1 */
144 /* 2 channels */
145 CH_MODE_2_0_0_0 = 0x0002, /* chCfg 2 */
146 /* 3 channels */
147 CH_MODE_3_0_0_0 = 0x0003, /* chCfg 3 */
148 CH_MODE_2_0_1_0 = 0x0102,
149 CH_MODE_2_0_0_1 = 0x1002,
150 /* 4 channels */
151 CH_MODE_3_0_1_0 = 0x0103, /* chCfg 4 */
152 CH_MODE_2_0_2_0 = 0x0202,
153 CH_MODE_2_0_1_1 = 0x1102,
154 /* 5 channels */
155 CH_MODE_3_0_2_0 = 0x0203, /* chCfg 5 */
156 CH_MODE_2_0_2_1 = 0x1202,
157 CH_MODE_3_0_1_1 = 0x1103,
158 CH_MODE_3_2_0_0 = 0x0023,
159 /* 6 channels */
160 CH_MODE_3_0_2_1 = 0x1203, /* chCfg 6 */
161 CH_MODE_3_2_1_0 = 0x0123,
162 /* 7 channels */
163 CH_MODE_2_2_2_1 = 0x1222,
164 CH_MODE_3_2_1_1 = 0x1123,
165 CH_MODE_3_2_2_0 = 0x0223,
166 /* 8 channels */
167 CH_MODE_3_2_2_1 = 0x1222, /* chCfg 7 */
168 CH_MODE_3_2_1_2 = 0x2123,
169 CH_MODE_2_2_2_2 = 0x2222
170
171 } PCM_DMX_CHANNEL_MODE;
172
173
174 /* These are the channel configurations linked to
175 the number of output channels give by the user: */
176 static const PCM_DMX_CHANNEL_MODE outChModeTable[PCM_DMX_MAX_CHANNELS] =
177 {
178 CH_MODE_1_0_0_0, /* 1 channel */
179 CH_MODE_2_0_0_0, /* 2 channels */
180 CH_MODE_3_0_0_0, /* 3 channels */
181 CH_MODE_3_0_1_0, /* 4 channels */
182 CH_MODE_3_0_2_0, /* 5 channels */
183 CH_MODE_3_0_2_1, /* 6 channels */
184 CH_MODE_3_2_2_0, /* 7 channels */
185 CH_MODE_3_2_2_1 /* 8 channels */
186 };
187
188 static const FIXP_SGL dvbDownmixFactors[8] =
189 {
190 FL2FXCONST_SGL(1.0f),
191 FL2FXCONST_SGL(0.841f),
192 FL2FXCONST_SGL(0.707f),
193 FL2FXCONST_SGL(0.596f),
194 FL2FXCONST_SGL(0.500f),
195 FL2FXCONST_SGL(0.422f),
196 FL2FXCONST_SGL(0.355f),
197 FL2FXCONST_SGL(0.0f)
198 };
199
200
201 /* MPEG matrix mixdown:
202 Set 1: L' = (1 + 2^-0.5 + A )^-1 * [L + C * 2^-0.5 + A * Ls];
203 R' = (1 + 2^-0.5 + A )^-1 * [R + C * 2^-0.5 + A * Rs];
204
205 Set 2: L' = (1 + 2^-0.5 + 2A )^-1 * [L + C * 2^-0.5 - A * (Ls + Rs)];
206 R' = (1 + 2^-0.5 + 2A )^-1 * [R + C * 2^-0.5 + A * (Ls + Rs)];
207
208 M = (3 + 2A)^-1 * [L + C + R + A*(Ls + Rs)];
209 */
210 static const FIXP_SGL mpegMixDownIdx2Coef[4] =
211 {
212 FL2FXCONST_SGL(0.70710678f),
213 FL2FXCONST_SGL(0.5f),
214 FL2FXCONST_SGL(0.35355339f),
215 FL2FXCONST_SGL(0.0f)
216 };
217
218 static const FIXP_SGL mpegMixDownIdx2PreFact[4] =
219 {
220 FL2FXCONST_SGL(0.4142135623730950f),
221 FL2FXCONST_SGL(0.4530818393219728f),
222 FL2FXCONST_SGL(0.4852813742385703f),
223 FL2FXCONST_SGL(0.5857864376269050f)
224 };
225
226 typedef struct
227 {
228 USHORT matrixMixdownIdx; /*!< MPEG mixdown index extracted from PCE. */
229 USHORT pseudoSurroundEnable; /*!< Pseudo surround enable flag extracted from PCE. */
230 USHORT mixdownAvailable; /*!< Will be set to 1 if we found a valid coefficient. */
231
232 } MPEG_MIXDOWN_INFO;
233
234
235 typedef struct
236 {
237 FIXP_SGL centerMixLevelValue; /*!< DVB mixdown level for the center channel extracted from anc data. */
238 FIXP_SGL surroundMixLevelValue; /*!< DVB mixdown level for back channels extracted from anc data. */
239
240 UCHAR mixLevelsAvail; /*!< Will be set to 1 if we found a valid coefficient. */
241
242 } DVB_MIXDOWN_LEVELS;
243
244
245 /* Modules main data structure: */
246 struct PCM_DMX_INSTANCE
247 {
248 DVB_MIXDOWN_LEVELS dvbMixDownLevels[PCM_DMX_MAX_DELAY_FRAMES+1];
249 MPEG_MIXDOWN_INFO mpegMixDownInfo[PCM_DMX_MAX_DELAY_FRAMES+1];
250 DUAL_CHANNEL_MODE dualChannelMode;
251 UINT expiryFrame;
252 UINT expiryCount;
253 SHORT numOutputChannels;
254 UCHAR applyProcessing;
255 UCHAR frameDelay;
256 };
257
258 /* Memory allocation macro */
259 C_ALLOC_MEM_STATIC(PcmDmxInstance, struct PCM_DMX_INSTANCE, 1)
260
261
262 /** Evaluate a given channel configuration and extract a packed channel mode and generate a channel offset table
263 * This function is the inverse to the getChannelDescription() routine.
264 * @param [in] The total number of channels of the given configuration.
265 * @param [in] Array holding the corresponding channel types for each channel.
266 * @param [in] Array holding the corresponding channel type indices for each channel.
267 * @param [out] Array where the buffer offsets for each channel are stored into.
268 * @returns Returns the packed channel mode.
269 **/
270 static
getChannelMode(const INT numChannels,const AUDIO_CHANNEL_TYPE channelType[],const UCHAR channelIndices[],UCHAR offsetTable[PCM_DMX_MAX_CHANNELS])271 PCM_DMX_CHANNEL_MODE getChannelMode (
272 const INT numChannels, /* in */
273 const AUDIO_CHANNEL_TYPE channelType[], /* in */
274 const UCHAR channelIndices[], /* in */
275 UCHAR offsetTable[PCM_DMX_MAX_CHANNELS] /* out */
276 )
277 {
278 UINT chMode = CH_MODE_UNDEFINED;
279 UCHAR chIdx[PCM_DMX_MAX_CHANNEL_GROUPS][PCM_DMX_MAX_CHANNELS_PER_GROUP];
280 UCHAR numChInGrp[PCM_DMX_MAX_CHANNEL_GROUPS];
281 int ch, grpIdx, err = 0;
282
283 FDK_ASSERT(channelType != NULL);
284 FDK_ASSERT(channelIndices != NULL);
285 FDK_ASSERT(offsetTable != NULL);
286
287 /* For details see ISO/IEC 13818-7:2005(E), 8.5.3 Channel configuration */
288 FDKmemclear(numChInGrp, PCM_DMX_MAX_CHANNEL_GROUPS*sizeof(UCHAR));
289 FDKmemset(offsetTable, 255, PCM_DMX_MAX_CHANNELS*sizeof(UCHAR));
290 FDKmemset(chIdx, 255, PCM_DMX_MAX_CHANNEL_GROUPS*PCM_DMX_MAX_CHANNELS_PER_GROUP*sizeof(UCHAR));
291
292 /* Categorize channels */
293 for (ch = 0; ch < numChannels; ch += 1) {
294 int i = 0, j, chGrpIdx = channelIndices[ch];
295
296 switch (channelType[ch]) {
297 case ACT_FRONT:
298 case ACT_FRONT_TOP:
299 grpIdx = CH_GROUP_FRONT;
300 break;
301 case ACT_SIDE:
302 case ACT_SIDE_TOP:
303 grpIdx = CH_GROUP_SIDE;
304 break;
305 case ACT_BACK:
306 case ACT_BACK_TOP:
307 grpIdx = CH_GROUP_REAR;
308 break;
309 case ACT_LFE:
310 grpIdx = CH_GROUP_LFE;
311 break;
312 default:
313 err = -1;
314 continue;
315 }
316
317 if (numChInGrp[grpIdx] < PCM_DMX_MAX_CHANNELS_PER_GROUP) {
318 /* Sort channels by index */
319 while ( (i < numChInGrp[grpIdx]) && (chGrpIdx > channelIndices[chIdx[grpIdx][i]]) ) {
320 i += 1;
321 }
322 for (j = numChInGrp[grpIdx]; j > i; j -= 1) {
323 chIdx[grpIdx][j] = chIdx[grpIdx][j-1];
324 }
325 chIdx[grpIdx][i] = ch;
326 numChInGrp[grpIdx] += 1;
327 }
328 }
329
330 /* Compose channel offset table */
331
332 /* Non-symmetric channels */
333 if (numChInGrp[CH_GROUP_FRONT] & 0x1) {
334 /* Odd number of front channels -> we have a center channel.
335 In MPEG-4 the center has the index 0. */
336 offsetTable[CENTER_FRONT_CHANNEL] = chIdx[CH_GROUP_FRONT][0];
337 }
338
339 for (grpIdx = 0; grpIdx < PCM_DMX_MAX_CHANNEL_GROUPS; grpIdx += 1) {
340 int chMapPos, maxChannels = 0;
341 ch = 0;
342
343 switch (grpIdx) {
344 case CH_GROUP_FRONT:
345 chMapPos = LEFT_FRONT_CHANNEL;
346 maxChannels = 3;
347 ch = numChInGrp[grpIdx] & 0x1;
348 break;
349 case CH_GROUP_SIDE:
350 chMapPos = LEFT_OUTSIDE_CHANNEL;
351 maxChannels = 2;
352 break;
353 case CH_GROUP_REAR:
354 chMapPos = LEFT_REAR_CHANNEL;
355 maxChannels = 2;
356 break;
357 case CH_GROUP_LFE:
358 chMapPos = LOW_FREQUENCY_CHANNEL;
359 maxChannels = 1;
360 break;
361 default:
362 err = -1;
363 continue;
364 }
365
366 for ( ; ch < numChInGrp[grpIdx]; ch += 1) {
367 if (ch < maxChannels) {
368 offsetTable[chMapPos] = chIdx[grpIdx][ch];
369 chMapPos += 1;
370 } else {
371 err = -1;
372 }
373 }
374 }
375
376 if (err == 0) {
377 /* Compose the channel mode */
378 chMode = (numChInGrp[CH_GROUP_LFE] & 0xF) << 12
379 | (numChInGrp[CH_GROUP_REAR] & 0xF) << 8
380 | (numChInGrp[CH_GROUP_SIDE] & 0xF) << 4
381 | (numChInGrp[CH_GROUP_FRONT] & 0xF);
382 }
383
384 return (PCM_DMX_CHANNEL_MODE)chMode;
385 }
386
387
388 /** Generate a channel offset table and complete channel description for a given (packed) channel mode.
389 * This function is the inverse to the getChannelMode() routine.
390 * @param [in] The total number of channels of the given configuration.
391 * @param [in] Array containing the channel mapping to be used (From MPEG PCE ordering to whatever is required).
392 * @param [out] Array where corresponding channel types for each channels are stored into.
393 * @param [out] Array where corresponding channel type indices for each output channel are stored into.
394 * @param [out] Array where the buffer offsets for each channel are stored into.
395 * @returns None.
396 **/
getChannelDescription(const PCM_DMX_CHANNEL_MODE chMode,const UCHAR channelMapping[][PCM_DMX_MAX_CHANNELS],AUDIO_CHANNEL_TYPE channelType[],UCHAR channelIndices[],UCHAR offsetTable[PCM_DMX_MAX_CHANNELS])397 void getChannelDescription (
398 const PCM_DMX_CHANNEL_MODE chMode, /* in */
399 const UCHAR channelMapping[][PCM_DMX_MAX_CHANNELS], /* in */
400 AUDIO_CHANNEL_TYPE channelType[], /* out */
401 UCHAR channelIndices[], /* out */
402 UCHAR offsetTable[PCM_DMX_MAX_CHANNELS] /* out */
403 )
404 {
405 const UCHAR *pChannelMap;
406 int grpIdx, ch = 0, numChannels = 0;
407 UCHAR numChInGrp[PCM_DMX_MAX_CHANNEL_GROUPS];
408
409 FDK_ASSERT(channelType != NULL);
410 FDK_ASSERT(channelIndices != NULL);
411 FDK_ASSERT(channelMapping != NULL);
412 FDK_ASSERT(offsetTable != NULL);
413
414 /* Init output arrays */
415 FDKmemclear(channelType, PCM_DMX_MAX_CHANNELS*sizeof(AUDIO_CHANNEL_TYPE));
416 FDKmemclear(channelIndices, PCM_DMX_MAX_CHANNELS*sizeof(UCHAR));
417 FDKmemset(offsetTable, 255, PCM_DMX_MAX_CHANNELS*sizeof(UCHAR));
418
419 /* Extract the number of channels per group */
420 numChInGrp[CH_GROUP_FRONT] = chMode & 0xF;
421 numChInGrp[CH_GROUP_SIDE] = (chMode >> 4) & 0xF;
422 numChInGrp[CH_GROUP_REAR] = (chMode >> 8) & 0xF;
423 numChInGrp[CH_GROUP_LFE] = (chMode >> 12) & 0xF;
424
425 /* Summerize to get the total number of channels */
426 for (grpIdx = 0; grpIdx < PCM_DMX_MAX_CHANNEL_GROUPS; grpIdx += 1) {
427 numChannels += numChInGrp[grpIdx];
428 }
429
430 /* Get the appropriate channel map */
431 pChannelMap = channelMapping[numChannels-1];
432
433 /* Compose channel offset table */
434
435 /* Non-symmetric channels */
436 if (numChInGrp[CH_GROUP_FRONT] & 0x1) {
437 /* Odd number of front channels -> we have a center channel.
438 In MPEG-4 the center has the index 0. */
439 offsetTable[CENTER_FRONT_CHANNEL] = pChannelMap[0];
440 channelType[0] = ACT_FRONT;
441 ch += 1;
442 }
443
444 for (grpIdx = 0; grpIdx < PCM_DMX_MAX_CHANNEL_GROUPS; grpIdx += 1) {
445 AUDIO_CHANNEL_TYPE type;
446 int chMapPos, maxChannels = 0;
447 int chIdx = 0;
448
449 switch (grpIdx) {
450 case CH_GROUP_FRONT:
451 type = ACT_FRONT;
452 chMapPos = LEFT_FRONT_CHANNEL;
453 maxChannels = 3;
454 chIdx = numChInGrp[grpIdx] & 0x1;
455 break;
456 case CH_GROUP_SIDE:
457 type = ACT_SIDE;
458 chMapPos = LEFT_OUTSIDE_CHANNEL;
459 maxChannels = 2;
460 break;
461 case CH_GROUP_REAR:
462 type = ACT_BACK;
463 chMapPos = LEFT_REAR_CHANNEL;
464 maxChannels = 2;
465 break;
466 case CH_GROUP_LFE:
467 type = ACT_LFE;
468 chMapPos = LOW_FREQUENCY_CHANNEL;
469 maxChannels = 1;
470 break;
471 default:
472 break;
473 }
474
475 for ( ; (chIdx < numChInGrp[grpIdx]) && (chIdx < maxChannels); chIdx += 1) {
476 offsetTable[chMapPos] = pChannelMap[ch];
477 channelType[ch] = type;
478 channelIndices[ch] = chIdx;
479 chMapPos += 1;
480 ch += 1;
481 }
482 }
483 }
484
485
486 /** Open and initialize an instance of the PCM downmix module
487 * @param [out] Pointer to a buffer receiving the handle of the new instance.
488 * @returns Returns an error code.
489 **/
pcmDmx_Open(HANDLE_PCM_DOWNMIX * pSelf)490 PCMDMX_ERROR pcmDmx_Open (
491 HANDLE_PCM_DOWNMIX *pSelf
492 )
493 {
494 HANDLE_PCM_DOWNMIX self;
495
496 if (pSelf == NULL) {
497 return (PCMDMX_INVALID_HANDLE);
498 }
499
500 *pSelf = NULL;
501
502 self = (HANDLE_PCM_DOWNMIX) GetPcmDmxInstance( 0 );
503 if (self == NULL) {
504 return (PCMDMX_OUT_OF_MEMORY);
505 }
506
507 /* Reset the full instance */
508 pcmDmx_Reset( self, PCMDMX_RESET_FULL );
509
510 *pSelf = self;
511
512 return (PCMDMX_OK);
513 }
514
515
516 /** Reset all static values like e.g. mixdown coefficients.
517 * @param [in] Handle of PCM downmix module instance.
518 * @param [in] Flags telling which parts of the module shall be reset.
519 * @returns Returns an error code.
520 **/
pcmDmx_Reset(HANDLE_PCM_DOWNMIX self,UINT flags)521 PCMDMX_ERROR pcmDmx_Reset (
522 HANDLE_PCM_DOWNMIX self,
523 UINT flags
524 )
525 {
526 if (self == NULL) { return (PCMDMX_INVALID_HANDLE); }
527
528 if (flags & PCMDMX_RESET_PARAMS) {
529 self->dualChannelMode = STEREO_MODE;
530 self->numOutputChannels = 0;
531 self->applyProcessing = 0;
532 self->frameDelay = 0;
533 self->expiryFrame = PCMDMX_DFLT_EXPIRY_FRAME;
534 }
535
536 if (flags & PCMDMX_RESET_BS_DATA) {
537 int slot;
538 for (slot = 0; slot <= PCM_DMX_MAX_DELAY_FRAMES; slot += 1) {
539 self->dvbMixDownLevels[slot].centerMixLevelValue = dvbDownmixFactors[2]; /* 0.707 */
540 self->dvbMixDownLevels[slot].surroundMixLevelValue = dvbDownmixFactors[0]; /* 1.000 */
541 self->dvbMixDownLevels[slot].mixLevelsAvail = 0;
542
543 self->mpegMixDownInfo[slot].mixdownAvailable = 0;
544 }
545 /* Reset expiry counter */
546 self->expiryCount = 0;
547 }
548
549 return (PCMDMX_OK);
550 }
551
552
553 /** Set one parameter for one instance of the PCM downmix module.
554 * @param [in] Handle of PCM downmix module instance.
555 * @param [in] Parameter to be set.
556 * @param [in] Parameter value.
557 * @returns Returns an error code.
558 **/
pcmDmx_SetParam(HANDLE_PCM_DOWNMIX self,PCMDMX_PARAM param,UINT value)559 PCMDMX_ERROR pcmDmx_SetParam (
560 HANDLE_PCM_DOWNMIX self,
561 PCMDMX_PARAM param,
562 UINT value
563 )
564 {
565 switch (param)
566 {
567 case DMX_BS_DATA_EXPIRY_FRAME:
568 if (self == NULL)
569 return (PCMDMX_INVALID_HANDLE);
570 self->expiryFrame = value;
571 break;
572
573 case DMX_BS_DATA_DELAY:
574 if (value > PCM_DMX_MAX_DELAY_FRAMES) {
575 return (PCMDMX_UNABLE_TO_SET_PARAM);
576 }
577 if (self == NULL) {
578 return (PCMDMX_INVALID_HANDLE);
579 }
580 self->frameDelay = value;
581 break;
582
583 case NUMBER_OF_OUTPUT_CHANNELS:
584 switch ((int)value) { /* supported output channels */
585 case -1: case 0: case 1: case 2:
586 case 6: case 8:
587 break;
588 default:
589 return (PCMDMX_UNABLE_TO_SET_PARAM);
590 }
591 if (self == NULL)
592 return (PCMDMX_INVALID_HANDLE);
593 if ((int)value > 0) {
594 self->numOutputChannels = (int)value;
595 self->applyProcessing = 1;
596 } else {
597 self->numOutputChannels = 0;
598 self->applyProcessing = 0;
599 }
600 break;
601
602 case DUAL_CHANNEL_DOWNMIX_MODE:
603 switch ((DUAL_CHANNEL_MODE)value) {
604 case STEREO_MODE:
605 case CH1_MODE:
606 case CH2_MODE:
607 case MIXED_MODE:
608 break;
609 default:
610 return (PCMDMX_UNABLE_TO_SET_PARAM);
611 }
612 if (self == NULL)
613 return (PCMDMX_INVALID_HANDLE);
614 self->dualChannelMode = (DUAL_CHANNEL_MODE)value;
615 self->applyProcessing = 1;
616 break;
617
618 default:
619 return (PCMDMX_UNKNOWN_PARAM);
620 }
621
622 return (PCMDMX_OK);
623 }
624
625
626 /** Read the ancillary data transported in DSEs of DVB streams with MPEG-4 content
627 * @param [in] Handle of PCM downmix module instance.
628 * @param [in] Pointer to ancillary data buffer.
629 * @param [in] Size of ancillary data.
630 * @param [in] Flag indicating wheter the DVB ancillary data is from an MPEG-1/2 or an MPEG-4 stream.
631 * @returns Returns an error code.
632 **/
pcmDmx_ReadDvbAncData(HANDLE_PCM_DOWNMIX self,UCHAR * pAncDataBuf,UINT ancDataBytes,int isMpeg2)633 PCMDMX_ERROR pcmDmx_ReadDvbAncData (
634 HANDLE_PCM_DOWNMIX self,
635 UCHAR *pAncDataBuf,
636 UINT ancDataBytes,
637 int isMpeg2
638 )
639 {
640 DVB_MIXDOWN_LEVELS *pDownmixLevels = &self->dvbMixDownLevels[0];
641
642 int offset = (isMpeg2) ? 2 : 0;
643 UCHAR ancDataStatus;
644
645 if (self == NULL) { return (PCMDMX_INVALID_HANDLE); }
646
647 /* sanity checks */
648 if (pAncDataBuf == NULL || ancDataBytes < (UCHAR)(3+offset)) {
649 return (PCMDMX_CORRUPT_ANC_DATA);
650 }
651
652 /* check sync word */
653 if (pAncDataBuf[offset] != ANC_DATA_SYNC_BYTE) {
654 return (PCMDMX_CORRUPT_ANC_DATA);
655 }
656
657 offset += 2;
658 ancDataStatus = pAncDataBuf[offset++];
659
660 if (isMpeg2) {
661 /* skip advanced_dynamic_range_control */
662 if (ancDataStatus & 0x80) offset += 3;
663 /* skip dialog_normalization */
664 if (ancDataStatus & 0x40) offset += 1;
665 /* skip reproduction_level */
666 if (ancDataStatus & 0x20) offset += 1;
667 }
668 else {
669 /* check reserved bits */
670 if (ancDataStatus & 0xE8) { return (PCMDMX_CORRUPT_ANC_DATA); }
671 }
672
673 /* downmix_levels_MPEGX */
674 if (ancDataStatus & 0x10)
675 {
676 int foundNewData = 0;
677 UCHAR downmixData = pAncDataBuf[offset++];
678
679 if (downmixData & 0x80) { /* center_mix_level_on */
680 pDownmixLevels->centerMixLevelValue =
681 dvbDownmixFactors[(downmixData >> 4) & 0x07];
682 foundNewData = 1;
683 } else {
684 pDownmixLevels->centerMixLevelValue = dvbDownmixFactors[0];
685 if (downmixData & 0x70) { return (PCMDMX_CORRUPT_ANC_DATA); }
686 }
687
688 if (downmixData & 0x08) { /* surround_mix_level_on */
689 pDownmixLevels->surroundMixLevelValue =
690 dvbDownmixFactors[downmixData & 0x07];
691 foundNewData = 1;
692 } else {
693 pDownmixLevels->surroundMixLevelValue = dvbDownmixFactors[0];
694 if (downmixData & 0x07) { return (PCMDMX_CORRUPT_ANC_DATA); }
695 }
696
697 pDownmixLevels->mixLevelsAvail = foundNewData;
698 }
699
700 /* Reset expiry counter */
701 self->expiryCount = 0;
702
703 return (PCMDMX_OK);
704 }
705
706 /** Set the matrix mixdown information extracted from the PCE of an AAC bitstream.
707 * Note: Call only if matrix_mixdown_idx_present is true.
708 * @param [in] Handle of PCM downmix module instance.
709 * @param [in] The 2 bit matrix mixdown index extracted from PCE.
710 * @param [in] The pseudo surround enable flag extracted from PCE.
711 * @returns Returns an error code.
712 **/
pcmDmx_SetMatrixMixdownFromPce(HANDLE_PCM_DOWNMIX self,int matrixMixdownPresent,int matrixMixdownIdx,int pseudoSurroundEnable)713 PCMDMX_ERROR pcmDmx_SetMatrixMixdownFromPce (
714 HANDLE_PCM_DOWNMIX self,
715 int matrixMixdownPresent,
716 int matrixMixdownIdx,
717 int pseudoSurroundEnable
718 )
719 {
720 MPEG_MIXDOWN_INFO *pMpegMixDownInfo;
721
722 if (self == NULL) {
723 return (PCMDMX_INVALID_HANDLE);
724 }
725
726 pMpegMixDownInfo = &self->mpegMixDownInfo[0];
727
728 if (matrixMixdownPresent) {
729 pMpegMixDownInfo->matrixMixdownIdx = matrixMixdownIdx & 0x03;
730 pMpegMixDownInfo->pseudoSurroundEnable = pseudoSurroundEnable;
731 }
732
733 pMpegMixDownInfo->mixdownAvailable = matrixMixdownPresent;
734 /* Reset expiry counter */
735 self->expiryCount = 0;
736
737 return (PCMDMX_OK);
738 }
739
740
741 /** Apply down or up mixing.
742 * @param [in] Handle of PCM downmix module instance.
743 * @param [inout] Pointer to time buffer. Depending on interface configuration, the content of pTimeData is ignored,
744 * and the internal QMF buffer will be used as input data source. Otherwise, the MPEG Surround processing is
745 * applied to the timesignal pTimeData. For both variants, the resulting MPEG Surround signal is written into pTimeData.
746 * @param [in] Pointer where the amount of output samples is returned into.
747 * @param [inout] Pointer where the amount of output channels is returned into.
748 * @param [in] Flag which indicates if output time data are writtern interleaved or as subsequent blocks.
749 * @param [inout] Array where the corresponding channel type for each output audio channel is stored into.
750 * @param [inout] Array where the corresponding channel type index for each output audio channel is stored into.
751 * @param [in] Array containing the output channel mapping to be used (From MPEG PCE ordering to whatever is required).
752 * @returns Returns an error code.
753 **/
pcmDmx_ApplyFrame(HANDLE_PCM_DOWNMIX self,INT_PCM * pPcmBuf,UINT frameSize,INT * nChannels,int fInterleaved,AUDIO_CHANNEL_TYPE channelType[],UCHAR channelIndices[],const UCHAR channelMapping[][8])754 PCMDMX_ERROR pcmDmx_ApplyFrame (
755 HANDLE_PCM_DOWNMIX self,
756 INT_PCM *pPcmBuf,
757 UINT frameSize,
758 INT *nChannels,
759
760 int fInterleaved,
761 AUDIO_CHANNEL_TYPE channelType[],
762 UCHAR channelIndices[],
763 const UCHAR channelMapping[][8]
764 )
765 {
766 PCMDMX_ERROR errorStatus = PCMDMX_OK;
767 DUAL_CHANNEL_MODE dualChannelMode;
768 PCM_DMX_CHANNEL_MODE inChMode;
769 int numOutChannels;
770 int numInChannels = *nChannels;
771 int slot;
772 UCHAR inOffsetTable[PCM_DMX_MAX_CHANNELS];
773
774 MPEG_MIXDOWN_INFO mpegMixDownInfo;
775 DVB_MIXDOWN_LEVELS dvbMixDownLevels;
776
777 if (self == NULL) { return (PCMDMX_INVALID_HANDLE); }
778
779 if ( (self->expiryFrame > 0)
780 && (++self->expiryCount > self->expiryFrame) )
781 { /* The metadata read from bitstream is too old. */
782 errorStatus = pcmDmx_Reset(self, PCMDMX_RESET_BS_DATA);
783 }
784
785 FDKmemcpy(&mpegMixDownInfo, &self->mpegMixDownInfo[self->frameDelay], sizeof(MPEG_MIXDOWN_INFO));
786 /* Maintain delay line */
787 for (slot = self->frameDelay; slot > 0; slot -= 1) {
788 FDKmemcpy(&self->mpegMixDownInfo[slot], &self->mpegMixDownInfo[slot-1], sizeof(MPEG_MIXDOWN_INFO));
789 }
790 FDKmemcpy(&dvbMixDownLevels, &self->dvbMixDownLevels[self->frameDelay], sizeof(DVB_MIXDOWN_LEVELS));
791 /* Maintain delay line */
792 for (slot = self->frameDelay; slot > 0; slot -= 1) {
793 FDKmemcpy(&self->dvbMixDownLevels[slot], &self->dvbMixDownLevels[slot-1], sizeof(DVB_MIXDOWN_LEVELS));
794 }
795
796 if (self->applyProcessing == 0) { return (errorStatus); }
797
798 if (pPcmBuf == NULL) { return (PCMDMX_INVALID_ARGUMENT); }
799 if (frameSize == 0) { return (PCMDMX_INVALID_ARGUMENT); }
800 if (numInChannels == 0) { return (PCMDMX_INVALID_ARGUMENT); }
801
802 if (self->numOutputChannels <= 0) {
803 numOutChannels = numInChannels;
804 } else {
805 numOutChannels = self->numOutputChannels;
806 }
807 dualChannelMode = self->dualChannelMode;
808
809 /* Analyse input channel configuration and get channel offset
810 * table that can be accessed with the fixed channel labels. */
811 inChMode = getChannelMode(
812 numInChannels,
813 channelType,
814 channelIndices,
815 inOffsetTable
816 );
817 if (inChMode == CH_MODE_UNDEFINED) {
818 /* We don't need to restore because the channel
819 configuration has not been changed. Just exit. */
820 return (PCMDMX_INVALID_CH_CONFIG);
821 }
822
823 /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
824 if ( numInChannels > numOutChannels )
825 { /* Apply downmix */
826 INT_PCM *pInCF, *pInLF, *pInRF, *pInLO, *pInRO, *pInLR, *pInRR, *pOutL, *pOutR;
827 FIXP_SGL flev, clev, slev;
828
829 UINT sample;
830 int inStride, outStride, offset;
831 int useGuidedDownMix = 0;
832 UCHAR outOffsetTable[PCM_DMX_MAX_CHANNELS];
833
834 /* Set I/O strides and offsets */
835 if (fInterleaved) {
836 inStride = numInChannels;
837 outStride = TWO_CHANNEL; /* The output of STAGE ONE is always STEREO !!!
838 STAGE TWO creates a downmix to mono if required. */
839 offset = 1; /* Channel specific offset factor */
840 } else {
841 inStride = 1;
842 outStride = 1;
843 offset = frameSize; /* Channel specific offset factor */
844 }
845
846 /* Get channel description and channel mapping for this
847 * stages number of output channels (always STEREO). */
848 getChannelDescription(
849 CH_MODE_2_0_0_0,
850 channelMapping,
851 channelType,
852 channelIndices,
853 outOffsetTable
854 );
855 /* Now there is no way back because we modified the channel configuration! */
856
857 /* Set channel pointer for input */
858 pInCF = &pPcmBuf[inOffsetTable[CENTER_FRONT_CHANNEL]*offset];
859 pInLF = &pPcmBuf[inOffsetTable[LEFT_FRONT_CHANNEL]*offset];
860 pInRF = &pPcmBuf[inOffsetTable[RIGHT_FRONT_CHANNEL]*offset];
861 pInLO = &pPcmBuf[inOffsetTable[LEFT_OUTSIDE_CHANNEL]*offset];
862 pInRO = &pPcmBuf[inOffsetTable[RIGHT_OUTSIDE_CHANNEL]*offset];
863 pInLR = &pPcmBuf[inOffsetTable[LEFT_REAR_CHANNEL]*offset];
864 pInRR = &pPcmBuf[inOffsetTable[RIGHT_REAR_CHANNEL]*offset];
865
866 /* Set channel pointer for output
867 Caution: Different channel mapping compared to input */
868 pOutL = &pPcmBuf[outOffsetTable[LEFT_FRONT_CHANNEL]*offset]; /* LEFT_FRONT_CHANNEL */
869 pOutR = &pPcmBuf[outOffsetTable[RIGHT_FRONT_CHANNEL]*offset]; /* RIGHT_FRONT_CHANNEL */
870
871 /* Set downmix levels: */
872 flev = ATTENUATION_FACTOR_1; /* 0.707 */
873 clev = ATTENUATION_FACTOR_1; /* 0.707 */
874 slev = ATTENUATION_FACTOR_1; /* 0.707 */
875
876 if ( dvbMixDownLevels.mixLevelsAvail ) {
877 clev = dvbMixDownLevels.centerMixLevelValue;
878 slev = dvbMixDownLevels.surroundMixLevelValue;
879 useGuidedDownMix = 1;
880 }
881
882 /* FIRST STAGE:
883 Always downmix to 2 channel output: */
884 switch ( inChMode )
885 {
886 case CH_MODE_2_0_0_0:
887 case CH_MODE_2_0_0_1:
888 /* 2/0 input: */
889 switch (dualChannelMode)
890 {
891 case CH1_MODE: /* L' = 0.707 * Ch1; R' = 0.707 * Ch1 */
892 for (sample = 0; sample < frameSize; sample++) {
893 *pOutL = *pOutR =
894 (INT_PCM)SATURATE_RIGHT_SHIFT(fMult((FIXP_PCM)*pInLF, flev), DFRACT_BITS-SAMPLE_BITS, SAMPLE_BITS);
895
896 pInLF += inStride;
897 pOutL += outStride; pOutR += outStride;
898 }
899 break;
900
901 case CH2_MODE: /* L' = 0.707 * Ch2; R' = 0.707 * Ch2 */
902 for (sample = 0; sample < frameSize; sample++) {
903 *pOutL = *pOutR =
904 (INT_PCM)SATURATE_RIGHT_SHIFT(fMult((FIXP_PCM)*pInRF, flev), DFRACT_BITS-SAMPLE_BITS, SAMPLE_BITS);
905
906 pInRF += inStride;
907 pOutL += outStride; pOutR += outStride;
908 }
909 break;
910 case MIXED_MODE: /* L' = 0.5*Ch1 + 0.5*Ch2; R' = 0.5*Ch1 + 0.5*Ch2 */
911 for (sample = 0; sample < frameSize; sample++) {
912 *pOutL = *pOutR = (*pInLF >> 1) + (*pInRF >> 1);
913
914 pInLF += inStride; pInRF += inStride;
915 pOutL += outStride; pOutR += outStride;
916 }
917 break;
918 default:
919 case STEREO_MODE:
920 /* nothing to do */
921 break;
922 }
923 break;
924
925 case CH_MODE_3_0_0_0:
926 /* 3/0 input: L' = L + 0.707*C; R' = R + 0.707*C; */
927 for (sample = 0; sample < frameSize; sample++)
928 {
929 FIXP_DBL tCF = fMultDiv2((FIXP_PCM)*pInCF, clev);
930 #if (SAMPLE_BITS == 32)
931 /* left channel */
932 *pOutL = (INT_PCM)SATURATE_LEFT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInLF)>>1)+tCF, 1, SAMPLE_BITS);
933 /* right channel */
934 *pOutR = (INT_PCM)SATURATE_LEFT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInRF)>>1)+tCF, 1, SAMPLE_BITS);
935 #else
936 /* left channel */
937 *pOutL = (INT_PCM)SATURATE_RIGHT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInLF)>>1)+tCF, DFRACT_BITS-SAMPLE_BITS-1, SAMPLE_BITS);
938 /* right channel */
939 *pOutR = (INT_PCM)SATURATE_RIGHT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInRF)>>1)+tCF, DFRACT_BITS-SAMPLE_BITS-1, SAMPLE_BITS);
940 #endif
941 pInLF += inStride; pInRF += inStride; pInCF += inStride;
942 pOutL += outStride; pOutR += outStride;
943 }
944 break;
945
946 /* 2/1 input: not supported!
947 case CH_MODE_2_0_1_0: */
948
949 case CH_MODE_3_0_1_0:
950 if (useGuidedDownMix) {
951 /* 3/1 input: L' = L + clev*C + 0.707*slev*S; R' = R + clev*C + 0.707*slev*S; */
952 slev = FX_DBL2FX_SGL(fMult(flev, slev)); /* 0.707*slef */
953
954 for (sample = 0; sample < frameSize; sample++)
955 {
956 FIXP_DBL tCF = fMultDiv2((FIXP_PCM)*pInCF, clev) >> 1;
957 FIXP_DBL tLR = fMultDiv2((FIXP_PCM)*pInLR, slev) >> 1;
958 #if (SAMPLE_BITS == 32)
959 /* left channel */
960 *pOutL = (INT_PCM)SATURATE_LEFT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInLF)>>2)+tCF+tLR, 2, SAMPLE_BITS);
961 /* right channel */
962 *pOutR = (INT_PCM)SATURATE_LEFT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInRF)>>2)+tCF+tLR, 2, SAMPLE_BITS);
963 #else
964 /* left channel */
965 *pOutL = (INT_PCM)SATURATE_RIGHT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInLF)>>2)+tCF-tLR, DFRACT_BITS-SAMPLE_BITS-2, SAMPLE_BITS);
966 /* right channel */
967 *pOutR = (INT_PCM)SATURATE_RIGHT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInRF)>>2)+tCF+tLR, DFRACT_BITS-SAMPLE_BITS-2, SAMPLE_BITS);
968 #endif
969 pInLF += inStride; pInRF += inStride; pInCF += inStride; pInLR += inStride;
970 pOutL += outStride; pOutR += outStride;
971 }
972 } else {
973 /* 3/1 input: L' = L + 0.707*C - 0.707*S; R' = R + 0.707*C + 0.707*S */
974 for (sample = 0; sample < frameSize; sample++)
975 {
976 FIXP_DBL tCF = fMultDiv2((FIXP_PCM)*pInCF, clev) >> 1;
977 FIXP_DBL tLR = fMultDiv2((FIXP_PCM)*pInLR, slev) >> 1;
978 #if (SAMPLE_BITS == 32)
979 /* left channel */
980 *pOutL = (INT_PCM)SATURATE_LEFT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInLF)>>2)+tCF-tLR, 2, SAMPLE_BITS);
981 /* right channel */
982 *pOutR = (INT_PCM)SATURATE_LEFT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInRF)>>2)+tCF+tLR, 2, SAMPLE_BITS);
983 #else
984 /* left channel */
985 *pOutL = (INT_PCM)SATURATE_RIGHT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInLF)>>2)+tCF-tLR, DFRACT_BITS-SAMPLE_BITS-2, SAMPLE_BITS);
986 /* right channel */
987 *pOutR = (INT_PCM)SATURATE_RIGHT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInRF)>>2)+tCF+tLR, DFRACT_BITS-SAMPLE_BITS-2, SAMPLE_BITS);
988 #endif
989 pInLF += inStride; pInRF += inStride; pInCF += inStride; pInLR += inStride;
990 pOutL += outStride; pOutR += outStride;
991 }
992 }
993 break;
994
995 /* 2/2 input: not supported!
996 case CH_MODE_2_0_2_0: */
997
998 case CH_MODE_3_0_2_0: /* 5.0ch input */
999 case CH_MODE_3_0_2_1: /* 5.1ch input */
1000 if (useGuidedDownMix) {
1001 /* 3/2 input: L' = L + clev*C + slev*Ls; R' = R + clev*C + slev*Rs; */
1002 for (sample = 0; sample < frameSize; sample++)
1003 {
1004 FIXP_DBL tCF = fMultDiv2((FIXP_PCM)*pInCF, clev) >> 1;
1005 FIXP_DBL tLR = fMultDiv2((FIXP_PCM)*pInLR, slev) >> 1;
1006 FIXP_DBL tRR = fMultDiv2((FIXP_PCM)*pInRR, slev) >> 1;
1007 #if (SAMPLE_BITS == 32)
1008 /* left channel */
1009 *pOutL = (INT_PCM)SATURATE_LEFT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInLF)>>2)+tCF+tLR, 2, SAMPLE_BITS);
1010 /* right channel */
1011 *pOutR = (INT_PCM)SATURATE_LEFT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInRF)>>2)+tCF+tRR, 2, SAMPLE_BITS);
1012 #else
1013 /* left channel */
1014 *pOutL = (INT_PCM)SATURATE_RIGHT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInLF)>>2)+tCF+tLR, DFRACT_BITS-SAMPLE_BITS-2, SAMPLE_BITS);
1015 /* right channel */
1016 *pOutR = (INT_PCM)SATURATE_RIGHT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInRF)>>2)+tCF+tRR, DFRACT_BITS-SAMPLE_BITS-2, SAMPLE_BITS);
1017 #endif
1018 pInLF += inStride; pInRF += inStride; pInCF += inStride; pInLR += inStride; pInRR += inStride;
1019 pOutL += outStride; pOutR += outStride;
1020 }
1021 }
1022 else if (mpegMixDownInfo.mixdownAvailable) {
1023 /* 3/2 input: L' = (1.707+A)^-1 * [L+0.707*C+A*Ls]; R'= (1.707+A)^-1 * [R+0.707*C+A*Rs]; */
1024 FIXP_SGL mtrxMixDwnCoef = mpegMixDownIdx2Coef[mpegMixDownInfo.matrixMixdownIdx];
1025 FIXP_SGL mtrxMixDwnPreFact = mpegMixDownIdx2PreFact[mpegMixDownInfo.matrixMixdownIdx];
1026 clev = FX_DBL2FX_SGL(fMult(mtrxMixDwnPreFact, flev /* 0.707 */));
1027 flev = mtrxMixDwnPreFact;
1028 slev = FX_DBL2FX_SGL(fMult(mtrxMixDwnPreFact, mtrxMixDwnCoef));
1029
1030 for (sample = 0; sample < frameSize; sample++)
1031 {
1032 FIXP_DBL tCF = fMultDiv2((FIXP_PCM)*pInCF, clev);
1033 FIXP_DBL tLF = fMultDiv2((FIXP_PCM)*pInLF, flev);
1034 FIXP_DBL tRF = fMultDiv2((FIXP_PCM)*pInRF, flev);
1035 FIXP_DBL tLR = fMultDiv2((FIXP_PCM)*pInLR, slev);
1036 FIXP_DBL tRR = fMultDiv2((FIXP_PCM)*pInRR, slev);
1037
1038 #if (SAMPLE_BITS == 32)
1039 /* left channel */
1040 *pOutL = (INT_PCM)SATURATE_LEFT_SHIFT(tLF+tCF+tLR, 1, SAMPLE_BITS);
1041 /* right channel */
1042 *pOutR = (INT_PCM)SATURATE_LEFT_SHIFT(tRF+tCF+tRR, 1, SAMPLE_BITS);
1043 #else
1044 /* left channel */
1045 *pOutL = (INT_PCM)SATURATE_RIGHT_SHIFT(tLF+tCF+tLR, DFRACT_BITS-SAMPLE_BITS-1, SAMPLE_BITS);
1046 /* right channel */
1047 *pOutR = (INT_PCM)SATURATE_RIGHT_SHIFT(tRF+tCF+tRR, DFRACT_BITS-SAMPLE_BITS-1, SAMPLE_BITS);
1048 #endif
1049
1050 pInLF += inStride; pInRF += inStride; pInCF += inStride; pInLR += inStride; pInRR += inStride;
1051 pOutL += outStride; pOutR += outStride;
1052 }
1053 }
1054 else {
1055 /* 3/2 input: L' = L + 0.707*C - 0.707*Ls - 0.707*Rs; R' = R + 0.707*C + 0.707*Ls + 0.707*Rs */
1056 for (sample = 0; sample < frameSize; sample++)
1057 {
1058 FIXP_DBL tCF = fMultDiv2((FIXP_PCM)*pInCF, clev) >> 2;
1059 FIXP_DBL tLR = fMultDiv2((FIXP_PCM)*pInLR, slev) >> 2;
1060 FIXP_DBL tRR = fMultDiv2((FIXP_PCM)*pInRR, slev) >> 2;
1061 #if (SAMPLE_BITS == 32)
1062 /* left channel */
1063 *pOutL = (INT_PCM)SATURATE_LEFT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInLF)>>3)+tCF-tLR-tRR, 3, SAMPLE_BITS);
1064 /* right channel */
1065 *pOutR = (INT_PCM)SATURATE_LEFT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInRF)>>3)+tCF+tLR+tRR, 3, SAMPLE_BITS);
1066 #else
1067 /* left channel */
1068 *pOutL = (INT_PCM)SATURATE_RIGHT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInLF)>>3)+tCF-tLR-tRR, DFRACT_BITS-SAMPLE_BITS-3, SAMPLE_BITS);
1069 /* right channel */
1070 *pOutR = (INT_PCM)SATURATE_RIGHT_SHIFT((FX_PCM2FX_DBL((FIXP_PCM)*pInRF)>>3)+tCF+tLR+tRR, DFRACT_BITS-SAMPLE_BITS-3, SAMPLE_BITS);
1071 #endif
1072 pInLF += inStride; pInRF += inStride; pInCF += inStride; pInLR += inStride; pInRR += inStride;
1073 pOutL += outStride; pOutR += outStride;
1074 }
1075 }
1076 break;
1077
1078 default:
1079 errorStatus = PCMDMX_INVALID_MODE;
1080 break;
1081 }
1082
1083 /* SECOND STAGE:
1084 If desired create a mono donwmix:
1085 Note: Input are always two channels! */
1086 if (numOutChannels == 1)
1087 {
1088 INT_PCM *pOutC;
1089 FIXP_SGL mlev;
1090
1091 if (useGuidedDownMix) mlev = FL2FXCONST_SGL(1.0f); else mlev = flev;
1092
1093 /* Output of STAGE ONE = Input of STAGE TWO */
1094 FDKmemcpy(inOffsetTable, outOffsetTable, PCM_DMX_MAX_CHANNELS*sizeof(UCHAR));
1095
1096 /* Set I/O strides and offsets */
1097 inStride = outStride; /* output from STAGE ONE */
1098 outStride = numOutChannels; /* final output */
1099
1100 /* Get channel description and channel mapping for this
1101 * stages number of output channels (always MONO). */
1102 getChannelDescription(
1103 CH_MODE_1_0_0_0,
1104 channelMapping,
1105 channelType,
1106 channelIndices,
1107 outOffsetTable
1108 );
1109
1110 /* Set input channel pointer. */
1111 pInLF = &pPcmBuf[inOffsetTable[LEFT_FRONT_CHANNEL]*offset];
1112 pInRF = &pPcmBuf[inOffsetTable[RIGHT_FRONT_CHANNEL]*offset];
1113
1114 /* Set output channel pointer */
1115 pOutC = &pPcmBuf[outOffsetTable[CENTER_FRONT_CHANNEL]*offset];
1116
1117 /* C' = 0.707*L + 0.707*R */
1118 for (sample = 0; sample < frameSize; sample++) {
1119 #if (SAMPLE_BITS == 32)
1120 *pOutC =
1121 (INT_PCM)SATURATE_LEFT_SHIFT(fMultDiv2((FIXP_PCM)*pInLF,mlev)+fMultDiv2((FIXP_PCM)*pInRF,mlev), 1, SAMPLE_BITS);
1122 #else
1123 *pOutC =
1124 (INT_PCM)SATURATE_RIGHT_SHIFT(fMultDiv2((FIXP_PCM)*pInLF,mlev)+fMultDiv2((FIXP_PCM)*pInRF,mlev), DFRACT_BITS-SAMPLE_BITS-1, SAMPLE_BITS);
1125 #endif
1126
1127 pInLF += inStride; pInRF += inStride;
1128 pOutC += 1;
1129 }
1130 /* Finished STAGE TWO */
1131 }
1132
1133 /* Update the number of output channels */
1134 *nChannels = self->numOutputChannels;
1135
1136 } /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
1137 else
1138 if ( numInChannels == numOutChannels )
1139 {
1140 /* Don't need to change the channel description here */
1141
1142 switch (numInChannels)
1143 {
1144 case 2:
1145 { /* Set up channel pointer */
1146 INT_PCM *pInLF, *pInRF, *pOutL, *pOutR;
1147 FIXP_SGL flev;
1148
1149 UINT sample;
1150 int inStride, outStride, offset;
1151
1152 if (fInterleaved) {
1153 inStride = numInChannels;
1154 outStride = 2; /* fixed !!! (below stereo is donwmixed to mono if required */
1155 offset = 1; /* Channel specific offset factor */
1156 } else {
1157 inStride = 1;
1158 outStride = 1;
1159 offset = frameSize; /* Channel specific offset factor */
1160 }
1161
1162 /* Set input channel pointer */
1163 pInLF = &pPcmBuf[inOffsetTable[LEFT_FRONT_CHANNEL]*offset];
1164 pInRF = &pPcmBuf[inOffsetTable[RIGHT_FRONT_CHANNEL]*offset];
1165
1166 /* Set output channel pointer (same as input) */
1167 pOutL = pInLF;
1168 pOutR = pInRF;
1169
1170 /* Set downmix levels: */
1171 flev = ATTENUATION_FACTOR_1; /* 0.707 */
1172 /* 2/0 input: */
1173 switch (dualChannelMode)
1174 {
1175 case CH1_MODE: /* L' = 0.707 * Ch1; R' = 0.707 * Ch1 */
1176 for (sample = 0; sample < frameSize; sample++) {
1177 *pOutL = *pOutR =
1178 (INT_PCM)SATURATE_RIGHT_SHIFT(fMult((FIXP_PCM)*pInLF, flev), DFRACT_BITS-SAMPLE_BITS, SAMPLE_BITS);
1179
1180 pInLF += inStride;
1181 pOutL += outStride; pOutR += outStride;
1182 }
1183 break;
1184 case CH2_MODE: /* L' = 0.707 * Ch2; R' = 0.707 * Ch2 */
1185 for (sample = 0; sample < frameSize; sample++) {
1186 *pOutL = *pOutR =
1187 (INT_PCM)SATURATE_RIGHT_SHIFT(fMult((FIXP_PCM)*pInRF, flev), DFRACT_BITS-SAMPLE_BITS, SAMPLE_BITS);
1188
1189 pInRF += inStride;
1190 pOutL += outStride; pOutR += outStride;
1191 }
1192 break;
1193 case MIXED_MODE: /* L' = 0.5*Ch1 + 0.5*Ch2; R' = 0.5*Ch1 + 0.5*Ch2 */
1194 for (sample = 0; sample < frameSize; sample++) {
1195 *pOutL = *pOutR = (*pInLF >> 1) + (*pInRF >> 1);
1196
1197 pInLF += inStride; pInRF += inStride;
1198 pOutL += outStride; pOutR += outStride;
1199 }
1200 break;
1201 default:
1202 case STEREO_MODE:
1203 /* nothing to do */
1204 break;
1205 }
1206 }
1207 break;
1208
1209 default:
1210 /* nothing to do */
1211 break;
1212 }
1213 }
1214
1215 return (errorStatus);
1216 }
1217
1218
1219 /** Close an instance of the PCM downmix module.
1220 * @param [inout] Pointer to a buffer containing the handle of the instance.
1221 * @returns Returns an error code.
1222 **/
pcmDmx_Close(HANDLE_PCM_DOWNMIX * pSelf)1223 PCMDMX_ERROR pcmDmx_Close (
1224 HANDLE_PCM_DOWNMIX *pSelf
1225 )
1226 {
1227 if (pSelf == NULL) {
1228 return (PCMDMX_INVALID_HANDLE);
1229 }
1230
1231 FreePcmDmxInstance( pSelf );
1232 *pSelf = NULL;
1233
1234 return (PCMDMX_OK);
1235 }
1236
1237
1238 /** Get library info for this module.
1239 * @param [out] Pointer to an allocated LIB_INFO structure.
1240 * @returns Returns an error code.
1241 */
pcmDmx_GetLibInfo(LIB_INFO * info)1242 PCMDMX_ERROR pcmDmx_GetLibInfo( LIB_INFO *info )
1243 {
1244 int i;
1245
1246 if (info == NULL) {
1247 return PCMDMX_INVALID_ARGUMENT;
1248 }
1249
1250 /* Search for next free tab */
1251 for (i = 0; i < FDK_MODULE_LAST; i++) {
1252 if (info[i].module_id == FDK_NONE) break;
1253 }
1254 if (i == FDK_MODULE_LAST) {
1255 return PCMDMX_UNKNOWN;
1256 }
1257 info += i;
1258
1259 /* Add the library info */
1260 info->module_id = FDK_PCMDMX;
1261 info->version = LIB_VERSION(PCMDMX_LIB_VL0, PCMDMX_LIB_VL1, PCMDMX_LIB_VL2);
1262 LIB_VERSION_STRING(info);
1263 info->build_date = PCMDMX_LIB_BUILD_DATE;
1264 info->build_time = PCMDMX_LIB_BUILD_TIME;
1265 info->title = PCMDMX_LIB_TITLE;
1266
1267 /* Set flags */
1268 info->flags = 0
1269 | CAPF_DMX_BLIND /* At least blind downmixing is possible */
1270 | CAPF_DMX_PCE /* Guided downmix with data from MPEG-2/4 Program Config Elements (PCE). */
1271 | CAPF_DMX_DVB /* Guided downmix with data from DVB ancillary data fields. */
1272 ;
1273
1274 return PCMDMX_OK;
1275 }
1276
1277
1278
1279