1 /* ------------------------------------------------------------------ 2 * Copyright (C) 1998-2009 PacketVideo 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either 13 * express or implied. 14 * See the License for the specific language governing permissions 15 * and limitations under the License. 16 * ------------------------------------------------------------------- 17 */ 18 /**************************************************************************************** 19 Portions of this file are derived from the following 3GPP standard: 20 21 3GPP TS 26.073 22 ANSI-C code for the Adaptive Multi-Rate (AMR) speech codec 23 Available from http://www.3gpp.org 24 25 (C) 2004, 3GPP Organizational Partners (ARIB, ATIS, CCSA, ETSI, TTA, TTC) 26 Permission to distribute, modify and use this file under the standard license 27 terms listed above has been obtained from the copyright holder. 28 ****************************************************************************************/ 29 /* 30 ------------------------------------------------------------------------------ 31 32 33 34 Pathname: ./audio/gsm-amr/c/src/c2_9pf.c 35 Funtions: code_2i40_9bits 36 search_2i40 37 Test_search_2i40 38 build_code 39 Test_build_code 40 41 Date: 05/26/2000 42 43 ------------------------------------------------------------------------------ 44 REVISION HISTORY 45 46 Description: Changed template used to PV coding template. First attempt at 47 optimizing C code. 48 49 Description: Updated file per comments gathered from Phase 2/3 review. 50 51 Description: Added setting of Overflow flag in inlined code. 52 53 Description: Synchronized file with UMTS version 3.2.0. Updated coding 54 template. 55 56 Description: Replaced basic_op.h with the header files of the math functions 57 used by the file. 58 59 Description: Made the following changes per comments from Phase 2/3 review: 60 1. Defined one local variable per line. 61 62 Description: Passed in pOverflow flag for EPOC compatibility. 63 64 Description: Optimized search_2i40() to reduce clock cycle usage. 65 66 Description: Removed unnecessary include files and #defines. 67 68 Description: Changed function name to pv_round to avoid conflict with 69 round function in C standard library. 70 71 Description: Replaced "int" and/or "char" with OSCL defined types. 72 73 Description: Added #ifdef __cplusplus around extern'ed table. 74 75 Description: 76 77 ------------------------------------------------------------------------------ 78 MODULE DESCRIPTION 79 80 This file contains the functions that search a 9 bit algebraic codebook 81 containing 2 pulses in a frame of 40 samples. 82 83 ------------------------------------------------------------------------------ 84 */ 85 86 /*---------------------------------------------------------------------------- 87 ; INCLUDES 88 ----------------------------------------------------------------------------*/ 89 #include "c2_9pf.h" 90 #include "typedef.h" 91 #include "basic_op.h" 92 #include "inv_sqrt.h" 93 #include "cnst.h" 94 #include "cor_h.h" 95 #include "cor_h_x.h" 96 #include "set_sign.h" 97 98 /*--------------------------------------------------------------------------*/ 99 #ifdef __cplusplus 100 extern "C" 101 { 102 #endif 103 104 /*---------------------------------------------------------------------------- 105 ; MACROS 106 ; Define module specific macros here 107 ----------------------------------------------------------------------------*/ 108 109 /*---------------------------------------------------------------------------- 110 ; DEFINES 111 ; Include all pre-processor statements here. Include conditional 112 ; compile variables also. 113 ----------------------------------------------------------------------------*/ 114 #define NB_PULSE 2 115 116 /*---------------------------------------------------------------------------- 117 ; LOCAL FUNCTION DEFINITIONS 118 ; Function Prototype declaration 119 ----------------------------------------------------------------------------*/ 120 static void search_2i40( 121 Word16 subNr, /* i : subframe number */ 122 Word16 dn[], /* i : correlation between target and h[] */ 123 Word16 rr[][L_CODE],/* i : matrix of autocorrelation */ 124 Word16 codvec[], /* o : algebraic codebook vector */ 125 Flag * pOverflow /* o : Flag set when overflow occurs */ 126 ); 127 128 static Word16 build_code( 129 Word16 subNr, /* i : subframe number */ 130 Word16 codvec[], /* i : algebraic codebook vector */ 131 Word16 dn_sign[], /* i : sign of dn[] */ 132 Word16 cod[], /* o : algebraic (fixed) codebook excitation */ 133 Word16 h[], /* i : impulse response of weighted synthesis filter */ 134 Word16 y[], /* o : filtered fixed codebook excitation */ 135 Word16 sign[], /* o : sign of 2 pulses */ 136 Flag * pOverflow /* o : Flag set when overflow occurs */ 137 ); 138 139 /*---------------------------------------------------------------------------- 140 ; LOCAL VARIABLE DEFINITIONS 141 ; Variable declaration - defined here and used outside this module 142 ----------------------------------------------------------------------------*/ 143 144 const Word16 trackTable[4*5] = 145 { 146 0, 1, 0, 1, -1, /* subframe 1; track to code; 147 * -1 do not code this position 148 */ 149 0, -1, 1, 0, 1, /* subframe 2 */ 150 0, 1, 0, -1, 1, /* subframe 3 */ 151 0, 1, -1, 0, 1 152 };/* subframe 4 */ 153 154 155 /*---------------------------------------------------------------------------- 156 ; EXTERNAL GLOBAL STORE/BUFFER/POINTER REFERENCES 157 ; Declare variables used in this module but defined elsewhere 158 ----------------------------------------------------------------------------*/ 159 extern const Word16 startPos[]; 160 161 /* 162 ------------------------------------------------------------------------------ 163 FUNCTION NAME: code_2i40_9bits 164 ------------------------------------------------------------------------------ 165 INPUT AND OUTPUT DEFINITIONS 166 167 Inputs: 168 subNr = subframe number (Word16) 169 x = target buffer (Word16) 170 h = buffer containing the impulse response of the 171 weighted synthesis filter; h[-L_subfr .. -1] must be 172 set to zero (Word16) 173 T0 = pitch lag (Word16) 174 pitch_sharp = last quantized pitch gain (Word16) 175 code = buffer containing the innovative codebook (Word16) 176 y = buffer containing the filtered fixed codebook excitation (Word16) 177 sign = pointer to the signs of 2 pulses (Word16) 178 179 Outputs: 180 code buffer contains the new innovation vector gains 181 182 Returns: 183 index = code index (Word16) 184 185 Global Variables Used: 186 Overflow = overflow flag (Flag) 187 188 Local Variables Needed: 189 None 190 191 ------------------------------------------------------------------------------ 192 FUNCTION DESCRIPTION 193 194 This function searches a 9 bit algebraic codebook containing 2 pulses in a 195 frame of 40 samples. 196 197 The code length is 40, containing 2 nonzero pulses: i0...i1. All pulses can 198 have two possible amplitudes: +1 or -1. Pulse i0 can have 8 possible positions, 199 pulse i1 can have 8 positions. Also coded is which track pair should be used, 200 i.e. first or second pair. Where each pair contains 2 tracks. 201 202 First subframe: 203 first i0 : 0, 5, 10, 15, 20, 25, 30, 35. 204 i1 : 2, 7, 12, 17, 22, 27, 32, 37. 205 second i0 : 1, 6, 11, 16, 21, 26, 31, 36. 206 i1 : 3, 8, 13, 18, 23, 28, 33, 38. 207 208 Second subframe: 209 first i0 : 0, 5, 10, 15, 20, 25, 30, 35. 210 i1 : 3, 8, 13, 18, 23, 28, 33, 38. 211 second i0 : 2, 7, 12, 17, 22, 27, 32, 37. 212 i1 : 4, 9, 14, 19, 24, 29, 34, 39. 213 214 Third subframe: 215 first i0 : 0, 5, 10, 15, 20, 25, 30, 35. 216 i1 : 2, 7, 12, 17, 22, 27, 32, 37. 217 second i0 : 1, 6, 11, 16, 21, 26, 31, 36. 218 i1 : 4, 9, 14, 19, 24, 29, 34, 39. 219 220 Fourth subframe: 221 first i0 : 0, 5, 10, 15, 20, 25, 30, 35. 222 i1 : 3, 8, 13, 18, 23, 28, 33, 38. 223 second i0 : 1, 6, 11, 16, 21, 26, 31, 36. 224 i1 : 4, 9, 14, 19, 24, 29, 34, 39. 225 226 ------------------------------------------------------------------------------ 227 REQUIREMENTS 228 229 None 230 231 ------------------------------------------------------------------------------ 232 REFERENCES 233 234 [1] c2_9pf.c, UMTS GSM AMR speech codec, R99 - Version 3.2.0, March 2, 2001 235 236 ------------------------------------------------------------------------------ 237 PSEUDO-CODE 238 239 Word16 code_2i40_9bits( 240 Word16 subNr, // i : subframe number 241 Word16 x[], // i : target vector 242 Word16 h[], // i : impulse response of weighted synthesis filter 243 // h[-L_subfr..-1] must be set to zero. 244 Word16 T0, // i : Pitch lag 245 Word16 pitch_sharp, // i : Last quantized pitch gain 246 Word16 code[], // o : Innovative codebook 247 Word16 y[], // o : filtered fixed codebook excitation 248 Word16 * sign // o : Signs of 2 pulses 249 ) 250 { 251 Word16 codvec[NB_PULSE]; 252 Word16 dn[L_CODE], dn2[L_CODE], dn_sign[L_CODE]; 253 Word16 rr[L_CODE][L_CODE]; 254 Word16 i, index, sharp; 255 256 sharp = shl(pitch_sharp, 1); 257 if (sub(T0, L_CODE) < 0) 258 for (i = T0; i < L_CODE; i++) { 259 h[i] = add(h[i], mult(h[i - T0], sharp)); 260 } 261 cor_h_x(h, x, dn, 1); 262 set_sign(dn, dn_sign, dn2, 8); // dn2[] not used in this codebook search 263 cor_h(h, dn_sign, rr); 264 search_2i40(subNr, dn, rr, codvec); 265 index = build_code(subNr, codvec, dn_sign, code, h, y, sign); 266 267 *-----------------------------------------------------------------* 268 * Compute innovation vector gain. * 269 * Include fixed-gain pitch contribution into code[]. * 270 *-----------------------------------------------------------------* 271 272 if (sub(T0, L_CODE) < 0) 273 for (i = T0; i < L_CODE; i++) { 274 code[i] = add(code[i], mult(code[i - T0], sharp)); 275 } 276 return index; 277 } 278 279 ------------------------------------------------------------------------------ 280 RESOURCES USED [optional] 281 282 When the code is written for a specific target processor the 283 the resources used should be documented below. 284 285 HEAP MEMORY USED: x bytes 286 287 STACK MEMORY USED: x bytes 288 289 CLOCK CYCLES: (cycle count equation for this function) + (variable 290 used to represent cycle count for each subroutine 291 called) 292 where: (cycle count variable) = cycle count for [subroutine 293 name] 294 295 ------------------------------------------------------------------------------ 296 CAUTION [optional] 297 [State any special notes, constraints or cautions for users of this function] 298 299 ------------------------------------------------------------------------------ 300 */ 301 302 Word16 code_2i40_9bits( 303 Word16 subNr, /* i : subframe number */ 304 Word16 x[], /* i : target vector */ 305 Word16 h[], /* i : impulse response of weighted synthesis */ 306 /* filter h[-L_subfr..-1] must be set to 0. */ 307 Word16 T0, /* i : Pitch lag */ 308 Word16 pitch_sharp, /* i : Last quantized pitch gain */ 309 Word16 code[], /* o : Innovative codebook */ 310 Word16 y[], /* o : filtered fixed codebook excitation */ 311 Word16 * sign, /* o : Signs of 2 pulses */ 312 Flag * pOverflow /* o : Flag set when overflow occurs */ 313 ) 314 { 315 Word16 codvec[NB_PULSE]; 316 Word16 dn[L_CODE]; 317 Word16 dn2[L_CODE]; 318 Word16 dn_sign[L_CODE]; 319 Word16 rr[L_CODE][L_CODE]; 320 321 Word16 i; 322 323 Word16 index; 324 Word16 sharp; 325 Word16 temp; 326 Word32 L_temp; 327 328 L_temp = ((Word32) pitch_sharp) << 1; 329 330 /* Check for overflow condition */ 331 if (L_temp != (Word32)((Word16) L_temp)) 332 { 333 *(pOverflow) = 1; 334 sharp = (pitch_sharp > 0) ? MAX_16 : MIN_16; 335 } 336 else 337 { 338 sharp = (Word16) L_temp; 339 } 340 341 if (T0 < L_CODE) 342 { 343 for (i = T0; i < L_CODE; i++) 344 { 345 temp = 346 mult( 347 *(h + i - T0), 348 sharp, 349 pOverflow); 350 351 *(h + i) = 352 add( 353 *(h + i), 354 temp, 355 pOverflow); 356 } 357 } 358 359 cor_h_x( 360 h, 361 x, 362 dn, 363 1, 364 pOverflow); 365 366 /* dn2[] not used in this codebook search */ 367 368 set_sign( 369 dn, 370 dn_sign, 371 dn2, 372 8); 373 374 cor_h( 375 h, 376 dn_sign, 377 rr, 378 pOverflow); 379 380 search_2i40( 381 subNr, 382 dn, 383 rr, 384 codvec, 385 pOverflow); 386 387 index = 388 build_code( 389 subNr, 390 codvec, 391 dn_sign, 392 code, 393 h, 394 y, 395 sign, 396 pOverflow); 397 398 /*-----------------------------------------------------------------* 399 * Compute innovation vector gain. * 400 * Include fixed-gain pitch contribution into code[]. * 401 *-----------------------------------------------------------------*/ 402 403 if (T0 < L_CODE) 404 { 405 for (i = T0; i < L_CODE; i++) 406 { 407 temp = 408 mult( 409 *(code + i - T0), 410 sharp, 411 pOverflow); 412 413 *(code + i) = 414 add( 415 *(code + i), 416 temp, 417 pOverflow); 418 } 419 } 420 421 return(index); 422 } 423 424 /****************************************************************************/ 425 426 427 /* 428 ------------------------------------------------------------------------------ 429 FUNCTION NAME: search_2i40 430 ------------------------------------------------------------------------------ 431 INPUT AND OUTPUT DEFINITIONS 432 433 Inputs: 434 subNr = subframe number (Word16) 435 dn = vector containing the correlation between target and the impulse 436 response of the weighted synthesis filter (Word16) 437 rr = autocorrelation matrix (Word16) 438 codvec = algebraic codebook vector (Word16) 439 440 Outputs: 441 codvec contains the newly calculated codevectors 442 443 Returns: 444 None 445 446 Global Variables Used: 447 None 448 449 Local Variables Needed: 450 startPos = table containing the start positions used by fixed codebook 451 routines (const Word16) 452 453 ------------------------------------------------------------------------------ 454 FUNCTION DESCRIPTION 455 456 This function searches the best codevector and determines the positions of 457 the 2 pulses in the 40-sample frame. 458 459 ------------------------------------------------------------------------------ 460 REQUIREMENTS 461 462 None 463 464 ------------------------------------------------------------------------------ 465 REFERENCES 466 467 [1] c2_9pf.c, UMTS GSM AMR speech codec, R99 - Version 3.2.0, March 2, 2001 468 469 ------------------------------------------------------------------------------ 470 PSEUDO-CODE 471 472 static void search_2i40( 473 Word16 subNr, // i : subframe number 474 Word16 dn[], // i : correlation between target and h[] 475 Word16 rr[][L_CODE], // i : matrix of autocorrelation 476 Word16 codvec[] // o : algebraic codebook vector 477 ) 478 { 479 Word16 i0, i1; 480 Word16 ix = 0; // initialization only needed to keep gcc silent 481 Word16 track1, ipos[NB_PULSE]; 482 Word16 psk, ps0, ps1, sq, sq1; 483 Word16 alpk, alp, alp_16; 484 Word32 s, alp0, alp1; 485 Word16 i; 486 487 psk = -1; 488 alpk = 1; 489 for (i = 0; i < NB_PULSE; i++) 490 { 491 codvec[i] = i; 492 } 493 494 for (track1 = 0; track1 < 2; track1++) { 495 // fix starting position 496 497 ipos[0] = startPos[subNr*2+8*track1]; 498 ipos[1] = startPos[subNr*2+1+8*track1]; 499 500 501 *----------------------------------------------------------------* 502 * i0 loop: try 8 positions. * 503 *----------------------------------------------------------------* 504 505 for (i0 = ipos[0]; i0 < L_CODE; i0 += STEP) { 506 507 ps0 = dn[i0]; 508 alp0 = L_mult(rr[i0][i0], _1_4); 509 510 *----------------------------------------------------------------* 511 * i1 loop: 8 positions. * 512 *----------------------------------------------------------------* 513 514 sq = -1; 515 alp = 1; 516 ix = ipos[1]; 517 518 *-------------------------------------------------------------------* 519 * These index have low complexity address computation because * 520 * they are, in fact, pointers with fixed increment. For example, * 521 * "rr[i0][i2]" is a pointer initialized to "&rr[i0][ipos[2]]" * 522 * and incremented by "STEP". * 523 *-------------------------------------------------------------------* 524 525 for (i1 = ipos[1]; i1 < L_CODE; i1 += STEP) { 526 ps1 = add(ps0, dn[i1]); // idx increment = STEP 527 528 // alp1 = alp0 + rr[i0][i1] + 1/2*rr[i1][i1]; 529 530 alp1 = L_mac(alp0, rr[i1][i1], _1_4); // idx incr = STEP 531 alp1 = L_mac(alp1, rr[i0][i1], _1_2); // idx incr = STEP 532 533 sq1 = mult(ps1, ps1); 534 535 alp_16 = pv_round(alp1); 536 537 s = L_msu(L_mult(alp, sq1), sq, alp_16); 538 539 if (s > 0) { 540 sq = sq1; 541 alp = alp_16; 542 ix = i1; 543 } 544 } 545 546 *----------------------------------------------------------------* 547 * memorise codevector if this one is better than the last one. * 548 *----------------------------------------------------------------* 549 550 s = L_msu(L_mult(alpk, sq), psk, alp); 551 552 if (s > 0) { 553 psk = sq; 554 alpk = alp; 555 codvec[0] = i0; 556 codvec[1] = ix; 557 } 558 } 559 } 560 561 return; 562 } 563 564 ------------------------------------------------------------------------------ 565 RESOURCES USED [optional] 566 567 When the code is written for a specific target processor the 568 the resources used should be documented below. 569 570 HEAP MEMORY USED: x bytes 571 572 STACK MEMORY USED: x bytes 573 574 CLOCK CYCLES: (cycle count equation for this function) + (variable 575 used to represent cycle count for each subroutine 576 called) 577 where: (cycle count variable) = cycle count for [subroutine 578 name] 579 580 ------------------------------------------------------------------------------ 581 CAUTION [optional] 582 [State any special notes, constraints or cautions for users of this function] 583 584 ------------------------------------------------------------------------------ 585 */ 586 587 static void search_2i40( 588 Word16 subNr, /* i : subframe number */ 589 Word16 dn[], /* i : correlation between target and h[] */ 590 Word16 rr[][L_CODE], /* i : matrix of autocorrelation */ 591 Word16 codvec[], /* o : algebraic codebook vector */ 592 Flag * pOverflow /* o : Flag set when overflow occurs */ 593 ) 594 { 595 Word16 i0; 596 Word16 i1; 597 Word16 ix = 0; /* initialization only needed to keep gcc silent */ 598 Word16 track1; 599 Word16 ipos[NB_PULSE]; 600 Word16 psk; 601 Word16 ps0; 602 Word16 ps1; 603 Word16 sq; 604 Word16 sq1; 605 Word16 alpk; 606 Word16 alp; 607 Word16 alp_16; 608 Word32 s; 609 Word32 alp0; 610 Word32 alp1; 611 Word16 i; 612 Word32 L_temp; 613 Word16 *p_codvec = &codvec[0]; 614 615 OSCL_UNUSED_ARG(pOverflow); 616 617 psk = -1; 618 alpk = 1; 619 620 /* Unrolled the following FOR loop to save MIPS */ 621 /* for (i = 0; i < NB_PULSE; i++) */ 622 /* { */ 623 /* *(codvec + i) = i; */ 624 /* } */ 625 626 *(p_codvec++) = 0; 627 *(p_codvec) = 1; 628 629 for (track1 = 0; track1 < 2; track1++) 630 { 631 /* fix starting position */ 632 633 i = (subNr << 1) + (track1 << 3); 634 *ipos = *(startPos + i); 635 *(ipos + 1) = *(startPos + i + 1); 636 637 638 /*----------------------------------------------------------* 639 * i0 loop: try 8 positions. * 640 *----------------------------------------------------------*/ 641 642 for (i0 = *ipos; i0 < L_CODE; i0 += STEP) 643 { 644 ps0 = *(dn + i0); 645 646 /* Left shift by 1 converts integer product to */ 647 /* fractional product. */ 648 alp0 = (Word32) rr[i0][i0] << 14; 649 650 /*--------------------------------------------------* 651 * i1 loop: 8 positions. * 652 *--------------------------------------------------*/ 653 654 sq = -1; 655 alp = 1; 656 ix = *(ipos + 1); 657 658 /*--------------------------------------------------* 659 * These index have low complexity address * 660 * computation because they are, in fact, pointers * 661 * with fixed increment. For example, "rr[i0][i2]" * 662 * is a pointer initialized to "&rr[i0][ipos[2]]" * 663 * and incremented by "STEP". * 664 *---------------------------------------------------*/ 665 666 for (i1 = *(ipos + 1); i1 < L_CODE; i1 += STEP) 667 { 668 /* idx increment = STEP */ 669 /* ps1 = add(ps0, *(dn + i1), pOverflow); */ 670 ps1 = ps0 + dn[i1]; 671 672 /* alp1 = alp0+rr[i0][i1]+1/2*rr[i1][i1]; */ 673 674 /* idx incr = STEP */ 675 /* Extra left shift by 1 converts integer */ 676 /* product to fractional product */ 677 /* alp1 = L_add(alp0, s, pOverflow); */ 678 alp1 = alp0 + ((Word32) rr[i1][i1] << 14); 679 680 /* idx incr = STEP */ 681 /* Extra left shift by 1 converts integer */ 682 /* product to fractional product */ 683 /* alp1 = L_add(alp1, s, pOverflow); */ 684 alp1 += (Word32) rr[i0][i1] << 15; 685 686 /* sq1 = mult(ps1, ps1, pOverflow); */ 687 sq1 = (Word16)(((Word32) ps1 * ps1) >> 15); 688 689 /* alp_16 = pv_round(alp1, pOverflow); */ 690 alp_16 = (Word16)((alp1 + (Word32) 0x00008000L) >> 16); 691 692 /* L_temp = L_mult(alp, sq1, pOverflow); */ 693 L_temp = ((Word32) alp * sq1) << 1; 694 695 /* s = L_msu(L_temp, sq, alp_16, pOverflow); */ 696 s = L_temp - (((Word32) sq * alp_16) << 1); 697 698 if (s > 0) 699 { 700 sq = sq1; 701 alp = alp_16; 702 ix = i1; 703 } 704 } 705 706 /* memorize codevector if this one is better than the last one. */ 707 708 /* L_temp = L_mult(alpk, sq, pOverflow); */ 709 L_temp = ((Word32) alpk * sq) << 1; 710 711 /* s = L_msu(L_temp, psk, alp, pOverflow); */ 712 s = L_temp - (((Word32) psk * alp) << 1); 713 714 if (s > 0) 715 { 716 psk = sq; 717 alpk = alp; 718 p_codvec = &codvec[0]; 719 *(p_codvec++) = i0; 720 *(p_codvec) = ix; 721 } 722 } 723 } 724 725 return; 726 } 727 728 /****************************************************************************/ 729 730 /* 731 ------------------------------------------------------------------------------ 732 FUNCTION NAME: Test_search_2i40 733 ------------------------------------------------------------------------------ 734 INPUT AND OUTPUT DEFINITIONS 735 736 Inputs: 737 subNr = subframe number (Word16) 738 dn = vector containing the correlation between target and the impulse 739 response of the weighted synthesis filter (Word16) 740 rr = autocorrelation matrix (Word16) 741 codvec = algebraic codebook vector (Word16) 742 743 Outputs: 744 codvec contains the newly calculated codevectors 745 746 Returns: 747 None 748 749 Global Variables Used: 750 None 751 752 Local Variables Needed: 753 startPos = table containing the start positions used by fixed codebook 754 routines (const Word16) 755 756 ------------------------------------------------------------------------------ 757 FUNCTION DESCRIPTION 758 759 This function provides external access to the local function search_2i40. 760 761 ------------------------------------------------------------------------------ 762 REQUIREMENTS 763 764 None 765 766 ------------------------------------------------------------------------------ 767 REFERENCES 768 769 [1] c2_9pf.c, UMTS GSM AMR speech codec, R99 - Version 3.2.0, March 2, 2001 770 771 ------------------------------------------------------------------------------ 772 PSEUDO-CODE 773 774 CALL search_2i40 ( subNr = subNr 775 dn = dn 776 rr = rr 777 codvec = codvec ) 778 MODIFYING(nothing) 779 RETURNING(nothing) 780 781 ------------------------------------------------------------------------------ 782 RESOURCES USED [optional] 783 784 When the code is written for a specific target processor the 785 the resources used should be documented below. 786 787 HEAP MEMORY USED: x bytes 788 789 STACK MEMORY USED: x bytes 790 791 CLOCK CYCLES: (cycle count equation for this function) + (variable 792 used to represent cycle count for each subroutine 793 called) 794 where: (cycle count variable) = cycle count for [subroutine 795 name] 796 797 ------------------------------------------------------------------------------ 798 CAUTION [optional] 799 [State any special notes, constraints or cautions for users of this function] 800 801 ------------------------------------------------------------------------------ 802 */ 803 804 void Test_search_2i40( 805 Word16 subNr, /* i : subframe number */ 806 Word16 dn[], /* i : correlation between target and h[] */ 807 Word16 rr[][L_CODE], /* i : matrix of autocorrelation */ 808 Word16 codvec[], /* o : algebraic codebook vector */ 809 Flag * pOverflow /* o : Flag set when overflow occurs */ 810 ) 811 { 812 /*---------------------------------------------------------------------------- 813 CALL search_2i40 ( subNr = subNr 814 dn = dn 815 rr = rr 816 codvec = codvec ) 817 MODIFYING(nothing) 818 RETURNING(nothing) 819 ----------------------------------------------------------------------------*/ 820 search_2i40( 821 subNr, 822 dn, 823 rr, 824 codvec, 825 pOverflow); 826 827 return; 828 } 829 830 /****************************************************************************/ 831 832 /* 833 ------------------------------------------------------------------------------ 834 FUNCTION NAME: build_code 835 ------------------------------------------------------------------------------ 836 INPUT AND OUTPUT DEFINITIONS 837 838 Inputs: 839 subNr = subframe number (Word16) 840 codvec = vector containing the position of pulses (Word16) 841 dn_sign = vector containing the sign of pulses (Word16) 842 cod = innovative code vector (Word16) 843 h = vector containing the impulse response of the weighted 844 synthesis filter (Word16) 845 y = vector containing the filtered innovative code (Word16) 846 sign = vector containing the sign of 2 pulses (Word16) 847 848 Outputs: 849 cod vector contains the new innovative code 850 y vector contains the new filtered innovative code 851 sign vector contains the sign of 2 pulses 852 853 Returns: 854 indx = codebook index (Word16) 855 856 Global Variables Used: 857 None 858 859 Local Variables Needed: 860 trackTable = table used for tracking codewords (Word16) 861 862 ------------------------------------------------------------------------------ 863 FUNCTION DESCRIPTION 864 865 This function builds the codeword, the filtered codeword and index of the 866 codevector, based on the signs and positions of 2 pulses. 867 868 ------------------------------------------------------------------------------ 869 REQUIREMENTS 870 871 None 872 873 ------------------------------------------------------------------------------ 874 REFERENCES 875 876 [1] c2_9pf.c, UMTS GSM AMR speech codec, R99 - Version 3.2.0, March 2, 2001 877 878 ------------------------------------------------------------------------------ 879 PSEUDO-CODE 880 881 static Word16 build_code( 882 Word16 subNr, // i : subframe number 883 Word16 codvec[], // i : position of pulses 884 Word16 dn_sign[], // i : sign of pulses 885 Word16 cod[], // o : innovative code vector 886 Word16 h[], // i : impulse response of weighted synthesis filter 887 Word16 y[], // o : filtered innovative code 888 Word16 sign[] // o : sign of 2 pulses 889 ) 890 { 891 Word16 i, j, k, track, first, index, _sign[NB_PULSE], indx, rsign; 892 Word16 *p0, *p1, *pt; 893 Word32 s; 894 static Word16 trackTable[4*5] = { 895 0, 1, 0, 1, -1, // subframe 1; track to code; -1 do not code this position 896 0, -1, 1, 0, 1, // subframe 2 897 0, 1, 0, -1, 1, // subframe 3 898 0, 1, -1, 0, 1};// subframe 4 899 900 pt = &trackTable[add(subNr, shl(subNr, 2))]; 901 902 for (i = 0; i < L_CODE; i++) { 903 cod[i] = 0; 904 } 905 906 indx = 0; 907 rsign = 0; 908 for (k = 0; k < NB_PULSE; k++) { 909 i = codvec[k]; // read pulse position 910 j = dn_sign[i]; // read sign 911 912 index = mult(i, 6554); // index = pos/5 913 // track = pos%5 914 track = sub(i, extract_l(L_shr(L_mult(index, 5), 1))); 915 916 first = pt[track]; 917 918 if (first == 0) { 919 if (k == 0) { 920 track = 0; 921 } else { 922 track = 1; 923 index = shl(index, 3); 924 } 925 } else { 926 if (k == 0) { 927 track = 0; 928 index = add(index, 64); // table bit is MSB 929 } else { 930 track = 1; 931 index = shl(index, 3); 932 } 933 } 934 935 if (j > 0) { 936 cod[i] = 8191; 937 _sign[k] = 32767; 938 rsign = add(rsign, shl(1, track)); 939 } else { 940 cod[i] = -8192; 941 _sign[k] = (Word16) - 32768L; 942 } 943 944 indx = add(indx, index); 945 } 946 *sign = rsign; 947 948 p0 = h - codvec[0]; 949 p1 = h - codvec[1]; 950 951 for (i = 0; i < L_CODE; i++) { 952 s = 0; 953 s = L_mac(s, *p0++, _sign[0]); 954 s = L_mac(s, *p1++, _sign[1]); 955 y[i] = pv_round(s); 956 } 957 958 return indx; 959 } 960 961 ------------------------------------------------------------------------------ 962 RESOURCES USED [optional] 963 964 When the code is written for a specific target processor the 965 the resources used should be documented below. 966 967 HEAP MEMORY USED: x bytes 968 969 STACK MEMORY USED: x bytes 970 971 CLOCK CYCLES: (cycle count equation for this function) + (variable 972 used to represent cycle count for each subroutine 973 called) 974 where: (cycle count variable) = cycle count for [subroutine 975 name] 976 977 ------------------------------------------------------------------------------ 978 CAUTION [optional] 979 [State any special notes, constraints or cautions for users of this function] 980 981 ------------------------------------------------------------------------------ 982 */ 983 984 static Word16 build_code( 985 Word16 subNr, /* i : subframe number */ 986 Word16 codvec[], /* i : position of pulses */ 987 Word16 dn_sign[], /* i : sign of pulses */ 988 Word16 cod[], /* o : innovative code vector */ 989 Word16 h[], /* i : impulse response of weighted synthesis */ 990 /* filter */ 991 Word16 y[], /* o : filtered innovative code */ 992 Word16 sign[], /* o : sign of 2 pulses */ 993 Flag *pOverflow /* o : Flag set when overflow occurs */ 994 ) 995 { 996 Word16 i; 997 Word16 j; 998 Word16 k; 999 Word16 track; 1000 Word16 first; 1001 Word16 index; 1002 Word16 rsign; 1003 Word16 indx; 1004 Word16 _sign[NB_PULSE]; 1005 Word16 *p0; 1006 Word16 *p1; 1007 1008 const Word16 *pt; 1009 1010 Word32 s; 1011 1012 pt = trackTable + subNr + (subNr << 2); 1013 1014 for (i = 0; i < L_CODE; i++) 1015 { 1016 *(cod + i) = 0; 1017 } 1018 1019 indx = 0; 1020 rsign = 0; 1021 1022 for (k = 0; k < NB_PULSE; k++) 1023 { 1024 i = *(codvec + k); /* read pulse position */ 1025 j = *(dn_sign + i); /* read sign */ 1026 1027 s = ((Word32)(i * 6554)) >> 15; 1028 index = (Word16) s; /* index = pos/5 */ 1029 1030 track = i - (5 * index); /* track = pos%5 */ 1031 1032 first = *(pt + track); 1033 1034 1035 if (k == 0) 1036 { 1037 track = 0; 1038 1039 if (first != 0) 1040 { 1041 index += 64; /* table bit is MSB */ 1042 } 1043 } 1044 else 1045 { 1046 track = 1; 1047 index <<= 3; 1048 } 1049 1050 if (j > 0) 1051 { 1052 *(cod + i) = 8191; 1053 *(_sign + k) = 32767; 1054 rsign += (1 << track); 1055 } 1056 else 1057 { 1058 *(cod + i) = ~(8192) + 1; 1059 *(_sign + k) = (Word16)(~(32768) + 1); 1060 } 1061 1062 indx += index; 1063 } 1064 1065 *sign = rsign; 1066 1067 p0 = h - *codvec; 1068 p1 = h - *(codvec + 1); 1069 1070 for (i = 0; i < L_CODE; i++) 1071 { 1072 s = 0; 1073 s = 1074 L_mult( 1075 *p0++, 1076 *_sign, 1077 pOverflow); 1078 1079 s = 1080 L_mac( 1081 s, 1082 *p1++, 1083 *(_sign + 1), 1084 pOverflow); 1085 1086 *(y + i) = 1087 pv_round( 1088 s, 1089 pOverflow); 1090 } 1091 1092 return(indx); 1093 } 1094 1095 /****************************************************************************/ 1096 1097 /* 1098 ------------------------------------------------------------------------------ 1099 FUNCTION NAME: Test_build_code 1100 ------------------------------------------------------------------------------ 1101 INPUT AND OUTPUT DEFINITIONS 1102 1103 Inputs: 1104 subNr = subframe number (Word16) 1105 codvec = vector containing the position of pulses (Word16) 1106 dn_sign = vector containing the sign of pulses (Word16) 1107 cod = innovative code vector (Word16) 1108 h = vector containing the impulse response of the weighted 1109 synthesis filter (Word16) 1110 y = vector containing the filtered innovative code (Word16) 1111 sign = vector containing the sign of 2 pulses (Word16) 1112 1113 Outputs: 1114 cod vector contains the new innovative code 1115 y vector contains the new filtered innovative code 1116 sign vector contains the sign of 2 pulses 1117 1118 Returns: 1119 indx = codebook index (Word16) 1120 1121 Global Variables Used: 1122 None 1123 1124 Local Variables Needed: 1125 trackTable = table used for tracking codewords (Word16) 1126 1127 ------------------------------------------------------------------------------ 1128 FUNCTION DESCRIPTION 1129 1130 This function provides external access to the local function build_code. 1131 1132 ------------------------------------------------------------------------------ 1133 REQUIREMENTS 1134 1135 None 1136 1137 ------------------------------------------------------------------------------ 1138 REFERENCES 1139 1140 [1] c2_9pf.c, UMTS GSM AMR speech codec, R99 - Version 3.2.0, March 2, 2001 1141 1142 ------------------------------------------------------------------------------ 1143 PSEUDO-CODE 1144 1145 CALL build_code ( subNr = subNr 1146 codvec = codvec 1147 dn_sign = dn_sign 1148 cod = cod 1149 h = h 1150 y = y 1151 sign = sign ) 1152 MODIFYING(nothing) 1153 RETURNING(indx) 1154 1155 ------------------------------------------------------------------------------ 1156 RESOURCES USED [optional] 1157 1158 When the code is written for a specific target processor the 1159 the resources used should be documented below. 1160 1161 HEAP MEMORY USED: x bytes 1162 1163 STACK MEMORY USED: x bytes 1164 1165 CLOCK CYCLES: (cycle count equation for this function) + (variable 1166 used to represent cycle count for each subroutine 1167 called) 1168 where: (cycle count variable) = cycle count for [subroutine 1169 name] 1170 1171 ------------------------------------------------------------------------------ 1172 CAUTION [optional] 1173 [State any special notes, constraints or cautions for users of this function] 1174 1175 ------------------------------------------------------------------------------ 1176 */ 1177 1178 Word16 Test_build_code( 1179 Word16 subNr, /* i : subframe number */ 1180 Word16 codvec[], /* i : position of pulses */ 1181 Word16 dn_sign[], /* i : sign of pulses */ 1182 Word16 cod[], /* o : innovative code vector */ 1183 Word16 h[], /* i : impulse response of weighted synthesis */ 1184 /* filter */ 1185 Word16 y[], /* o : filtered innovative code */ 1186 Word16 sign[], /* o : sign of 2 pulses */ 1187 Flag * pOverflow /* o : Flag set when overflow occurs */ 1188 ) 1189 { 1190 Word16 test_index; 1191 1192 /*---------------------------------------------------------------------------- 1193 CALL build_code ( subNr = subNr 1194 codvec = codvec 1195 dn_sign = dn_sign 1196 cod = cod 1197 h = h 1198 y = y 1199 sign = sign ) 1200 MODIFYING(nothing) 1201 RETURNING(indx) 1202 ----------------------------------------------------------------------------*/ 1203 test_index = 1204 build_code( 1205 subNr, 1206 codvec, 1207 dn_sign, 1208 cod, 1209 h, 1210 y, 1211 sign, 1212 pOverflow); 1213 1214 return(test_index); 1215 } 1216 1217 #ifdef __cplusplus 1218 } 1219 #endif 1220