1 2 /* ----------------------------------------------------------------------------------------------------------- 3 Software License for The Fraunhofer FDK AAC Codec Library for Android 4 5 Copyright 1995 - 2013 Fraunhofer-Gesellschaft zur Frderung 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 (at) iis.fraunhofer.de 82 ----------------------------------------------------------------------------------------------------------- */ 83 84 #include "fram_gen.h" 85 #include "sbr_misc.h" 86 87 #include "genericStds.h" 88 89 static const SBR_FRAME_INFO frameInfo1_2048 = { 90 1, 91 { 0, 16}, 92 {FREQ_RES_HIGH}, 93 0, 94 1, 95 {0, 16} }; 96 97 static const SBR_FRAME_INFO frameInfo2_2048 = { 98 2, 99 { 0, 8, 16}, 100 {FREQ_RES_HIGH, FREQ_RES_HIGH}, 101 0, 102 2, 103 { 0, 8, 16} }; 104 105 static const SBR_FRAME_INFO frameInfo4_2048 = { 106 4, 107 { 0, 4, 8, 12, 16}, 108 {FREQ_RES_HIGH, FREQ_RES_HIGH, FREQ_RES_HIGH, FREQ_RES_HIGH}, 109 0, 110 2, 111 { 0, 8, 16} }; 112 113 static const SBR_FRAME_INFO frameInfo1_2304 = { 114 1, 115 { 0, 18}, 116 {FREQ_RES_HIGH}, 117 0, 118 1, 119 { 0, 18} }; 120 121 static const SBR_FRAME_INFO frameInfo2_2304 = { 122 2, 123 { 0, 9, 18}, 124 {FREQ_RES_HIGH, FREQ_RES_HIGH}, 125 0, 126 2, 127 { 0, 9, 18} }; 128 129 static const SBR_FRAME_INFO frameInfo4_2304 = { 130 4, 131 { 0, 5, 9, 14, 18}, 132 {FREQ_RES_HIGH, FREQ_RES_HIGH, FREQ_RES_HIGH, FREQ_RES_HIGH}, 133 0, 134 2, 135 { 0, 9, 18} }; 136 137 static const SBR_FRAME_INFO frameInfo1_1920 = { 138 1, 139 { 0, 15}, 140 {FREQ_RES_HIGH}, 141 0, 142 1, 143 { 0, 15} }; 144 145 static const SBR_FRAME_INFO frameInfo2_1920 = { 146 2, 147 { 0, 8, 15}, 148 {FREQ_RES_HIGH, FREQ_RES_HIGH}, 149 0, 150 2, 151 { 0, 8, 15} }; 152 153 static const SBR_FRAME_INFO frameInfo4_1920 = { 154 4, 155 { 0, 4, 8, 12, 15}, 156 {FREQ_RES_HIGH, FREQ_RES_HIGH, FREQ_RES_HIGH, FREQ_RES_HIGH}, 157 0, 158 2, 159 { 0, 8, 15} }; 160 161 static const SBR_FRAME_INFO frameInfo1_1152 = { 162 1, 163 { 0, 9}, 164 {FREQ_RES_HIGH}, 165 0, 166 1, 167 { 0, 9} }; 168 169 static const SBR_FRAME_INFO frameInfo2_1152 = { 170 2, 171 { 0, 5, 9}, 172 {FREQ_RES_HIGH, FREQ_RES_HIGH}, 173 0, 174 2, 175 { 0, 5, 9} }; 176 177 static const SBR_FRAME_INFO frameInfo4_1152 = { 178 4, 179 { 0, 2, 5, 180 7, 9}, 181 {FREQ_RES_HIGH, FREQ_RES_HIGH, FREQ_RES_HIGH, FREQ_RES_HIGH}, 182 0, 183 2, 184 { 0, 5, 9} }; 185 186 187 /* AACLD frame info */ 188 static const SBR_FRAME_INFO frameInfo1_512LD = { 189 1, 190 {0, 8}, 191 {FREQ_RES_HIGH}, 192 0, 193 1, 194 {0, 8}}; 195 196 static const SBR_FRAME_INFO frameInfo2_512LD = { 197 2, 198 {0, 4, 8}, 199 {FREQ_RES_HIGH, FREQ_RES_HIGH}, 200 0, 201 2, 202 {0, 4, 8}}; 203 204 static const SBR_FRAME_INFO frameInfo4_512LD = { 205 4, 206 {0, 2, 4, 6, 8}, 207 {FREQ_RES_HIGH, FREQ_RES_HIGH, FREQ_RES_HIGH, FREQ_RES_HIGH}, 208 0, 209 2, 210 {0, 4, 8}}; 211 212 static int 213 calcFillLengthMax (int tranPos, /*!< input : transient position (ref: tran det) */ 214 int numberTimeSlots /*!< input : number of timeslots */ 215 ); 216 217 static void 218 fillFrameTran (const int *v_tuningSegm, /*!< tuning: desired segment lengths */ 219 const int *v_tuningFreq, /*!< tuning: desired frequency resolutions */ 220 int tran, /*!< input : position of transient */ 221 int *v_bord, /*!< memNew: borders */ 222 int *length_v_bord, /*!< memNew: # borders */ 223 int *v_freq, /*!< memNew: frequency resolutions */ 224 int *length_v_freq, /*!< memNew: # frequency resolutions */ 225 int *bmin, /*!< hlpNew: first mandatory border */ 226 int *bmax /*!< hlpNew: last mandatory border */ 227 ); 228 229 static void fillFramePre (INT dmax, INT *v_bord, INT *length_v_bord, 230 INT *v_freq, INT *length_v_freq, INT bmin, 231 INT rest); 232 233 static void fillFramePost (INT *parts, INT *d, INT dmax, INT *v_bord, 234 INT *length_v_bord, INT *v_freq, 235 INT *length_v_freq, INT bmax, 236 INT bufferFrameStart, INT numberTimeSlots, INT fmax); 237 238 static void fillFrameInter (INT *nL, const int *v_tuningSegm, INT *v_bord, 239 INT *length_v_bord, INT bmin, INT *v_freq, 240 INT *length_v_freq, INT *v_bordFollow, 241 INT *length_v_bordFollow, INT *v_freqFollow, 242 INT *length_v_freqFollow, INT i_fillFollow, 243 INT dmin, INT dmax, INT numberTimeSlots); 244 245 static void calcFrameClass (FRAME_CLASS *frameClass, FRAME_CLASS *frameClassOld, INT tranFlag, 246 INT *spreadFlag); 247 248 static void specialCase (INT *spreadFlag, INT allowSpread, INT *v_bord, 249 INT *length_v_bord, INT *v_freq, INT *length_v_freq, 250 INT *parts, INT d); 251 252 static void calcCmonBorder (INT *i_cmon, INT *i_tran, INT *v_bord, 253 INT *length_v_bord, INT tran, 254 INT bufferFrameStart, INT numberTimeSlots); 255 256 static void keepForFollowUp (INT *v_bordFollow, INT *length_v_bordFollow, 257 INT *v_freqFollow, INT *length_v_freqFollow, 258 INT *i_tranFollow, INT *i_fillFollow, 259 INT *v_bord, INT *length_v_bord, INT *v_freq, 260 INT i_cmon, INT i_tran, INT parts, INT numberTimeSlots); 261 262 static void calcCtrlSignal (HANDLE_SBR_GRID hSbrGrid, FRAME_CLASS frameClass, 263 INT *v_bord, INT length_v_bord, INT *v_freq, 264 INT length_v_freq, INT i_cmon, INT i_tran, 265 INT spreadFlag, INT nL); 266 267 static void ctrlSignal2FrameInfo (HANDLE_SBR_GRID hSbrGrid, 268 HANDLE_SBR_FRAME_INFO hFrameInfo, 269 INT freq_res_fixfix); 270 271 272 /* table for 8 time slot index */ 273 static const int envelopeTable_8 [8][5] = { 274 /* transientIndex nEnv, tranIdx, shortEnv, border1, border2, ... */ 275 /* borders from left to right side; -1 = not in use */ 276 /*[|T-|------]*/ { 2, 0, 0, 1, -1 }, 277 /*[|-T-|-----]*/ { 2, 0, 0, 2, -1 }, 278 /*[--|T-|----]*/ { 3, 1, 1, 2, 4 }, 279 /*[---|T-|---]*/ { 3, 1, 1, 3, 5 }, 280 /*[----|T-|--]*/ { 3, 1, 1, 4, 6 }, 281 /*[-----|T--|]*/ { 2, 1, 1, 5, -1 }, 282 /*[------|T-|]*/ { 2, 1, 1, 6, -1 }, 283 /*[-------|T|]*/ { 2, 1, 1, 7, -1 }, 284 }; 285 286 /* table for 16 time slot index */ 287 static const int envelopeTable_16 [16][6] = { 288 /* transientIndex nEnv, tranIdx, shortEnv, border1, border2, ... */ 289 /* length from left to right side; -1 = not in use */ 290 /*[|T---|------------|]*/ { 2, 0, 0, 4, -1, -1}, 291 /*[|-T---|-----------|]*/ { 2, 0, 0, 5, -1, -1}, 292 /*[|--|T---|----------]*/ { 3, 1, 1, 2, 6, -1}, 293 /*[|---|T---|---------]*/ { 3, 1, 1, 3, 7, -1}, 294 /*[|----|T---|--------]*/ { 3, 1, 1, 4, 8, -1}, 295 /*[|-----|T---|-------]*/ { 3, 1, 1, 5, 9, -1}, 296 /*[|------|T---|------]*/ { 3, 1, 1, 6, 10, -1}, 297 /*[|-------|T---|-----]*/ { 3, 1, 1, 7, 11, -1}, 298 /*[|--------|T---|----]*/ { 3, 1, 1, 8, 12, -1}, 299 /*[|---------|T---|---]*/ { 3, 1, 1, 9, 13, -1}, 300 /*[|----------|T---|--]*/ { 3, 1, 1,10, 14, -1}, 301 /*[|-----------|T----|]*/ { 2, 1, 1,11, -1, -1}, 302 /*[|------------|T---|]*/ { 2, 1, 1,12, -1, -1}, 303 /*[|-------------|T--|]*/ { 2, 1, 1,13, -1, -1}, 304 /*[|--------------|T-|]*/ { 2, 1, 1,14, -1, -1}, 305 /*[|---------------|T|]*/ { 2, 1, 1,15, -1, -1}, 306 }; 307 308 /* table for 15 time slot index */ 309 static const int envelopeTable_15 [15][6] = { 310 /* transientIndex nEnv, tranIdx, shortEnv, border1, border2, ... */ 311 /* length from left to right side; -1 = not in use */ 312 /*[|T---|------------]*/ { 2, 0, 0, 4, -1, -1}, 313 /*[|-T---|-----------]*/ { 2, 0, 0, 5, -1, -1}, 314 /*[|--|T---|---------]*/ { 3, 1, 1, 2, 6, -1}, 315 /*[|---|T---|--------]*/ { 3, 1, 1, 3, 7, -1}, 316 /*[|----|T---|-------]*/ { 3, 1, 1, 4, 8, -1}, 317 /*[|-----|T---|------]*/ { 3, 1, 1, 5, 9, -1}, 318 /*[|------|T---|-----]*/ { 3, 1, 1, 6, 10, -1}, 319 /*[|-------|T---|----]*/ { 3, 1, 1, 7, 11, -1}, 320 /*[|--------|T---|---]*/ { 3, 1, 1, 8, 12, -1}, 321 /*[|---------|T---|--]*/ { 3, 1, 1, 9, 13, -1}, 322 /*[|----------|T----|]*/ { 2, 1, 1,10, -1, -1}, 323 /*[|-----------|T---|]*/ { 2, 1, 1,11, -1, -1}, 324 /*[|------------|T--|]*/ { 2, 1, 1,12, -1, -1}, 325 /*[|-------------|T-|]*/ { 2, 1, 1,13, -1, -1}, 326 /*[|--------------|T|]*/ { 2, 1, 1,14, -1, -1}, 327 }; 328 329 static const int minFrameTranDistance = 4; 330 331 static const FREQ_RES freqRes_table_8[] = {FREQ_RES_LOW, FREQ_RES_LOW, FREQ_RES_LOW, FREQ_RES_LOW, FREQ_RES_LOW, 332 FREQ_RES_HIGH, FREQ_RES_HIGH, FREQ_RES_HIGH, FREQ_RES_HIGH}; 333 334 static const FREQ_RES freqRes_table_16[16] = { 335 /* size of envelope */ 336 /* 0-4 */ FREQ_RES_LOW, FREQ_RES_LOW, FREQ_RES_LOW, FREQ_RES_LOW, FREQ_RES_LOW, 337 /* 5-9 */ FREQ_RES_LOW, FREQ_RES_HIGH, FREQ_RES_HIGH, FREQ_RES_HIGH, FREQ_RES_HIGH, 338 /* 10-16 */ FREQ_RES_HIGH, FREQ_RES_HIGH, FREQ_RES_HIGH, FREQ_RES_HIGH, FREQ_RES_HIGH, 339 FREQ_RES_HIGH }; 340 341 static void generateFixFixOnly ( HANDLE_SBR_FRAME_INFO hSbrFrameInfo, 342 HANDLE_SBR_GRID hSbrGrid, 343 int tranPosInternal, 344 int numberTimeSlots 345 ); 346 347 348 /*! 349 Functionname: FDKsbrEnc_frameInfoGenerator 350 351 Description: produces the FRAME_INFO struct for the current frame 352 353 Arguments: hSbrEnvFrame - pointer to sbr envelope handle 354 v_pre_transient_info - pointer to transient info vector 355 v_transient_info - pointer to previous transient info vector 356 v_tuning - pointer to tuning vector 357 358 Return: frame_info - pointer to SBR_FRAME_INFO struct 359 360 *******************************************************************************/ 361 HANDLE_SBR_FRAME_INFO 362 FDKsbrEnc_frameInfoGenerator (HANDLE_SBR_ENVELOPE_FRAME hSbrEnvFrame, 363 UCHAR *v_transient_info, 364 UCHAR *v_transient_info_pre, 365 int ldGrid, 366 const int *v_tuning) 367 { 368 INT numEnv, tranPosInternal=0, bmin=0, bmax=0, parts, d, i_cmon=0, i_tran=0, nL; 369 INT fmax = 0; 370 371 INT *v_bord = hSbrEnvFrame->v_bord; 372 INT *v_freq = hSbrEnvFrame->v_freq; 373 INT *v_bordFollow = hSbrEnvFrame->v_bordFollow; 374 INT *v_freqFollow = hSbrEnvFrame->v_freqFollow; 375 376 377 INT *length_v_bordFollow = &hSbrEnvFrame->length_v_bordFollow; 378 INT *length_v_freqFollow = &hSbrEnvFrame->length_v_freqFollow; 379 INT *length_v_bord = &hSbrEnvFrame->length_v_bord; 380 INT *length_v_freq = &hSbrEnvFrame->length_v_freq; 381 INT *spreadFlag = &hSbrEnvFrame->spreadFlag; 382 INT *i_tranFollow = &hSbrEnvFrame->i_tranFollow; 383 INT *i_fillFollow = &hSbrEnvFrame->i_fillFollow; 384 FRAME_CLASS *frameClassOld = &hSbrEnvFrame->frameClassOld; 385 FRAME_CLASS frameClass = FIXFIX; 386 387 388 INT allowSpread = hSbrEnvFrame->allowSpread; 389 INT numEnvStatic = hSbrEnvFrame->numEnvStatic; 390 INT staticFraming = hSbrEnvFrame->staticFraming; 391 INT dmin = hSbrEnvFrame->dmin; 392 INT dmax = hSbrEnvFrame->dmax; 393 394 INT bufferFrameStart = hSbrEnvFrame->SbrGrid.bufferFrameStart; 395 INT numberTimeSlots = hSbrEnvFrame->SbrGrid.numberTimeSlots; 396 INT frameMiddleSlot = hSbrEnvFrame->frameMiddleSlot; 397 398 INT tranPos = v_transient_info[0]; 399 INT tranFlag = v_transient_info[1]; 400 401 const int *v_tuningSegm = v_tuning; 402 const int *v_tuningFreq = v_tuning + 3; 403 404 hSbrEnvFrame->v_tuningSegm = v_tuningSegm; 405 INT freq_res_fixfix = hSbrEnvFrame->freq_res_fixfix; 406 407 if (ldGrid) { 408 /* in case there was a transient at the very end of the previous frame, start with a transient envelope */ 409 if(v_transient_info_pre[1] && (numberTimeSlots - v_transient_info_pre[0] < minFrameTranDistance)){ 410 tranFlag = 1; 411 tranPos = 0; 412 } 413 } 414 415 /* 416 * Synopsis: 417 * 418 * The frame generator creates the time-/frequency-grid for one SBR frame. 419 * Input signals are provided by the transient detector and the frame 420 * splitter (transientDetectNew() & FrameSplitter() in tran_det.c). The 421 * framing is controlled by adjusting tuning parameters stored in 422 * FRAME_GEN_TUNING. The parameter values are dependent on frame lengths 423 * and bitrates, and may in the future be signal dependent. 424 * 425 * The envelope borders are stored for frame generator internal use in 426 * aBorders. The contents of aBorders represent positions along the time 427 * axis given in the figures in fram_gen.h (the "frame-generator" rows). 428 * The unit is "time slot". The figures in fram_gen.h also define the 429 * detection ranges for the transient detector. For every border in 430 * aBorders, there is a corresponding entry in aFreqRes, which defines the 431 * frequency resolution of the envelope following (delimited by) the 432 * border. 433 * 434 * When no transients are present, FIXFIX class frames are used. The 435 * frame splitter decides whether to use one or two envelopes in the 436 * FIXFIX frame. "Sparse transients" (separated by a few frames without 437 * transients) are handeled by [FIXVAR, VARFIX] pairs or (depending on 438 * tuning and transient position relative the nominal frame boundaries) 439 * by [FIXVAR, VARVAR, VARFIX] triples. "Tight transients" (in 440 * consecutive frames) are handeled by [..., VARVAR, VARVAR, ...] 441 * sequences. 442 * 443 * The generator assumes that transients are "sparse", and designs 444 * borders for [FIXVAR, VARFIX] pairs right away, where the first frame 445 * corresponds to the present frame. At the next call of the generator 446 * it is known whether the transient actually is "sparse" or not. If 447 * 'yes', the already calculated VARFIX borders are used. If 'no', new 448 * borders, meeting the requirements of the "tight" transient, are 449 * calculated. 450 * 451 * The generator produces two outputs: A "clear-text bitstream" stored in 452 * SBR_GRID, and a straight-forward representation of the grid stored in 453 * SBR_FRAME_INFO. The former is subsequently converted to the actual 454 * bitstream sbr_grid() (encodeSbrGrid() in bit_sbr.c). The latter is 455 * used by other encoder functions, such as the envelope estimator 456 * (calculateSbrEnvelope() in env_est.c) and the noise floor and missing 457 * harmonics detector (TonCorrParamExtr() in nf_est.c). 458 */ 459 460 if (staticFraming) { 461 /*-------------------------------------------------------------------------- 462 Ignore transient detector 463 ---------------------------------------------------------------------------*/ 464 465 frameClass = FIXFIX; 466 numEnv = numEnvStatic; /* {1,2,4,8} */ 467 *frameClassOld = FIXFIX; /* for change to dyn */ 468 hSbrEnvFrame->SbrGrid.bs_num_env = numEnv; 469 hSbrEnvFrame->SbrGrid.frameClass = frameClass; 470 } 471 else { 472 /*-------------------------------------------------------------------------- 473 Calculate frame class to use 474 ---------------------------------------------------------------------------*/ 475 calcFrameClass (&frameClass, frameClassOld, tranFlag, spreadFlag); 476 477 /* patch for new frame class FIXFIXonly for AAC LD */ 478 if (tranFlag && ldGrid) { 479 frameClass = FIXFIXonly; 480 *frameClassOld = FIXFIX; 481 } 482 483 /* 484 * every transient is processed below by inserting 485 * 486 * - one border at the onset of the transient 487 * - one or more "decay borders" (after the onset of the transient) 488 * - optionally one "attack border" (before the onset of the transient) 489 * 490 * those borders are referred to as "mandatory borders" and are 491 * defined by the 'segmentLength' array in FRAME_GEN_TUNING 492 * 493 * the frequency resolutions of the corresponding envelopes are 494 * defined by the 'segmentRes' array in FRAME_GEN_TUNING 495 */ 496 497 /*-------------------------------------------------------------------------- 498 Design frame (or follow-up old design) 499 ---------------------------------------------------------------------------*/ 500 if (tranFlag) { /* Always for FixVar, often but not always for VarVar */ 501 /*-------------------------------------------------------------------------- 502 Design part of T/F-grid around the new transient 503 ---------------------------------------------------------------------------*/ 504 505 tranPosInternal = frameMiddleSlot + tranPos + bufferFrameStart ; /* FH 00-06-26 */ 506 /* 507 add mandatory borders around transient 508 */ 509 510 fillFrameTran ( v_tuningSegm, 511 v_tuningFreq, 512 tranPosInternal, 513 v_bord, 514 length_v_bord, 515 v_freq, 516 length_v_freq, 517 &bmin, 518 &bmax ); 519 520 /* make sure we stay within the maximum SBR frame overlap */ 521 fmax = calcFillLengthMax(tranPos, numberTimeSlots); 522 } 523 524 switch (frameClass) { 525 526 case FIXFIXonly: 527 FDK_ASSERT(ldGrid); 528 tranPosInternal = tranPos; 529 generateFixFixOnly ( &(hSbrEnvFrame->SbrFrameInfo), 530 &(hSbrEnvFrame->SbrGrid), 531 tranPosInternal, 532 numberTimeSlots 533 ); 534 535 return &(hSbrEnvFrame->SbrFrameInfo); 536 537 case FIXVAR: 538 539 /*-------------------------------------------------------------------------- 540 Design remaining parts of T/F-grid (assuming next frame is VarFix) 541 ---------------------------------------------------------------------------*/ 542 543 /*-------------------------------------------------------------------------- 544 Fill region before new transient: 545 ---------------------------------------------------------------------------*/ 546 fillFramePre (dmax, v_bord, length_v_bord, v_freq, length_v_freq, 547 bmin, bmin - bufferFrameStart); /* FH 00-06-26 */ 548 549 /*-------------------------------------------------------------------------- 550 Fill region after new transient: 551 ---------------------------------------------------------------------------*/ 552 fillFramePost (&parts, &d, dmax, v_bord, length_v_bord, v_freq, 553 length_v_freq, bmax, bufferFrameStart, numberTimeSlots, fmax); 554 555 /*-------------------------------------------------------------------------- 556 Take care of special case: 557 ---------------------------------------------------------------------------*/ 558 if (parts == 1 && d < dmin) /* no fill, short last envelope */ 559 specialCase (spreadFlag, allowSpread, v_bord, length_v_bord, 560 v_freq, length_v_freq, &parts, d); 561 562 /*-------------------------------------------------------------------------- 563 Calculate common border (split-point) 564 ---------------------------------------------------------------------------*/ 565 calcCmonBorder (&i_cmon, &i_tran, v_bord, length_v_bord, tranPosInternal, 566 bufferFrameStart, numberTimeSlots); /* FH 00-06-26 */ 567 568 /*-------------------------------------------------------------------------- 569 Extract data for proper follow-up in next frame 570 ---------------------------------------------------------------------------*/ 571 keepForFollowUp (v_bordFollow, length_v_bordFollow, v_freqFollow, 572 length_v_freqFollow, i_tranFollow, i_fillFollow, 573 v_bord, length_v_bord, v_freq, i_cmon, i_tran, parts, numberTimeSlots); /* FH 00-06-26 */ 574 575 /*-------------------------------------------------------------------------- 576 Calculate control signal 577 ---------------------------------------------------------------------------*/ 578 calcCtrlSignal (&hSbrEnvFrame->SbrGrid, frameClass, 579 v_bord, *length_v_bord, v_freq, *length_v_freq, 580 i_cmon, i_tran, *spreadFlag, DC); 581 break; 582 case VARFIX: 583 /*-------------------------------------------------------------------------- 584 Follow-up old transient - calculate control signal 585 ---------------------------------------------------------------------------*/ 586 calcCtrlSignal (&hSbrEnvFrame->SbrGrid, frameClass, 587 v_bordFollow, *length_v_bordFollow, v_freqFollow, 588 *length_v_freqFollow, DC, *i_tranFollow, 589 *spreadFlag, DC); 590 break; 591 case VARVAR: 592 if (*spreadFlag) { /* spread across three frames */ 593 /*-------------------------------------------------------------------------- 594 Follow-up old transient - calculate control signal 595 ---------------------------------------------------------------------------*/ 596 calcCtrlSignal (&hSbrEnvFrame->SbrGrid, 597 frameClass, v_bordFollow, *length_v_bordFollow, 598 v_freqFollow, *length_v_freqFollow, DC, 599 *i_tranFollow, *spreadFlag, DC); 600 601 *spreadFlag = 0; 602 603 /*-------------------------------------------------------------------------- 604 Extract data for proper follow-up in next frame 605 ---------------------------------------------------------------------------*/ 606 v_bordFollow[0] = hSbrEnvFrame->SbrGrid.bs_abs_bord_1 - numberTimeSlots; /* FH 00-06-26 */ 607 v_freqFollow[0] = 1; 608 *length_v_bordFollow = 1; 609 *length_v_freqFollow = 1; 610 611 *i_tranFollow = -DC; 612 *i_fillFollow = -DC; 613 } 614 else { 615 /*-------------------------------------------------------------------------- 616 Design remaining parts of T/F-grid (assuming next frame is VarFix) 617 adapt or fill region before new transient: 618 ---------------------------------------------------------------------------*/ 619 fillFrameInter (&nL, v_tuningSegm, v_bord, length_v_bord, bmin, 620 v_freq, length_v_freq, v_bordFollow, 621 length_v_bordFollow, v_freqFollow, 622 length_v_freqFollow, *i_fillFollow, dmin, dmax, 623 numberTimeSlots); 624 625 /*-------------------------------------------------------------------------- 626 Fill after transient: 627 ---------------------------------------------------------------------------*/ 628 fillFramePost (&parts, &d, dmax, v_bord, length_v_bord, v_freq, 629 length_v_freq, bmax, bufferFrameStart, numberTimeSlots, fmax); 630 631 /*-------------------------------------------------------------------------- 632 Take care of special case: 633 ---------------------------------------------------------------------------*/ 634 if (parts == 1 && d < dmin) /*% no fill, short last envelope */ 635 specialCase (spreadFlag, allowSpread, v_bord, length_v_bord, 636 v_freq, length_v_freq, &parts, d); 637 638 /*-------------------------------------------------------------------------- 639 Calculate common border (split-point) 640 ---------------------------------------------------------------------------*/ 641 calcCmonBorder (&i_cmon, &i_tran, v_bord, length_v_bord, tranPosInternal, 642 bufferFrameStart, numberTimeSlots); 643 644 /*-------------------------------------------------------------------------- 645 Extract data for proper follow-up in next frame 646 ---------------------------------------------------------------------------*/ 647 keepForFollowUp (v_bordFollow, length_v_bordFollow, 648 v_freqFollow, length_v_freqFollow, 649 i_tranFollow, i_fillFollow, v_bord, 650 length_v_bord, v_freq, i_cmon, i_tran, parts, numberTimeSlots); 651 652 /*-------------------------------------------------------------------------- 653 Calculate control signal 654 ---------------------------------------------------------------------------*/ 655 calcCtrlSignal (&hSbrEnvFrame->SbrGrid, 656 frameClass, v_bord, *length_v_bord, v_freq, 657 *length_v_freq, i_cmon, i_tran, 0, nL); 658 } 659 break; 660 case FIXFIX: 661 if (tranPos == 0) 662 numEnv = 1; 663 else 664 numEnv = 2; 665 666 hSbrEnvFrame->SbrGrid.bs_num_env = numEnv; 667 hSbrEnvFrame->SbrGrid.frameClass = frameClass; 668 669 break; 670 default: 671 FDK_ASSERT(0); 672 } 673 } 674 675 /*------------------------------------------------------------------------- 676 Convert control signal to frame info struct 677 ---------------------------------------------------------------------------*/ 678 ctrlSignal2FrameInfo (&hSbrEnvFrame->SbrGrid, 679 &hSbrEnvFrame->SbrFrameInfo, 680 freq_res_fixfix); 681 682 return &hSbrEnvFrame->SbrFrameInfo; 683 } 684 685 686 /***************************************************************************/ 687 /*! 688 \brief Gnerates frame info for FIXFIXonly frame class used for low delay version 689 690 \return nothing 691 ****************************************************************************/ 692 static void generateFixFixOnly ( HANDLE_SBR_FRAME_INFO hSbrFrameInfo, 693 HANDLE_SBR_GRID hSbrGrid, 694 int tranPosInternal, 695 int numberTimeSlots 696 ) 697 { 698 int nEnv, i, k=0, tranIdx; 699 const int *pTable = NULL; 700 const FREQ_RES *freqResTable = NULL; 701 702 switch (numberTimeSlots) { 703 case 8: 704 pTable = envelopeTable_8[tranPosInternal]; 705 freqResTable = freqRes_table_8; 706 break; 707 case 15: 708 pTable = envelopeTable_15[tranPosInternal]; 709 freqResTable = freqRes_table_16; 710 break; 711 case 16: 712 pTable = envelopeTable_16[tranPosInternal]; 713 freqResTable = freqRes_table_16; 714 break; 715 } 716 717 /* look number of envolpes in table */ 718 nEnv = pTable[0]; 719 /* look up envolpe distribution in table */ 720 for (i=1; i<nEnv; i++) 721 hSbrFrameInfo->borders[i] = pTable[i+2]; 722 723 /* open and close frame border */ 724 hSbrFrameInfo->borders[0] = 0; 725 hSbrFrameInfo->borders[nEnv] = numberTimeSlots; 726 727 /* adjust segment-frequency-resolution according to the segment-length */ 728 for (i=0; i<nEnv; i++){ 729 k = hSbrFrameInfo->borders[i+1] - hSbrFrameInfo->borders[i]; 730 hSbrFrameInfo->freqRes[i] = freqResTable[k]; 731 hSbrGrid->v_f[i] = freqResTable[k]; 732 } 733 734 hSbrFrameInfo->nEnvelopes = nEnv; 735 hSbrFrameInfo->shortEnv = pTable[2]; 736 /* transient idx */ 737 tranIdx = pTable[1]; 738 739 /* add noise floors */ 740 hSbrFrameInfo->bordersNoise[0] = 0; 741 hSbrFrameInfo->bordersNoise[1] = hSbrFrameInfo->borders[tranIdx?tranIdx:1]; 742 hSbrFrameInfo->bordersNoise[2] = numberTimeSlots; 743 hSbrFrameInfo->nNoiseEnvelopes = 2; 744 745 hSbrGrid->frameClass = FIXFIXonly; 746 hSbrGrid->bs_abs_bord = tranPosInternal; 747 hSbrGrid->bs_num_env = nEnv; 748 749 } 750 751 752 753 /******************************************************************************* 754 Functionname: FDKsbrEnc_initFrameInfoGenerator 755 ******************************************************************************* 756 757 Description: 758 759 Arguments: hSbrEnvFrame - pointer to sbr envelope handle 760 allowSpread - commandline parameter 761 numEnvStatic - commandline parameter 762 staticFraming - commandline parameter 763 764 Return: none 765 766 *******************************************************************************/ 767 void 768 FDKsbrEnc_initFrameInfoGenerator (HANDLE_SBR_ENVELOPE_FRAME hSbrEnvFrame, 769 INT allowSpread, 770 INT numEnvStatic, 771 INT staticFraming, 772 INT timeSlots, 773 INT freq_res_fixfix 774 ,int ldGrid 775 ) 776 777 { /* FH 00-06-26 */ 778 779 FDKmemclear(hSbrEnvFrame,sizeof(SBR_ENVELOPE_FRAME )); 780 781 782 /* Initialisation */ 783 hSbrEnvFrame->frameClassOld = FIXFIX; 784 hSbrEnvFrame->spreadFlag = 0; 785 786 hSbrEnvFrame->allowSpread = allowSpread; 787 hSbrEnvFrame->numEnvStatic = numEnvStatic; 788 hSbrEnvFrame->staticFraming = staticFraming; 789 hSbrEnvFrame->freq_res_fixfix = freq_res_fixfix; 790 791 hSbrEnvFrame->length_v_bord = 0; 792 hSbrEnvFrame->length_v_bordFollow = 0; 793 794 hSbrEnvFrame->length_v_freq = 0; 795 hSbrEnvFrame->length_v_freqFollow = 0; 796 797 hSbrEnvFrame->i_tranFollow = 0; 798 hSbrEnvFrame->i_fillFollow = 0; 799 800 hSbrEnvFrame->SbrGrid.numberTimeSlots = timeSlots; 801 802 if (ldGrid) { 803 /*case CODEC_AACLD:*/ 804 hSbrEnvFrame->dmin = 2; 805 hSbrEnvFrame->dmax = 16; 806 hSbrEnvFrame->frameMiddleSlot = FRAME_MIDDLE_SLOT_512LD; 807 } else 808 switch(timeSlots){ 809 case NUMBER_TIME_SLOTS_1920: 810 hSbrEnvFrame->dmin = 4; 811 hSbrEnvFrame->dmax = 12; 812 hSbrEnvFrame->SbrGrid.bufferFrameStart = 0; 813 hSbrEnvFrame->frameMiddleSlot = FRAME_MIDDLE_SLOT_1920; 814 break; 815 case NUMBER_TIME_SLOTS_2048: 816 hSbrEnvFrame->dmin = 4; 817 hSbrEnvFrame->dmax = 12; 818 hSbrEnvFrame->SbrGrid.bufferFrameStart = 0; 819 hSbrEnvFrame->frameMiddleSlot = FRAME_MIDDLE_SLOT_2048; 820 break; 821 case NUMBER_TIME_SLOTS_1152: 822 hSbrEnvFrame->dmin = 2; 823 hSbrEnvFrame->dmax = 8; 824 hSbrEnvFrame->SbrGrid.bufferFrameStart = 0; 825 hSbrEnvFrame->frameMiddleSlot = FRAME_MIDDLE_SLOT_1152; 826 break; 827 case NUMBER_TIME_SLOTS_2304: 828 hSbrEnvFrame->dmin = 4; 829 hSbrEnvFrame->dmax = 15; 830 hSbrEnvFrame->SbrGrid.bufferFrameStart = 0; 831 hSbrEnvFrame->frameMiddleSlot = FRAME_MIDDLE_SLOT_2304; 832 break; 833 default: 834 FDK_ASSERT(0); 835 } 836 837 } 838 839 840 /******************************************************************************* 841 Functionname: fillFrameTran 842 ******************************************************************************* 843 844 Description: Add mandatory borders, as described by the tuning vector 845 and the current transient position 846 847 Arguments: 848 modified: 849 v_bord - int pointer to v_bord vector 850 length_v_bord - length of v_bord vector 851 v_freq - int pointer to v_freq vector 852 length_v_freq - length of v_freq vector 853 bmin - int pointer to bmin (call by reference) 854 bmax - int pointer to bmax (call by reference) 855 not modified: 856 tran - position of transient 857 v_tuningSegm - int pointer to v_tuningSegm vector 858 v_tuningFreq - int pointer to v_tuningFreq vector 859 860 Return: none 861 862 *******************************************************************************/ 863 static void 864 fillFrameTran (const int *v_tuningSegm, /*!< tuning: desired segment lengths */ 865 const int *v_tuningFreq, /*!< tuning: desired frequency resolutions */ 866 int tran, /*!< input : position of transient */ 867 int *v_bord, /*!< memNew: borders */ 868 int *length_v_bord, /*!< memNew: # borders */ 869 int *v_freq, /*!< memNew: frequency resolutions */ 870 int *length_v_freq, /*!< memNew: # frequency resolutions */ 871 int *bmin, /*!< hlpNew: first mandatory border */ 872 int *bmax /*!< hlpNew: last mandatory border */ 873 ) 874 { 875 int bord, i; 876 877 *length_v_bord = 0; 878 *length_v_freq = 0; 879 880 /* add attack env leading border (optional) */ 881 if (v_tuningSegm[0]) { 882 /* v_bord = [(Ba)] start of attack env */ 883 FDKsbrEnc_AddRight (v_bord, length_v_bord, (tran - v_tuningSegm[0])); 884 885 /* v_freq = [(Fa)] res of attack env */ 886 FDKsbrEnc_AddRight (v_freq, length_v_freq, v_tuningFreq[0]); 887 } 888 889 /* add attack env trailing border/first decay env leading border */ 890 bord = tran; 891 FDKsbrEnc_AddRight (v_bord, length_v_bord, tran); /* v_bord = [(Ba),Bd1] */ 892 893 /* add first decay env trailing border/2:nd decay env leading border */ 894 if (v_tuningSegm[1]) { 895 bord += v_tuningSegm[1]; 896 897 /* v_bord = [(Ba),Bd1,Bd2] */ 898 FDKsbrEnc_AddRight (v_bord, length_v_bord, bord); 899 900 /* v_freq = [(Fa),Fd1] */ 901 FDKsbrEnc_AddRight (v_freq, length_v_freq, v_tuningFreq[1]); 902 } 903 904 /* add 2:nd decay env trailing border (optional) */ 905 if (v_tuningSegm[2] != 0) { 906 bord += v_tuningSegm[2]; 907 908 /* v_bord = [(Ba),Bd1, Bd2,(Bd3)] */ 909 FDKsbrEnc_AddRight (v_bord, length_v_bord, bord); 910 911 /* v_freq = [(Fa),Fd1,(Fd2)] */ 912 FDKsbrEnc_AddRight (v_freq, length_v_freq, v_tuningFreq[2]); 913 } 914 915 /* v_freq = [(Fa),Fd1,(Fd2),1] */ 916 FDKsbrEnc_AddRight (v_freq, length_v_freq, 1); 917 918 919 /* calc min and max values of mandatory borders */ 920 *bmin = v_bord[0]; 921 for (i = 0; i < *length_v_bord; i++) 922 if (v_bord[i] < *bmin) 923 *bmin = v_bord[i]; 924 925 *bmax = v_bord[0]; 926 for (i = 0; i < *length_v_bord; i++) 927 if (v_bord[i] > *bmax) 928 *bmax = v_bord[i]; 929 930 } 931 932 933 934 /******************************************************************************* 935 Functionname: fillFramePre 936 ******************************************************************************* 937 938 Description: Add borders before mandatory borders, if needed 939 940 Arguments: 941 modified: 942 v_bord - int pointer to v_bord vector 943 length_v_bord - length of v_bord vector 944 v_freq - int pointer to v_freq vector 945 length_v_freq - length of v_freq vector 946 not modified: 947 dmax - int value 948 bmin - int value 949 rest - int value 950 951 Return: none 952 953 *******************************************************************************/ 954 static void 955 fillFramePre (INT dmax, 956 INT *v_bord, INT *length_v_bord, 957 INT *v_freq, INT *length_v_freq, 958 INT bmin, INT rest) 959 { 960 /* 961 input state: 962 v_bord = [(Ba),Bd1, Bd2 ,(Bd3)] 963 v_freq = [(Fa),Fd1,(Fd2),1 ] 964 */ 965 966 INT parts, d, j, S, s = 0, segm, bord; 967 968 /* 969 start with one envelope 970 */ 971 972 parts = 1; 973 d = rest; 974 975 /* 976 calc # of additional envelopes and corresponding lengths 977 */ 978 979 while (d > dmax) { 980 parts++; 981 982 segm = rest / parts; 983 S = (segm - 2)>>1; 984 s = fixMin (8, 2 * S + 2); 985 d = rest - (parts - 1) * s; 986 } 987 988 /* 989 add borders before mandatory borders 990 */ 991 992 bord = bmin; 993 994 for (j = 0; j <= parts - 2; j++) { 995 bord = bord - s; 996 997 /* v_bord = [...,(Bf),(Ba),Bd1, Bd2 ,(Bd3)] */ 998 FDKsbrEnc_AddLeft (v_bord, length_v_bord, bord); 999 1000 /* v_freq = [...,(1 ),(Fa),Fd1,(Fd2), 1 ] */ 1001 FDKsbrEnc_AddLeft (v_freq, length_v_freq, 1); 1002 } 1003 } 1004 1005 /***************************************************************************/ 1006 /*! 1007 \brief Overlap control 1008 1009 Calculate max length of trailing fill segments, such that we always get a 1010 border within the frame overlap region 1011 1012 \return void 1013 1014 ****************************************************************************/ 1015 static int 1016 calcFillLengthMax (int tranPos, /*!< input : transient position (ref: tran det) */ 1017 int numberTimeSlots /*!< input : number of timeslots */ 1018 ) 1019 { 1020 int fmax; 1021 1022 /* 1023 calculate transient position within envelope buffer 1024 */ 1025 switch (numberTimeSlots) 1026 { 1027 case NUMBER_TIME_SLOTS_2048: 1028 if (tranPos < 4) 1029 fmax = 6; 1030 else if (tranPos == 4 || tranPos == 5) 1031 fmax = 4; 1032 else 1033 fmax = 8; 1034 break; 1035 1036 case NUMBER_TIME_SLOTS_1920: 1037 if (tranPos < 4) 1038 fmax = 5; 1039 else if (tranPos == 4 || tranPos == 5) 1040 fmax = 3; 1041 else 1042 fmax = 7; 1043 break; 1044 1045 default: 1046 fmax = 8; 1047 break; 1048 } 1049 1050 return fmax; 1051 } 1052 1053 /******************************************************************************* 1054 Functionname: fillFramePost 1055 ******************************************************************************* 1056 1057 Description: -Add borders after mandatory borders, if needed 1058 Make a preliminary design of next frame, 1059 assuming no transient is present there 1060 1061 Arguments: 1062 modified: 1063 parts - int pointer to parts (call by reference) 1064 d - int pointer to d (call by reference) 1065 v_bord - int pointer to v_bord vector 1066 length_v_bord - length of v_bord vector 1067 v_freq - int pointer to v_freq vector 1068 length_v_freq - length of v_freq vector 1069 not modified: 1070 bmax - int value 1071 dmax - int value 1072 1073 Return: none 1074 1075 *******************************************************************************/ 1076 static void 1077 fillFramePost (INT *parts, INT *d, INT dmax, INT *v_bord, INT *length_v_bord, 1078 INT *v_freq, INT *length_v_freq, INT bmax, 1079 INT bufferFrameStart, INT numberTimeSlots, INT fmax) 1080 { 1081 INT j, rest, segm, S, s = 0, bord; 1082 1083 /* 1084 input state: 1085 v_bord = [...,(Bf),(Ba),Bd1, Bd2 ,(Bd3)] 1086 v_freq = [...,(1 ),(Fa),Fd1,(Fd2),1 ] 1087 */ 1088 1089 rest = bufferFrameStart + 2 * numberTimeSlots - bmax; 1090 *d = rest; 1091 1092 if (*d > 0) { 1093 *parts = 1; /* start with one envelope */ 1094 1095 /* calc # of additional envelopes and corresponding lengths */ 1096 1097 while (*d > dmax) { 1098 *parts = *parts + 1; 1099 1100 segm = rest / (*parts); 1101 S = (segm - 2)>>1; 1102 s = fixMin (fmax, 2 * S + 2); 1103 *d = rest - (*parts - 1) * s; 1104 } 1105 1106 /* add borders after mandatory borders */ 1107 1108 bord = bmax; 1109 for (j = 0; j <= *parts - 2; j++) { 1110 bord += s; 1111 1112 /* v_bord = [...,(Bf),(Ba),Bd1, Bd2 ,(Bd3),(Bf)] */ 1113 FDKsbrEnc_AddRight (v_bord, length_v_bord, bord); 1114 1115 /* v_freq = [...,(1 ),(Fa),Fd1,(Fd2), 1 , 1! ,1] */ 1116 FDKsbrEnc_AddRight (v_freq, length_v_freq, 1); 1117 } 1118 } 1119 else { 1120 *parts = 1; 1121 1122 /* remove last element from v_bord and v_freq */ 1123 1124 *length_v_bord = *length_v_bord - 1; 1125 *length_v_freq = *length_v_freq - 1; 1126 1127 } 1128 } 1129 1130 1131 1132 /******************************************************************************* 1133 Functionname: fillFrameInter 1134 ******************************************************************************* 1135 1136 Description: 1137 1138 Arguments: nL - 1139 v_tuningSegm - 1140 v_bord - 1141 length_v_bord - 1142 bmin - 1143 v_freq - 1144 length_v_freq - 1145 v_bordFollow - 1146 length_v_bordFollow - 1147 v_freqFollow - 1148 length_v_freqFollow - 1149 i_fillFollow - 1150 dmin - 1151 dmax - 1152 1153 Return: none 1154 1155 *******************************************************************************/ 1156 static void 1157 fillFrameInter (INT *nL, const int *v_tuningSegm, INT *v_bord, INT *length_v_bord, 1158 INT bmin, INT *v_freq, INT *length_v_freq, INT *v_bordFollow, 1159 INT *length_v_bordFollow, INT *v_freqFollow, 1160 INT *length_v_freqFollow, INT i_fillFollow, INT dmin, 1161 INT dmax, INT numberTimeSlots) 1162 { 1163 INT middle, b_new, numBordFollow, bordMaxFollow, i; 1164 1165 if (numberTimeSlots != NUMBER_TIME_SLOTS_1152) { 1166 1167 /* % remove fill borders: */ 1168 if (i_fillFollow >= 1) { 1169 *length_v_bordFollow = i_fillFollow; 1170 *length_v_freqFollow = i_fillFollow; 1171 } 1172 1173 numBordFollow = *length_v_bordFollow; 1174 bordMaxFollow = v_bordFollow[numBordFollow - 1]; 1175 1176 /* remove even more borders if needed */ 1177 middle = bmin - bordMaxFollow; 1178 while (middle < 0) { 1179 numBordFollow--; 1180 bordMaxFollow = v_bordFollow[numBordFollow - 1]; 1181 middle = bmin - bordMaxFollow; 1182 } 1183 1184 *length_v_bordFollow = numBordFollow; 1185 *length_v_freqFollow = numBordFollow; 1186 *nL = numBordFollow - 1; 1187 1188 b_new = *length_v_bord; 1189 1190 1191 if (middle <= dmax) { 1192 if (middle >= dmin) { /* concatenate */ 1193 FDKsbrEnc_AddVecLeft (v_bord, length_v_bord, v_bordFollow, *length_v_bordFollow); 1194 FDKsbrEnc_AddVecLeft (v_freq, length_v_freq, v_freqFollow, *length_v_freqFollow); 1195 } 1196 1197 else { 1198 if (v_tuningSegm[0] != 0) { /* remove one new border and concatenate */ 1199 *length_v_bord = b_new - 1; 1200 FDKsbrEnc_AddVecLeft (v_bord, length_v_bord, v_bordFollow, 1201 *length_v_bordFollow); 1202 1203 *length_v_freq = b_new - 1; 1204 FDKsbrEnc_AddVecLeft (v_freq + 1, length_v_freq, v_freqFollow, 1205 *length_v_freqFollow); 1206 } 1207 else { 1208 if (*length_v_bordFollow > 1) { /* remove one old border and concatenate */ 1209 FDKsbrEnc_AddVecLeft (v_bord, length_v_bord, v_bordFollow, 1210 *length_v_bordFollow - 1); 1211 FDKsbrEnc_AddVecLeft (v_freq, length_v_freq, v_freqFollow, 1212 *length_v_bordFollow - 1); 1213 1214 *nL = *nL - 1; 1215 } 1216 else { /* remove new "transient" border and concatenate */ 1217 1218 for (i = 0; i < *length_v_bord - 1; i++) 1219 v_bord[i] = v_bord[i + 1]; 1220 1221 for (i = 0; i < *length_v_freq - 1; i++) 1222 v_freq[i] = v_freq[i + 1]; 1223 1224 *length_v_bord = b_new - 1; 1225 *length_v_freq = b_new - 1; 1226 1227 FDKsbrEnc_AddVecLeft (v_bord, length_v_bord, v_bordFollow, 1228 *length_v_bordFollow); 1229 FDKsbrEnc_AddVecLeft (v_freq, length_v_freq, v_freqFollow, 1230 *length_v_freqFollow); 1231 } 1232 } 1233 } 1234 } 1235 else { /* middle > dmax */ 1236 1237 fillFramePre (dmax, v_bord, length_v_bord, v_freq, length_v_freq, bmin, 1238 middle); 1239 FDKsbrEnc_AddVecLeft (v_bord, length_v_bord, v_bordFollow, *length_v_bordFollow); 1240 FDKsbrEnc_AddVecLeft (v_freq, length_v_freq, v_freqFollow, *length_v_freqFollow); 1241 } 1242 1243 1244 } 1245 else { /* numberTimeSlots==NUMBER_TIME_SLOTS_1152 */ 1246 1247 INT l,m; 1248 1249 1250 /*------------------------------------------------------------------------ 1251 remove fill borders 1252 ------------------------------------------------------------------------*/ 1253 if (i_fillFollow >= 1) { 1254 *length_v_bordFollow = i_fillFollow; 1255 *length_v_freqFollow = i_fillFollow; 1256 } 1257 1258 numBordFollow = *length_v_bordFollow; 1259 bordMaxFollow = v_bordFollow[numBordFollow - 1]; 1260 1261 /*------------------------------------------------------------------------ 1262 remove more borders if necessary to eliminate overlap 1263 ------------------------------------------------------------------------*/ 1264 1265 /* check for overlap */ 1266 middle = bmin - bordMaxFollow; 1267 1268 /* intervals: 1269 i) middle < 0 : overlap, must remove borders 1270 ii) 0 <= middle < dmin : no overlap but too tight, must remove borders 1271 iii) dmin <= middle <= dmax : ok, just concatenate 1272 iv) dmax <= middle : too wide, must add borders 1273 */ 1274 1275 /* first remove old non-fill-borders... */ 1276 while (middle < 0) { 1277 1278 /* ...but don't remove all of them */ 1279 if (numBordFollow == 1) 1280 break; 1281 1282 numBordFollow--; 1283 bordMaxFollow = v_bordFollow[numBordFollow - 1]; 1284 middle = bmin - bordMaxFollow; 1285 } 1286 1287 /* if this isn't enough, remove new non-fill borders */ 1288 if (middle < 0) 1289 { 1290 for (l = 0, m = 0 ; l < *length_v_bord ; l++) 1291 { 1292 if(v_bord[l]> bordMaxFollow) 1293 { 1294 v_bord[m] = v_bord[l]; 1295 v_freq[m] = v_freq[l]; 1296 m++; 1297 } 1298 } 1299 1300 *length_v_bord = l; 1301 *length_v_freq = l; 1302 1303 bmin = v_bord[0]; 1304 1305 } 1306 1307 /*------------------------------------------------------------------------ 1308 update modified follow-up data 1309 ------------------------------------------------------------------------*/ 1310 1311 *length_v_bordFollow = numBordFollow; 1312 *length_v_freqFollow = numBordFollow; 1313 1314 /* left relative borders correspond to follow-up */ 1315 *nL = numBordFollow - 1; 1316 1317 /*------------------------------------------------------------------------ 1318 take care of intervals ii through iv 1319 ------------------------------------------------------------------------*/ 1320 1321 /* now middle should be >= 0 */ 1322 middle = bmin - bordMaxFollow; 1323 1324 if (middle <= dmin) /* (ii) */ 1325 { 1326 b_new = *length_v_bord; 1327 1328 if (v_tuningSegm[0] != 0) 1329 { 1330 /* remove new "luxury" border and concatenate */ 1331 *length_v_bord = b_new - 1; 1332 FDKsbrEnc_AddVecLeft (v_bord, length_v_bord, v_bordFollow, 1333 *length_v_bordFollow); 1334 1335 *length_v_freq = b_new - 1; 1336 FDKsbrEnc_AddVecLeft (v_freq + 1, length_v_freq, v_freqFollow, 1337 *length_v_freqFollow); 1338 1339 } 1340 else if (*length_v_bordFollow > 1) 1341 { 1342 /* remove old border and concatenate */ 1343 FDKsbrEnc_AddVecLeft (v_bord, length_v_bord, v_bordFollow, 1344 *length_v_bordFollow - 1); 1345 FDKsbrEnc_AddVecLeft (v_freq, length_v_freq, v_freqFollow, 1346 *length_v_bordFollow - 1); 1347 1348 *nL = *nL - 1; 1349 } 1350 else 1351 { 1352 /* remove new border and concatenate */ 1353 for (i = 0; i < *length_v_bord - 1; i++) 1354 v_bord[i] = v_bord[i + 1]; 1355 1356 for (i = 0; i < *length_v_freq - 1; i++) 1357 v_freq[i] = v_freq[i + 1]; 1358 1359 *length_v_bord = b_new - 1; 1360 *length_v_freq = b_new - 1; 1361 1362 FDKsbrEnc_AddVecLeft (v_bord, length_v_bord, v_bordFollow, 1363 *length_v_bordFollow); 1364 FDKsbrEnc_AddVecLeft (v_freq, length_v_freq, v_freqFollow, 1365 *length_v_freqFollow); 1366 } 1367 } 1368 else if ((middle >= dmin) && (middle <= dmax)) /* (iii) */ 1369 { 1370 /* concatenate */ 1371 FDKsbrEnc_AddVecLeft (v_bord, length_v_bord, v_bordFollow, *length_v_bordFollow); 1372 FDKsbrEnc_AddVecLeft (v_freq, length_v_freq, v_freqFollow, *length_v_freqFollow); 1373 1374 } 1375 else /* (iv) */ 1376 { 1377 fillFramePre (dmax, v_bord, length_v_bord, v_freq, length_v_freq, bmin, 1378 middle); 1379 FDKsbrEnc_AddVecLeft (v_bord, length_v_bord, v_bordFollow, *length_v_bordFollow); 1380 FDKsbrEnc_AddVecLeft (v_freq, length_v_freq, v_freqFollow, *length_v_freqFollow); 1381 } 1382 } 1383 } 1384 1385 1386 1387 /******************************************************************************* 1388 Functionname: calcFrameClass 1389 ******************************************************************************* 1390 1391 Description: 1392 1393 Arguments: INT* frameClass, INT* frameClassOld, INT tranFlag, INT* spreadFlag) 1394 1395 Return: none 1396 1397 *******************************************************************************/ 1398 static void 1399 calcFrameClass (FRAME_CLASS *frameClass, FRAME_CLASS *frameClassOld, INT tranFlag, 1400 INT *spreadFlag) 1401 { 1402 1403 switch (*frameClassOld) { 1404 case FIXFIXonly: 1405 case FIXFIX: 1406 if (tranFlag) *frameClass = FIXVAR; 1407 else *frameClass = FIXFIX; 1408 break; 1409 case FIXVAR: 1410 if (tranFlag) { *frameClass = VARVAR; *spreadFlag = 0; } 1411 else { 1412 if (*spreadFlag) *frameClass = VARVAR; 1413 else *frameClass = VARFIX; 1414 } 1415 break; 1416 case VARFIX: 1417 if (tranFlag) *frameClass = FIXVAR; 1418 else *frameClass = FIXFIX; 1419 break; 1420 case VARVAR: 1421 if (tranFlag) { *frameClass = VARVAR; *spreadFlag = 0; } 1422 else { 1423 if (*spreadFlag) *frameClass = VARVAR; 1424 else *frameClass = VARFIX; 1425 } 1426 break; 1427 }; 1428 1429 *frameClassOld = *frameClass; 1430 } 1431 1432 1433 1434 /******************************************************************************* 1435 Functionname: specialCase 1436 ******************************************************************************* 1437 1438 Description: 1439 1440 Arguments: spreadFlag 1441 allowSpread 1442 v_bord 1443 length_v_bord 1444 v_freq 1445 length_v_freq 1446 parts 1447 d 1448 1449 Return: none 1450 1451 *******************************************************************************/ 1452 static void 1453 specialCase (INT *spreadFlag, INT allowSpread, INT *v_bord, 1454 INT *length_v_bord, INT *v_freq, INT *length_v_freq, INT *parts, 1455 INT d) 1456 { 1457 INT L; 1458 1459 L = *length_v_bord; 1460 1461 if (allowSpread) { /* add one "step 8" */ 1462 *spreadFlag = 1; 1463 FDKsbrEnc_AddRight (v_bord, length_v_bord, v_bord[L - 1] + 8); 1464 FDKsbrEnc_AddRight (v_freq, length_v_freq, 1); 1465 (*parts)++; 1466 } 1467 else { 1468 if (d == 1) { /* stretch one slot */ 1469 *length_v_bord = L - 1; 1470 *length_v_freq = L - 1; 1471 } 1472 else { 1473 if ((v_bord[L - 1] - v_bord[L - 2]) > 2) { /* compress one quant step */ 1474 v_bord[L - 1] = v_bord[L - 1] - 2; 1475 v_freq[*length_v_freq - 1] = 0; /* use low res for short segment */ 1476 } 1477 } 1478 } 1479 } 1480 1481 1482 1483 /******************************************************************************* 1484 Functionname: calcCmonBorder 1485 ******************************************************************************* 1486 1487 Description: 1488 1489 Arguments: i_cmon 1490 i_tran 1491 v_bord 1492 length_v_bord 1493 tran 1494 1495 Return: none 1496 1497 *******************************************************************************/ 1498 static void 1499 calcCmonBorder (INT *i_cmon, INT *i_tran, INT *v_bord, INT *length_v_bord, 1500 INT tran, INT bufferFrameStart, INT numberTimeSlots) 1501 { /* FH 00-06-26 */ 1502 INT i; 1503 1504 for (i = 0; i < *length_v_bord; i++) 1505 if (v_bord[i] >= bufferFrameStart + numberTimeSlots) { /* FH 00-06-26 */ 1506 *i_cmon = i; 1507 break; 1508 } 1509 1510 /* keep track of transient: */ 1511 for (i = 0; i < *length_v_bord; i++) 1512 if (v_bord[i] >= tran) { 1513 *i_tran = i; 1514 break; 1515 } 1516 else 1517 *i_tran = EMPTY; 1518 } 1519 1520 /******************************************************************************* 1521 Functionname: keepForFollowUp 1522 ******************************************************************************* 1523 1524 Description: 1525 1526 Arguments: v_bordFollow 1527 length_v_bordFollow 1528 v_freqFollow 1529 length_v_freqFollow 1530 i_tranFollow 1531 i_fillFollow 1532 v_bord 1533 length_v_bord 1534 v_freq 1535 i_cmon 1536 i_tran 1537 parts) 1538 1539 Return: none 1540 1541 *******************************************************************************/ 1542 static void 1543 keepForFollowUp (INT *v_bordFollow, INT *length_v_bordFollow, 1544 INT *v_freqFollow, INT *length_v_freqFollow, 1545 INT *i_tranFollow, INT *i_fillFollow, INT *v_bord, 1546 INT *length_v_bord, INT *v_freq, INT i_cmon, INT i_tran, 1547 INT parts, INT numberTimeSlots) 1548 { /* FH 00-06-26 */ 1549 INT L, i, j; 1550 1551 L = *length_v_bord; 1552 1553 (*length_v_bordFollow) = 0; 1554 (*length_v_freqFollow) = 0; 1555 1556 for (j = 0, i = i_cmon; i < L; i++, j++) { 1557 v_bordFollow[j] = v_bord[i] - numberTimeSlots; /* FH 00-06-26 */ 1558 v_freqFollow[j] = v_freq[i]; 1559 (*length_v_bordFollow)++; 1560 (*length_v_freqFollow)++; 1561 } 1562 if (i_tran != EMPTY) 1563 *i_tranFollow = i_tran - i_cmon; 1564 else 1565 *i_tranFollow = EMPTY; 1566 *i_fillFollow = L - (parts - 1) - i_cmon; 1567 1568 } 1569 1570 /******************************************************************************* 1571 Functionname: calcCtrlSignal 1572 ******************************************************************************* 1573 1574 Description: 1575 1576 Arguments: hSbrGrid 1577 frameClass 1578 v_bord 1579 length_v_bord 1580 v_freq 1581 length_v_freq 1582 i_cmon 1583 i_tran 1584 spreadFlag 1585 nL 1586 1587 Return: none 1588 1589 *******************************************************************************/ 1590 static void 1591 calcCtrlSignal (HANDLE_SBR_GRID hSbrGrid, 1592 FRAME_CLASS frameClass, INT *v_bord, INT length_v_bord, INT *v_freq, 1593 INT length_v_freq, INT i_cmon, INT i_tran, INT spreadFlag, 1594 INT nL) 1595 { 1596 1597 1598 INT i, r, a, n, p, b, aL, aR, ntot, nmax, nR; 1599 1600 INT *v_f = hSbrGrid->v_f; 1601 INT *v_fLR = hSbrGrid->v_fLR; 1602 INT *v_r = hSbrGrid->bs_rel_bord; 1603 INT *v_rL = hSbrGrid->bs_rel_bord_0; 1604 INT *v_rR = hSbrGrid->bs_rel_bord_1; 1605 1606 INT length_v_r = 0; 1607 INT length_v_rR = 0; 1608 INT length_v_rL = 0; 1609 1610 switch (frameClass) { 1611 case FIXVAR: 1612 /* absolute border: */ 1613 1614 a = v_bord[i_cmon]; 1615 1616 /* relative borders: */ 1617 length_v_r = 0; 1618 i = i_cmon; 1619 1620 while (i >= 1) { 1621 r = v_bord[i] - v_bord[i - 1]; 1622 FDKsbrEnc_AddRight (v_r, &length_v_r, r); 1623 i--; 1624 } 1625 1626 1627 /* number of relative borders: */ 1628 n = length_v_r; 1629 1630 1631 /* freq res: */ 1632 for (i = 0; i < i_cmon; i++) 1633 v_f[i] = v_freq[i_cmon - 1 - i]; 1634 v_f[i_cmon] = 1; 1635 1636 /* pointer: */ 1637 p = (i_cmon >= i_tran && i_tran != EMPTY) ? (i_cmon - i_tran + 1) : (0) ; 1638 1639 hSbrGrid->frameClass = frameClass; 1640 hSbrGrid->bs_abs_bord = a; 1641 hSbrGrid->n = n; 1642 hSbrGrid->p = p; 1643 1644 break; 1645 case VARFIX: 1646 /* absolute border: */ 1647 a = v_bord[0]; 1648 1649 /* relative borders: */ 1650 length_v_r = 0; 1651 1652 for (i = 1; i < length_v_bord; i++) { 1653 r = v_bord[i] - v_bord[i - 1]; 1654 FDKsbrEnc_AddRight (v_r, &length_v_r, r); 1655 } 1656 1657 /* number of relative borders: */ 1658 n = length_v_r; 1659 1660 /* freq res: */ 1661 FDKmemcpy (v_f, v_freq, length_v_freq * sizeof (INT)); 1662 1663 1664 /* pointer: */ 1665 p = (i_tran >= 0 && i_tran != EMPTY) ? (i_tran + 1) : (0) ; 1666 1667 hSbrGrid->frameClass = frameClass; 1668 hSbrGrid->bs_abs_bord = a; 1669 hSbrGrid->n = n; 1670 hSbrGrid->p = p; 1671 1672 break; 1673 case VARVAR: 1674 if (spreadFlag) { 1675 /* absolute borders: */ 1676 b = length_v_bord; 1677 1678 aL = v_bord[0]; 1679 aR = v_bord[b - 1]; 1680 1681 1682 /* number of relative borders: */ 1683 ntot = b - 2; 1684 1685 nmax = 2; /* n: {0,1,2} */ 1686 if (ntot > nmax) { 1687 nL = nmax; 1688 nR = ntot - nmax; 1689 } 1690 else { 1691 nL = ntot; 1692 nR = 0; 1693 } 1694 1695 /* relative borders: */ 1696 length_v_rL = 0; 1697 for (i = 1; i <= nL; i++) { 1698 r = v_bord[i] - v_bord[i - 1]; 1699 FDKsbrEnc_AddRight (v_rL, &length_v_rL, r); 1700 } 1701 1702 length_v_rR = 0; 1703 i = b - 1; 1704 while (i >= b - nR) { 1705 r = v_bord[i] - v_bord[i - 1]; 1706 FDKsbrEnc_AddRight (v_rR, &length_v_rR, r); 1707 i--; 1708 } 1709 1710 /* pointer (only one due to constraint in frame info): */ 1711 p = (i_tran > 0 && i_tran != EMPTY) ? (b - i_tran) : (0) ; 1712 1713 /* freq res: */ 1714 1715 for (i = 0; i < b - 1; i++) 1716 v_fLR[i] = v_freq[i]; 1717 } 1718 else { 1719 1720 length_v_bord = i_cmon + 1; 1721 length_v_freq = i_cmon + 1; 1722 1723 1724 /* absolute borders: */ 1725 b = length_v_bord; 1726 1727 aL = v_bord[0]; 1728 aR = v_bord[b - 1]; 1729 1730 /* number of relative borders: */ 1731 ntot = b - 2; 1732 nR = ntot - nL; 1733 1734 /* relative borders: */ 1735 length_v_rL = 0; 1736 for (i = 1; i <= nL; i++) { 1737 r = v_bord[i] - v_bord[i - 1]; 1738 FDKsbrEnc_AddRight (v_rL, &length_v_rL, r); 1739 } 1740 1741 length_v_rR = 0; 1742 i = b - 1; 1743 while (i >= b - nR) { 1744 r = v_bord[i] - v_bord[i - 1]; 1745 FDKsbrEnc_AddRight (v_rR, &length_v_rR, r); 1746 i--; 1747 } 1748 1749 /* pointer (only one due to constraint in frame info): */ 1750 p = (i_cmon >= i_tran && i_tran != EMPTY) ? (i_cmon - i_tran + 1) : (0) ; 1751 1752 /* freq res: */ 1753 for (i = 0; i < b - 1; i++) 1754 v_fLR[i] = v_freq[i]; 1755 } 1756 1757 hSbrGrid->frameClass = frameClass; 1758 hSbrGrid->bs_abs_bord_0 = aL; 1759 hSbrGrid->bs_abs_bord_1 = aR; 1760 hSbrGrid->bs_num_rel_0 = nL; 1761 hSbrGrid->bs_num_rel_1 = nR; 1762 hSbrGrid->p = p; 1763 1764 break; 1765 1766 default: 1767 /* do nothing */ 1768 break; 1769 } 1770 } 1771 1772 /******************************************************************************* 1773 Functionname: createDefFrameInfo 1774 ******************************************************************************* 1775 1776 Description: Copies the default (static) frameInfo structs to the frameInfo 1777 passed by reference; only used for FIXFIX frames 1778 1779 Arguments: hFrameInfo - HANLDE_SBR_FRAME_INFO 1780 nEnv - INT 1781 nTimeSlots - INT 1782 1783 Return: none; hSbrFrameInfo contains a copy of the default frameInfo 1784 1785 Written: Andreas Schneider 1786 Revised: 1787 *******************************************************************************/ 1788 static void 1789 createDefFrameInfo(HANDLE_SBR_FRAME_INFO hSbrFrameInfo, INT nEnv, INT nTimeSlots) 1790 { 1791 switch (nEnv) { 1792 case 1: 1793 switch (nTimeSlots) { 1794 case NUMBER_TIME_SLOTS_1920: 1795 FDKmemcpy (hSbrFrameInfo, &frameInfo1_1920, sizeof (SBR_FRAME_INFO)); 1796 break; 1797 case NUMBER_TIME_SLOTS_2048: 1798 FDKmemcpy (hSbrFrameInfo, &frameInfo1_2048, sizeof (SBR_FRAME_INFO)); 1799 break; 1800 case NUMBER_TIME_SLOTS_1152: 1801 FDKmemcpy (hSbrFrameInfo, &frameInfo1_1152, sizeof (SBR_FRAME_INFO)); 1802 break; 1803 case NUMBER_TIME_SLOTS_2304: 1804 FDKmemcpy (hSbrFrameInfo, &frameInfo1_2304, sizeof (SBR_FRAME_INFO)); 1805 break; 1806 case NUMBER_TIME_SLOTS_512LD: 1807 FDKmemcpy (hSbrFrameInfo, &frameInfo1_512LD, sizeof (SBR_FRAME_INFO)); 1808 break; 1809 default: 1810 FDK_ASSERT(0); 1811 } 1812 break; 1813 case 2: 1814 switch (nTimeSlots) { 1815 case NUMBER_TIME_SLOTS_1920: 1816 FDKmemcpy (hSbrFrameInfo, &frameInfo2_1920, sizeof (SBR_FRAME_INFO)); 1817 break; 1818 case NUMBER_TIME_SLOTS_2048: 1819 FDKmemcpy (hSbrFrameInfo, &frameInfo2_2048, sizeof (SBR_FRAME_INFO)); 1820 break; 1821 case NUMBER_TIME_SLOTS_1152: 1822 FDKmemcpy (hSbrFrameInfo, &frameInfo2_1152, sizeof (SBR_FRAME_INFO)); 1823 break; 1824 case NUMBER_TIME_SLOTS_2304: 1825 FDKmemcpy (hSbrFrameInfo, &frameInfo2_2304, sizeof (SBR_FRAME_INFO)); 1826 break; 1827 case NUMBER_TIME_SLOTS_512LD: 1828 FDKmemcpy (hSbrFrameInfo, &frameInfo2_512LD, sizeof (SBR_FRAME_INFO)); 1829 break; 1830 default: 1831 FDK_ASSERT(0); 1832 } 1833 break; 1834 case 4: 1835 switch (nTimeSlots) { 1836 case NUMBER_TIME_SLOTS_1920: 1837 FDKmemcpy (hSbrFrameInfo, &frameInfo4_1920, sizeof (SBR_FRAME_INFO)); 1838 break; 1839 case NUMBER_TIME_SLOTS_2048: 1840 FDKmemcpy (hSbrFrameInfo, &frameInfo4_2048, sizeof (SBR_FRAME_INFO)); 1841 break; 1842 case NUMBER_TIME_SLOTS_1152: 1843 FDKmemcpy (hSbrFrameInfo, &frameInfo4_1152, sizeof (SBR_FRAME_INFO)); 1844 break; 1845 case NUMBER_TIME_SLOTS_2304: 1846 FDKmemcpy (hSbrFrameInfo, &frameInfo4_2304, sizeof (SBR_FRAME_INFO)); 1847 break; 1848 case NUMBER_TIME_SLOTS_512LD: 1849 FDKmemcpy (hSbrFrameInfo, &frameInfo4_512LD, sizeof (SBR_FRAME_INFO)); 1850 break; 1851 default: 1852 FDK_ASSERT(0); 1853 } 1854 break; 1855 default: 1856 FDK_ASSERT(0); 1857 } 1858 } 1859 1860 1861 /******************************************************************************* 1862 Functionname: ctrlSignal2FrameInfo 1863 ******************************************************************************* 1864 1865 Description: Calculates frame_info struct from control signal. 1866 1867 Arguments: hSbrGrid - source 1868 hSbrFrameInfo - destination 1869 1870 Return: void; hSbrFrameInfo contains the updated FRAME_INFO struct 1871 1872 *******************************************************************************/ 1873 static void 1874 ctrlSignal2FrameInfo (HANDLE_SBR_GRID hSbrGrid, 1875 HANDLE_SBR_FRAME_INFO hSbrFrameInfo, 1876 INT freq_res_fixfix) 1877 { 1878 INT nEnv = 0, border = 0, i, k, p /*?*/; 1879 INT *v_r = hSbrGrid->bs_rel_bord; 1880 INT *v_f = hSbrGrid->v_f; 1881 1882 FRAME_CLASS frameClass = hSbrGrid->frameClass; 1883 INT bufferFrameStart = hSbrGrid->bufferFrameStart; 1884 INT numberTimeSlots = hSbrGrid->numberTimeSlots; 1885 1886 switch (frameClass) { 1887 case FIXFIX: 1888 createDefFrameInfo(hSbrFrameInfo, hSbrGrid->bs_num_env, numberTimeSlots); 1889 1890 /* At this point all frequency resolutions are set to FREQ_RES_HIGH, so 1891 * only if freq_res_fixfix is set to FREQ_RES_LOW, they all have to be 1892 * changed. 1893 * snd */ 1894 if (freq_res_fixfix == FREQ_RES_LOW) { 1895 for (i = 0; i < hSbrFrameInfo->nEnvelopes; i++) { 1896 hSbrFrameInfo->freqRes[i] = FREQ_RES_LOW; 1897 } 1898 } 1899 /* ELD: store current frequency resolution */ 1900 hSbrGrid->v_f[0] = hSbrFrameInfo->freqRes[0]; 1901 break; 1902 1903 case FIXVAR: 1904 case VARFIX: 1905 nEnv = hSbrGrid->n + 1; /* read n [SBR_NUM_BITS bits] */ /*? snd*/ 1906 FDK_ASSERT(nEnv <= MAX_ENVELOPES_FIXVAR_VARFIX); 1907 1908 hSbrFrameInfo->nEnvelopes = nEnv; 1909 1910 border = hSbrGrid->bs_abs_bord; /* read the absolute border */ 1911 1912 if (nEnv == 1) 1913 hSbrFrameInfo->nNoiseEnvelopes = 1; 1914 else 1915 hSbrFrameInfo->nNoiseEnvelopes = 2; 1916 1917 break; 1918 1919 default: 1920 /* do nothing */ 1921 break; 1922 } 1923 1924 switch (frameClass) { 1925 case FIXVAR: 1926 hSbrFrameInfo->borders[0] = bufferFrameStart; /* start-position of 1st envelope */ 1927 1928 hSbrFrameInfo->borders[nEnv] = border; 1929 1930 for (k = 0, i = nEnv - 1; k < nEnv - 1; k++, i--) { 1931 border -= v_r[k]; 1932 hSbrFrameInfo->borders[i] = border; 1933 } 1934 1935 /* make either envelope nr. nEnv + 1 - p short; or don't shorten if p == 0 */ 1936 p = hSbrGrid->p; 1937 if (p == 0) { 1938 hSbrFrameInfo->shortEnv = 0; 1939 } else { 1940 hSbrFrameInfo->shortEnv = nEnv + 1 - p; 1941 } 1942 1943 for (k = 0, i = nEnv - 1; k < nEnv; k++, i--) { 1944 hSbrFrameInfo->freqRes[i] = (FREQ_RES)v_f[k]; 1945 } 1946 1947 /* if either there is no short envelope or the last envelope is short... */ 1948 if (p == 0 || p == 1) { 1949 hSbrFrameInfo->bordersNoise[1] = hSbrFrameInfo->borders[nEnv - 1]; 1950 } else { 1951 hSbrFrameInfo->bordersNoise[1] = hSbrFrameInfo->borders[hSbrFrameInfo->shortEnv]; 1952 } 1953 1954 break; 1955 1956 case VARFIX: 1957 /* in this case 'border' indicates the start of the 1st envelope */ 1958 hSbrFrameInfo->borders[0] = border; 1959 1960 for (k = 0; k < nEnv - 1; k++) { 1961 border += v_r[k]; 1962 hSbrFrameInfo->borders[k + 1] = border; 1963 } 1964 1965 hSbrFrameInfo->borders[nEnv] = bufferFrameStart + numberTimeSlots; 1966 1967 p = hSbrGrid->p; 1968 if (p == 0 || p == 1) { 1969 hSbrFrameInfo->shortEnv = 0; 1970 } else { 1971 hSbrFrameInfo->shortEnv = p - 1; 1972 } 1973 1974 for (k = 0; k < nEnv; k++) { 1975 hSbrFrameInfo->freqRes[k] = (FREQ_RES)v_f[k]; 1976 } 1977 1978 switch (p) { 1979 case 0: 1980 hSbrFrameInfo->bordersNoise[1] = hSbrFrameInfo->borders[1]; 1981 break; 1982 case 1: 1983 hSbrFrameInfo->bordersNoise[1] = hSbrFrameInfo->borders[nEnv - 1]; 1984 break; 1985 default: 1986 hSbrFrameInfo->bordersNoise[1] = hSbrFrameInfo->borders[hSbrFrameInfo->shortEnv]; 1987 break; 1988 } 1989 break; 1990 1991 case VARVAR: 1992 nEnv = hSbrGrid->bs_num_rel_0 + hSbrGrid->bs_num_rel_1 + 1; 1993 FDK_ASSERT(nEnv <= MAX_ENVELOPES_VARVAR); /* just to be sure */ 1994 hSbrFrameInfo->nEnvelopes = nEnv; 1995 1996 hSbrFrameInfo->borders[0] = border = hSbrGrid->bs_abs_bord_0; 1997 1998 for (k = 0, i = 1; k < hSbrGrid->bs_num_rel_0; k++, i++) { 1999 border += hSbrGrid->bs_rel_bord_0[k]; 2000 hSbrFrameInfo->borders[i] = border; 2001 } 2002 2003 border = hSbrGrid->bs_abs_bord_1; 2004 hSbrFrameInfo->borders[nEnv] = border; 2005 2006 for (k = 0, i = nEnv - 1; k < hSbrGrid->bs_num_rel_1; k++, i--) { 2007 border -= hSbrGrid->bs_rel_bord_1[k]; 2008 hSbrFrameInfo->borders[i] = border; 2009 } 2010 2011 p = hSbrGrid->p; 2012 if (p == 0) { 2013 hSbrFrameInfo->shortEnv = 0; 2014 } else { 2015 hSbrFrameInfo->shortEnv = nEnv + 1 - p; 2016 } 2017 2018 for (k = 0; k < nEnv; k++) { 2019 hSbrFrameInfo->freqRes[k] = (FREQ_RES)hSbrGrid->v_fLR[k]; 2020 } 2021 2022 if (nEnv == 1) { 2023 hSbrFrameInfo->nNoiseEnvelopes = 1; 2024 hSbrFrameInfo->bordersNoise[0] = hSbrGrid->bs_abs_bord_0; 2025 hSbrFrameInfo->bordersNoise[1] = hSbrGrid->bs_abs_bord_1; 2026 } else { 2027 hSbrFrameInfo->nNoiseEnvelopes = 2; 2028 hSbrFrameInfo->bordersNoise[0] = hSbrGrid->bs_abs_bord_0; 2029 2030 if (p == 0 || p == 1) { 2031 hSbrFrameInfo->bordersNoise[1] = hSbrFrameInfo->borders[nEnv - 1]; 2032 } else { 2033 hSbrFrameInfo->bordersNoise[1] = hSbrFrameInfo->borders[hSbrFrameInfo->shortEnv]; 2034 } 2035 hSbrFrameInfo->bordersNoise[2] = hSbrGrid->bs_abs_bord_1; 2036 } 2037 break; 2038 2039 default: 2040 /* do nothing */ 2041 break; 2042 } 2043 2044 if (frameClass == VARFIX || frameClass == FIXVAR) { 2045 hSbrFrameInfo->bordersNoise[0] = hSbrFrameInfo->borders[0]; 2046 if (nEnv == 1) { 2047 hSbrFrameInfo->bordersNoise[1] = hSbrFrameInfo->borders[nEnv]; 2048 } else { 2049 hSbrFrameInfo->bordersNoise[2] = hSbrFrameInfo->borders[nEnv]; 2050 } 2051 } 2052 } 2053 2054