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 Pathname: ./audio/gsm-amr/c/src/gc_pred.c 32 Functions: 33 gc_pred_reset 34 gc_pred 35 gc_pred_update 36 gc_pred_average_limited 37 38 ------------------------------------------------------------------------------ 39 MODULE DESCRIPTION 40 41 This file contains the functions that perform codebook gain MA prediction. 42 43 ------------------------------------------------------------------------------ 44 */ 45 46 /*---------------------------------------------------------------------------- 47 ; INCLUDES 48 ----------------------------------------------------------------------------*/ 49 #include "gc_pred.h" 50 #include "basicop_malloc.h" 51 #include "basic_op.h" 52 #include "cnst.h" 53 #include "log2.h" 54 55 /*---------------------------------------------------------------------------- 56 ; MACROS 57 ; Define module specific macros here 58 ----------------------------------------------------------------------------*/ 59 60 /*---------------------------------------------------------------------------- 61 ; DEFINES 62 ; Include all pre-processor statements here. Include conditional 63 ; compile variables also. 64 ----------------------------------------------------------------------------*/ 65 #define NPRED 4 /* number of prediction taps */ 66 67 /* average innovation energy. */ 68 /* MEAN_ENER = 36.0/constant, constant = 20*Log10(2) */ 69 #define MEAN_ENER_MR122 783741L /* 36/(20*log10(2)) (Q17) */ 70 71 /* minimum quantized energy: -14 dB */ 72 #define MIN_ENERGY -14336 /* 14 Q10 */ 73 #define MIN_ENERGY_MR122 -2381 /* 14 / (20*log10(2)) Q10 */ 74 75 /*---------------------------------------------------------------------------- 76 ; LOCAL FUNCTION DEFINITIONS 77 ; Function Prototype declaration 78 ----------------------------------------------------------------------------*/ 79 80 /*---------------------------------------------------------------------------- 81 ; LOCAL VARIABLE DEFINITIONS 82 ; Variable declaration - defined here and used outside this module 83 ----------------------------------------------------------------------------*/ 84 85 /* MA prediction coefficients (Q13) */ 86 static const Word16 pred[NPRED] = {5571, 4751, 2785, 1556}; 87 88 /* MA prediction coefficients (Q6) */ 89 static const Word16 pred_MR122[NPRED] = {44, 37, 22, 12}; 90 91 /* 92 ------------------------------------------------------------------------------ 93 FUNCTION NAME: gc_pred_reset 94 ------------------------------------------------------------------------------ 95 INPUT AND OUTPUT DEFINITIONS 96 97 Inputs: 98 state = pointer to a structure of type gc_predState 99 100 Outputs: 101 past_qua_en field in the structure pointed to by state is initialized 102 to MIN_ENERGY 103 past_qua_en_MR122 field in the structure pointed to by state is 104 initialized to MIN_ENERGY_MR122 105 106 Returns: 107 return_value = 0, if reset was successful; -1, otherwise (int) 108 109 Global Variables Used: 110 None 111 112 Local Variables Needed: 113 None 114 115 ------------------------------------------------------------------------------ 116 FUNCTION DESCRIPTION 117 118 This function initializes the state memory used by gc_pred to zero. 119 120 ------------------------------------------------------------------------------ 121 REQUIREMENTS 122 123 None 124 125 ------------------------------------------------------------------------------ 126 REFERENCES 127 128 gc_pred.c, UMTS GSM AMR speech codec, R99 - Version 3.2.0, March 2, 2001 129 130 ------------------------------------------------------------------------------ 131 PSEUDO-CODE 132 133 int gc_pred_reset (gc_predState *state) 134 { 135 Word16 i; 136 137 if (state == (gc_predState *) NULL){ 138 fprintf(stderr, "gc_pred_reset: invalid parameter\n"); 139 return -1; 140 } 141 142 for(i = 0; i < NPRED; i++) 143 { 144 state->past_qua_en[i] = MIN_ENERGY; 145 state->past_qua_en_MR122[i] = MIN_ENERGY_MR122; 146 } 147 return 0; 148 } 149 150 ------------------------------------------------------------------------------ 151 RESOURCES USED [optional] 152 153 When the code is written for a specific target processor the 154 the resources used should be documented below. 155 156 HEAP MEMORY USED: x bytes 157 158 STACK MEMORY USED: x bytes 159 160 CLOCK CYCLES: (cycle count equation for this function) + (variable 161 used to represent cycle count for each subroutine 162 called) 163 where: (cycle count variable) = cycle count for [subroutine 164 name] 165 166 ------------------------------------------------------------------------------ 167 CAUTION [optional] 168 [State any special notes, constraints or cautions for users of this function] 169 170 ------------------------------------------------------------------------------ 171 */ 172 173 Word16 gc_pred_reset(gc_predState *state) 174 { 175 Word16 i; 176 177 if (state == (gc_predState *) NULL) 178 { 179 /* fprintf(stderr, "gc_pred_reset: invalid parameter\n"); */ 180 return -1; 181 } 182 183 for (i = 0; i < NPRED; i++) 184 { 185 state->past_qua_en[i] = MIN_ENERGY; 186 state->past_qua_en_MR122[i] = MIN_ENERGY_MR122; 187 } 188 189 return(0); 190 } 191 192 /****************************************************************************/ 193 194 /* 195 ------------------------------------------------------------------------------ 196 FUNCTION NAME: gc_pred 197 ------------------------------------------------------------------------------ 198 INPUT AND OUTPUT DEFINITIONS 199 200 Inputs: 201 st = pointer to a structure of type gc_predState 202 mode = AMR mode (enum Mode) 203 code = pointer to the innovative codebook vector; Q12 in MR122 mode, 204 otherwise, Q13 (Word16) 205 exp_gcode0 = pointer to the exponent part of predicted gain factor 206 (Q0) (Word16) 207 frac_gcode0 = pointer to the fractional part of predicted gain factor 208 (Q15) (Word16) 209 exp_en = pointer to the exponent part of the innovation energy; this 210 is calculated for MR795 mode, Q0 (Word16) 211 frac_en = pointer to the fractional part of the innovation energy; 212 this is calculated for MR795 mode, Q15 (Word16) 213 pOverflow = pointer to overflow (Flag) 214 215 Outputs: 216 store pointed to by exp_gcode0 contains the exponent part of the 217 recently calculated predicted gain factor 218 store pointed to by frac_gcode0 contains the fractional part of the 219 recently calculated predicted gain factor 220 store pointed to by exp_en contains the exponent part of the 221 recently calculated innovation energy 222 store pointed to by frac_en contains the fractional part of the 223 recently calculated innovation energy 224 pOverflow = 1 if the math functions called by gc_pred 225 results in overflow else zero. 226 227 Returns: 228 None 229 230 Global Variables Used: 231 None 232 233 Local Variables Needed: 234 pred = table of MA prediction coefficients (Q13) (Word16) 235 pred_MR122 = table of MA prediction coefficients (Q6) (Word16) 236 237 ------------------------------------------------------------------------------ 238 FUNCTION DESCRIPTION 239 240 This function performs the MA prediction of the innovation energy (in 241 dB/(20*log10(2))), with the mean removed. 242 243 ------------------------------------------------------------------------------ 244 REQUIREMENTS 245 246 None 247 248 ------------------------------------------------------------------------------ 249 REFERENCES 250 251 gc_pred.c, UMTS GSM AMR speech codec, R99 - Version 3.2.0, March 2, 2001 252 253 ------------------------------------------------------------------------------ 254 PSEUDO-CODE 255 256 The original etsi reference code uses a global flag Overflow. However, in the 257 actual implementation a pointer to a the overflow flag is passed in. 258 259 void 260 gc_pred( 261 gc_predState *st, // i/o: State struct 262 enum Mode mode, // i : AMR mode 263 Word16 *code, // i : innovative codebook vector (L_SUBFR) 264 // MR122: Q12, other modes: Q13 265 Word16 *exp_gcode0, // o : exponent of predicted gain factor, Q0 266 Word16 *frac_gcode0,// o : fraction of predicted gain factor Q15 267 Word16 *exp_en, // o : exponent of innovation energy, Q0 268 // (only calculated for MR795) 269 Word16 *frac_en // o : fraction of innovation energy, Q15 270 // (only calculated for MR795) 271 ) 272 { 273 Word16 i; 274 Word32 ener_code; 275 Word16 exp, frac; 276 277 *-------------------------------------------------------------------* 278 * energy of code: * 279 * ~~~~~~~~~~~~~~~ * 280 * ener_code = sum(code[i]^2) * 281 *-------------------------------------------------------------------* 282 ener_code = L_mac((Word32) 0, code[0], code[0]); 283 // MR122: Q12*Q12 -> Q25 284 // others: Q13*Q13 -> Q27 285 for (i = 1; i < L_SUBFR; i++) 286 ener_code = L_mac(ener_code, code[i], code[i]); 287 288 if (sub (mode, MR122) == 0) 289 { 290 Word32 ener; 291 292 // ener_code = ener_code / lcode; lcode = 40; 1/40 = 26214 Q20 293 ener_code = L_mult (pv_round (ener_code), 26214); // Q9 * Q20 -> Q30 294 295 *-------------------------------------------------------------------* 296 * energy of code: * 297 * ~~~~~~~~~~~~~~~ * 298 * ener_code(Q17) = 10 * Log10(energy) / constant * 299 * = 1/2 * Log2(energy) * 300 * constant = 20*Log10(2) * 301 *-------------------------------------------------------------------* 302 // ener_code = 1/2 * Log2(ener_code); Note: Log2=log2+30 303 Log2(ener_code, &exp, &frac); 304 ener_code = L_Comp (sub (exp, 30), frac); // Q16 for log() 305 // ->Q17 for 1/2 log() 306 307 *-------------------------------------------------------------------* 308 * predicted energy: * 309 * ~~~~~~~~~~~~~~~~~ * 310 * ener(Q24) = (Emean + sum{pred[i]*past_en[i]})/constant * 311 * = MEAN_ENER + sum(pred[i]*past_qua_en[i]) * 312 * constant = 20*Log10(2) * 313 *-------------------------------------------------------------------* 314 315 ener = MEAN_ENER_MR122; // Q24 (Q17) 316 for (i = 0; i < NPRED; i++) 317 { 318 ener = L_mac (ener, st->past_qua_en_MR122[i], pred_MR122[i]); 319 // Q10 * Q13 -> Q24 320 // Q10 * Q6 -> Q17 321 } 322 323 *-------------------------------------------------------------------* 324 * predicted codebook gain * 325 * ~~~~~~~~~~~~~~~~~~~~~~~ * 326 * gc0 = Pow10( (ener*constant - ener_code*constant) / 20 ) * 327 * = Pow2(ener-ener_code) * 328 * = Pow2(int(d)+frac(d)) * 329 * * 330 * (store exp and frac for pow2()) * 331 *-------------------------------------------------------------------* 332 333 ener = L_shr (L_sub (ener, ener_code), 1); // Q16 334 L_Extract(ener, exp_gcode0, frac_gcode0); 335 } 336 else // all modes except 12.2 337 { 338 Word32 L_tmp; 339 Word16 exp_code, gcode0; 340 341 *-----------------------------------------------------------------* 342 * Compute: means_ener - 10log10(ener_code/ L_sufr) * 343 *-----------------------------------------------------------------* 344 345 exp_code = norm_l (ener_code); 346 ener_code = L_shl (ener_code, exp_code); 347 348 // Log2 = log2 + 27 349 Log2_norm (ener_code, exp_code, &exp, &frac); 350 351 // fact = 10/log2(10) = 3.01 = 24660 Q13 352 L_tmp = Mpy_32_16(exp, frac, -24660); // Q0.Q15 * Q13 -> Q14 353 354 * L_tmp = means_ener - 10log10(ener_code/L_SUBFR) 355 * = means_ener - 10log10(ener_code) + 10log10(L_SUBFR) 356 * = K - fact * Log2(ener_code) 357 * = K - fact * log2(ener_code) - fact*27 358 * 359 * ==> K = means_ener + fact*27 + 10log10(L_SUBFR) 360 * 361 * means_ener = 33 = 540672 Q14 (MR475, MR515, MR59) 362 * means_ener = 28.75 = 471040 Q14 (MR67) 363 * means_ener = 30 = 491520 Q14 (MR74) 364 * means_ener = 36 = 589824 Q14 (MR795) 365 * means_ener = 33 = 540672 Q14 (MR102) 366 * 10log10(L_SUBFR) = 16.02 = 262481.51 Q14 367 * fact * 27 = 1331640 Q14 368 * ----------------------------------------- 369 * (MR475, MR515, MR59) K = 2134793.51 Q14 ~= 16678 * 64 * 2 370 * (MR67) K = 2065161.51 Q14 ~= 32268 * 32 * 2 371 * (MR74) K = 2085641.51 Q14 ~= 32588 * 32 * 2 372 * (MR795) K = 2183945.51 Q14 ~= 17062 * 64 * 2 373 * (MR102) K = 2134793.51 Q14 ~= 16678 * 64 * 2 374 375 376 if (sub (mode, MR102) == 0) 377 { 378 // mean = 33 dB 379 L_tmp = L_mac(L_tmp, 16678, 64); // Q14 380 } 381 else if (sub (mode, MR795) == 0) 382 { 383 // ener_code = <xn xn> * 2^27*2^exp_code 384 // frac_en = ener_code / 2^16 385 // = <xn xn> * 2^11*2^exp_code 386 // <xn xn> = <xn xn>*2^11*2^exp * 2^exp_en 387 // := frac_en * 2^exp_en 388 389 // ==> exp_en = -11-exp_code; 390 391 *frac_en = extract_h (ener_code); 392 *exp_en = sub (-11, exp_code); 393 394 // mean = 36 dB 395 L_tmp = L_mac(L_tmp, 17062, 64); // Q14 396 } 397 else if (sub (mode, MR74) == 0) 398 { 399 // mean = 30 dB 400 L_tmp = L_mac(L_tmp, 32588, 32); // Q14 401 } 402 else if (sub (mode, MR67) == 0) 403 { 404 // mean = 28.75 dB 405 L_tmp = L_mac(L_tmp, 32268, 32); // Q14 406 } 407 else // MR59, MR515, MR475 408 { 409 // mean = 33 dB 410 L_tmp = L_mac(L_tmp, 16678, 64); // Q14 411 } 412 413 *-----------------------------------------------------------------* 414 * Compute gcode0. * 415 * = Sum(i=0,3) pred[i]*past_qua_en[i] - ener_code + mean_ener * 416 *-----------------------------------------------------------------* 417 418 L_tmp = L_shl(L_tmp, 10); // Q24 419 for (i = 0; i < 4; i++) 420 L_tmp = L_mac(L_tmp, pred[i], st->past_qua_en[i]); 421 // Q13 * Q10 -> Q24 422 423 gcode0 = extract_h(L_tmp); // Q8 424 425 *-----------------------------------------------------------------* 426 * gcode0 = pow(10.0, gcode0/20) * 427 * = pow(2, 3.3219*gcode0/20) * 428 * = pow(2, 0.166*gcode0) * 429 *-----------------------------------------------------------------* 430 431 // 5439 Q15 = 0.165985 432 // (correct: 1/(20*log10(2)) 0.166096 = 5443 Q15) 433 if (sub (mode, MR74) == 0) // For IS641 bitexactness 434 L_tmp = L_mult(gcode0, 5439); // Q8 * Q15 -> Q24 435 else 436 L_tmp = L_mult(gcode0, 5443); // Q8 * Q15 -> Q24 437 438 L_tmp = L_shr(L_tmp, 8); // -> Q16 439 L_Extract(L_tmp, exp_gcode0, frac_gcode0); // -> Q0.Q15 440 } 441 } 442 443 ------------------------------------------------------------------------------ 444 RESOURCES USED [optional] 445 446 When the code is written for a specific target processor the 447 the resources used should be documented below. 448 449 HEAP MEMORY USED: x bytes 450 451 STACK MEMORY USED: x bytes 452 453 CLOCK CYCLES: (cycle count equation for this function) + (variable 454 used to represent cycle count for each subroutine 455 called) 456 where: (cycle count variable) = cycle count for [subroutine 457 name] 458 459 ------------------------------------------------------------------------------ 460 CAUTION [optional] 461 [State any special notes, constraints or cautions for users of this function] 462 463 ------------------------------------------------------------------------------ 464 */ 465 466 void gc_pred( 467 gc_predState *st, /* i/o: State struct */ 468 enum Mode mode, /* i : AMR mode */ 469 Word16 *code, /* i : innovative codebook vector (L_SUBFR) */ 470 /* MR122: Q12, other modes: Q13 */ 471 Word16 *exp_gcode0, /* o : exponent of predicted gain factor, Q0 */ 472 Word16 *frac_gcode0,/* o : fraction of predicted gain factor Q15 */ 473 Word16 *exp_en, /* o : exponent of innovation energy, Q0 */ 474 /* (only calculated for MR795) */ 475 Word16 *frac_en, /* o : fraction of innovation energy, Q15 */ 476 /* (only calculated for MR795) */ 477 Flag *pOverflow 478 ) 479 { 480 Word16 i; 481 Word32 L_temp1, L_temp2; 482 Word32 L_tmp; 483 Word32 ener_code; 484 Word32 ener; 485 Word16 exp, frac; 486 Word16 exp_code, gcode0; 487 Word16 tmp; 488 Word16 *p_code = &code[0]; 489 490 /*-------------------------------------------------------------------* 491 * energy of code: * 492 * ~~~~~~~~~~~~~~~ * 493 * ener_code = sum(code[i]^2) * 494 *-------------------------------------------------------------------*/ 495 ener_code = 0; 496 497 /* MR122: Q12*Q12 -> Q25 */ 498 /* others: Q13*Q13 -> Q27 */ 499 500 for (i = L_SUBFR >> 2; i != 0; i--) 501 { 502 tmp = *(p_code++); 503 ener_code += ((Word32) tmp * tmp) >> 3; 504 tmp = *(p_code++); 505 ener_code += ((Word32) tmp * tmp) >> 3; 506 tmp = *(p_code++); 507 ener_code += ((Word32) tmp * tmp) >> 3; 508 tmp = *(p_code++); 509 ener_code += ((Word32) tmp * tmp) >> 3; 510 } 511 512 ener_code <<= 4; 513 514 if (ener_code < 0) /* Check for saturation */ 515 { 516 ener_code = MAX_32; 517 } 518 519 if (mode == MR122) 520 { 521 /* ener_code = ener_code / lcode; lcode = 40; 1/40 = 26214 Q20 */ 522 /* Q9 * Q20 -> Q30 */ 523 524 ener_code = ((Word32)(pv_round(ener_code, pOverflow) * 26214)) << 1; 525 526 /*-------------------------------------------------------------* 527 * energy of code: * 528 * ~~~~~~~~~~~~~~~ * 529 * ener_code(Q17) = 10 * Log10(energy) / constant * 530 * = 1/2 * Log2(energy) * 531 * constant = 20*Log10(2) * 532 *-------------------------------------------------------------*/ 533 /* ener_code = 1/2 * Log2(ener_code); Note: Log2=log2+30 */ 534 Log2(ener_code, &exp, &frac, pOverflow); 535 536 /* Q16 for log() */ 537 /* ->Q17 for 1/2 log()*/ 538 539 L_temp1 = (Word32)(exp - 30) << 16; 540 ener_code = L_temp1 + ((Word32)frac << 1); 541 542 /*-------------------------------------------------------------* 543 * predicted energy: * 544 * ~~~~~~~~~~~~~~~~~ * 545 * ener(Q24) = (Emean + sum{pred[i]*past_en[i]})/constant * 546 * = MEAN_ENER + sum(pred[i]*past_qua_en[i]) * 547 * constant = 20*Log10(2) * 548 *-------------------------------------------------------------*/ 549 550 ener = MEAN_ENER_MR122; /* Q24 (Q17) */ 551 for (i = 0; i < NPRED; i++) 552 { 553 L_temp1 = (((Word32) st->past_qua_en_MR122[i]) * 554 pred_MR122[i]) << 1; 555 ener = L_add(ener, L_temp1, pOverflow); 556 557 /* Q10 * Q13 -> Q24 */ 558 /* Q10 * Q6 -> Q17 */ 559 } 560 561 /*---------------------------------------------------------------* 562 * predicted codebook gain * 563 * ~~~~~~~~~~~~~~~~~~~~~~~ * 564 * gc0 = Pow10( (ener*constant - ener_code*constant) / 20 ) * 565 * = Pow2(ener-ener_code) * 566 * = Pow2(int(d)+frac(d)) * 567 * * 568 * (store exp and frac for pow2()) * 569 *---------------------------------------------------------------*/ 570 /* Q16 */ 571 572 L_temp1 = L_sub(ener, ener_code, pOverflow); 573 574 575 *exp_gcode0 = (Word16)(L_temp1 >> 17); 576 577 L_temp2 = (Word32) * exp_gcode0 << 15; 578 L_temp1 >>= 2; 579 580 *frac_gcode0 = (Word16)(L_temp1 - L_temp2); 581 582 } 583 else /* all modes except 12.2 */ 584 { 585 /*-----------------------------------------------------------------* 586 * Compute: means_ener - 10log10(ener_code/ L_sufr) * 587 *-----------------------------------------------------------------*/ 588 589 exp_code = norm_l(ener_code); 590 ener_code = L_shl(ener_code, exp_code, pOverflow); 591 592 /* Log2 = log2 + 27 */ 593 Log2_norm(ener_code, exp_code, &exp, &frac); 594 595 /* fact = 10/log2(10) = 3.01 = 24660 Q13 */ 596 /* Q0.Q15 * Q13 -> Q14 */ 597 598 L_temp2 = (((Word32) exp) * -24660) << 1; 599 L_tmp = (((Word32) frac) * -24660) >> 15; 600 601 /* Sign-extend resulting product */ 602 if (L_tmp & (Word32) 0x00010000L) 603 { 604 L_tmp = L_tmp | (Word32) 0xffff0000L; 605 } 606 607 L_tmp = L_tmp << 1; 608 L_tmp = L_add(L_tmp, L_temp2, pOverflow); 609 610 611 /* L_tmp = means_ener - 10log10(ener_code/L_SUBFR) 612 * = means_ener - 10log10(ener_code) + 10log10(L_SUBFR) 613 * = K - fact * Log2(ener_code) 614 * = K - fact * log2(ener_code) - fact*27 615 * 616 * ==> K = means_ener + fact*27 + 10log10(L_SUBFR) 617 * 618 * means_ener = 33 = 540672 Q14 (MR475, MR515, MR59) 619 * means_ener = 28.75 = 471040 Q14 (MR67) 620 * means_ener = 30 = 491520 Q14 (MR74) 621 * means_ener = 36 = 589824 Q14 (MR795) 622 * means_ener = 33 = 540672 Q14 (MR102) 623 * 10log10(L_SUBFR) = 16.02 = 262481.51 Q14 624 * fact * 27 = 1331640 Q14 625 * ----------------------------------------- 626 * (MR475, MR515, MR59) K = 2134793.51 Q14 ~= 16678 * 64 * 2 627 * (MR67) K = 2065161.51 Q14 ~= 32268 * 32 * 2 628 * (MR74) K = 2085641.51 Q14 ~= 32588 * 32 * 2 629 * (MR795) K = 2183945.51 Q14 ~= 17062 * 64 * 2 630 * (MR102) K = 2134793.51 Q14 ~= 16678 * 64 * 2 631 */ 632 633 if (mode == MR102) 634 { 635 /* mean = 33 dB */ 636 L_temp2 = (Word32) 16678 << 7; 637 L_tmp = L_add(L_tmp, L_temp2, pOverflow); /* Q14 */ 638 } 639 else if (mode == MR795) 640 { 641 /* ener_code = <xn xn> * 2^27*2^exp_code 642 frac_en = ener_code / 2^16 643 = <xn xn> * 2^11*2^exp_code 644 <xn xn> = <xn xn>*2^11*2^exp * 2^exp_en 645 : = frac_en * 2^exp_en 646 ==> exp_en = -11-exp_code; */ 647 *frac_en = (Word16)(ener_code >> 16); 648 *exp_en = sub(-11, exp_code, pOverflow); 649 650 /* mean = 36 dB */ 651 L_temp2 = (Word32) 17062 << 7; 652 L_tmp = L_add(L_tmp, L_temp2, pOverflow); /* Q14 */ 653 } 654 else if (mode == MR74) 655 { 656 /* mean = 30 dB */ 657 L_temp2 = (Word32) 32588 << 6; 658 L_tmp = L_add(L_tmp, L_temp2, pOverflow); /* Q14 */ 659 } 660 else if (mode == MR67) 661 { 662 /* mean = 28.75 dB */ 663 L_temp2 = (Word32) 32268 << 6; 664 L_tmp = L_add(L_tmp, L_temp2, pOverflow); /* Q14 */ 665 } 666 else /* MR59, MR515, MR475 */ 667 { 668 /* mean = 33 dB */ 669 L_temp2 = (Word32) 16678 << 7; 670 L_tmp = L_add(L_tmp, L_temp2, pOverflow); /* Q14 */ 671 } 672 673 /*-------------------------------------------------------------* 674 * Compute gcode0. * 675 * = Sum(i=0,3) pred[i]*past_qua_en[i] - ener_code + mean_ener * 676 *--------------------------------------------------------------*/ 677 /* Q24 */ 678 if (L_tmp > (Word32) 0X001fffffL) 679 { 680 *pOverflow = 1; 681 L_tmp = MAX_32; 682 } 683 else if (L_tmp < (Word32) 0xffe00000L) 684 { 685 *pOverflow = 1; 686 L_tmp = MIN_32; 687 } 688 else 689 { 690 L_tmp = L_tmp << 10; 691 } 692 693 for (i = 0; i < 4; i++) 694 { 695 L_temp2 = ((((Word32) pred[i]) * st->past_qua_en[i]) << 1); 696 L_tmp = L_add(L_tmp, L_temp2, pOverflow); /* Q13 * Q10 -> Q24 */ 697 } 698 699 gcode0 = (Word16)(L_tmp >> 16); /* Q8 */ 700 701 /*-----------------------------------------------------------* 702 * gcode0 = pow(10.0, gcode0/20) * 703 * = pow(2, 3.3219*gcode0/20) * 704 * = pow(2, 0.166*gcode0) * 705 *-----------------------------------------------------------*/ 706 707 /* 5439 Q15 = 0.165985 */ 708 /* (correct: 1/(20*log10(2)) 0.166096 = 5443 Q15) */ 709 710 if (mode == MR74) /* For IS641 bitexactness */ 711 { 712 L_tmp = (((Word32) gcode0) * 5439) << 1; /* Q8 * Q15 -> Q24 */ 713 } 714 else 715 { 716 L_tmp = (((Word32) gcode0) * 5443) << 1; /* Q8 * Q15 -> Q24 */ 717 } 718 719 if (L_tmp < 0) 720 { 721 L_tmp = ~((~L_tmp) >> 8); 722 } 723 else 724 { 725 L_tmp = L_tmp >> 8; /* -> Q16 */ 726 } 727 728 *exp_gcode0 = (Word16)(L_tmp >> 16); 729 if (L_tmp < 0) 730 { 731 L_temp1 = ~((~L_tmp) >> 1); 732 } 733 else 734 { 735 L_temp1 = L_tmp >> 1; 736 } 737 L_temp2 = (Word32) * exp_gcode0 << 15; 738 *frac_gcode0 = (Word16)(L_sub(L_temp1, L_temp2, pOverflow)); 739 /* -> Q0.Q15 */ 740 } 741 742 return; 743 } 744 745 /****************************************************************************/ 746 747 /* 748 ------------------------------------------------------------------------------ 749 FUNCTION NAME: gc_pred_update 750 ------------------------------------------------------------------------------ 751 INPUT AND OUTPUT DEFINITIONS 752 753 Inputs: 754 st = pointer to a structure of type gc_predState 755 qua_ener_MR122 = quantized energy for update (Q10); calculated as 756 (log2(qua_err)) (Word16) 757 qua_ener = quantized energy for update (Q10); calculated as 758 (20*log10(qua_err)) (Word16) 759 760 Outputs: 761 structure pointed to by st contains the calculated quantized energy 762 for update 763 764 Returns: 765 None 766 767 Global Variables Used: 768 None 769 770 Local Variables Needed: 771 None 772 773 ------------------------------------------------------------------------------ 774 FUNCTION DESCRIPTION 775 776 This function updates the MA predictor with the last quantized energy. 777 778 ------------------------------------------------------------------------------ 779 REQUIREMENTS 780 781 None 782 783 ------------------------------------------------------------------------------ 784 REFERENCES 785 786 gc_pred.c, UMTS GSM AMR speech codec, R99 - Version 3.2.0, March 2, 2001 787 788 ------------------------------------------------------------------------------ 789 PSEUDO-CODE 790 791 void gc_pred_update( 792 gc_predState *st, // i/o: State struct 793 Word16 qua_ener_MR122, // i : quantized energy for update, Q10 794 // (log2(qua_err)) 795 Word16 qua_ener // i : quantized energy for update, Q10 796 // (20*log10(qua_err)) 797 ) 798 { 799 Word16 i; 800 801 for (i = 3; i > 0; i--) 802 { 803 st->past_qua_en[i] = st->past_qua_en[i - 1]; 804 st->past_qua_en_MR122[i] = st->past_qua_en_MR122[i - 1]; 805 } 806 807 st->past_qua_en_MR122[0] = qua_ener_MR122; // log2 (qua_err), Q10 808 809 st->past_qua_en[0] = qua_ener; // 20*log10(qua_err), Q10 810 811 } 812 813 ------------------------------------------------------------------------------ 814 RESOURCES USED [optional] 815 816 When the code is written for a specific target processor the 817 the resources used should be documented below. 818 819 HEAP MEMORY USED: x bytes 820 821 STACK MEMORY USED: x bytes 822 823 CLOCK CYCLES: (cycle count equation for this function) + (variable 824 used to represent cycle count for each subroutine 825 called) 826 where: (cycle count variable) = cycle count for [subroutine 827 name] 828 829 ------------------------------------------------------------------------------ 830 CAUTION [optional] 831 [State any special notes, constraints or cautions for users of this function] 832 833 ------------------------------------------------------------------------------ 834 */ 835 836 void gc_pred_update( 837 gc_predState *st, /* i/o: State struct */ 838 Word16 qua_ener_MR122, /* i : quantized energy for update, Q10 */ 839 /* (log2(qua_err)) */ 840 Word16 qua_ener /* i : quantized energy for update, Q10 */ 841 /* (20*log10(qua_err)) */ 842 ) 843 { 844 st->past_qua_en[3] = st->past_qua_en[2]; 845 st->past_qua_en_MR122[3] = st->past_qua_en_MR122[2]; 846 847 st->past_qua_en[2] = st->past_qua_en[1]; 848 st->past_qua_en_MR122[2] = st->past_qua_en_MR122[1]; 849 850 st->past_qua_en[1] = st->past_qua_en[0]; 851 st->past_qua_en_MR122[1] = st->past_qua_en_MR122[0]; 852 853 st->past_qua_en_MR122[0] = qua_ener_MR122; /* log2 (qua_err), Q10 */ 854 855 st->past_qua_en[0] = qua_ener; /* 20*log10(qua_err), Q10 */ 856 857 return; 858 } 859 860 /****************************************************************************/ 861 862 /* 863 ------------------------------------------------------------------------------ 864 FUNCTION NAME: gc_pred_average_limited 865 ------------------------------------------------------------------------------ 866 INPUT AND OUTPUT DEFINITIONS 867 868 Inputs: 869 st = pointer to a structure of type gc_predState 870 ener_avg_MR122 = pointer to the averaged quantized energy (Q10); 871 calculated as (log2(qua_err)) (Word16) 872 ener_avg = pointer to the averaged quantized energy (Q10); calculated 873 as (20*log10(qua_err)) (Word16) 874 pOverflow = pointer to overflow (Flag) 875 876 Outputs: 877 store pointed to by ener_avg_MR122 contains the new averaged quantized 878 energy 879 store pointed to by ener_avg contains the new averaged quantized 880 energy 881 pOverflow = 1 if the math functions called by gc_pred_average_limited 882 results in overflow else zero. 883 884 Returns: 885 None 886 887 Global Variables Used: 888 None 889 890 Local Variables Needed: 891 None 892 893 ------------------------------------------------------------------------------ 894 FUNCTION DESCRIPTION 895 896 This function calculates the average of MA predictor state values (with a 897 lower limit) used in error concealment. 898 899 ------------------------------------------------------------------------------ 900 REQUIREMENTS 901 902 None 903 904 ------------------------------------------------------------------------------ 905 REFERENCES 906 907 gc_pred.c, UMTS GSM AMR speech codec, R99 - Version 3.2.0, March 2, 2001 908 909 ------------------------------------------------------------------------------ 910 PSEUDO-CODE 911 912 The original etsi reference code uses a global flag Overflow. However, in the 913 actual implementation a pointer to a the overflow flag is passed in. 914 915 void gc_pred_average_limited( 916 gc_predState *st, // i: State struct 917 Word16 *ener_avg_MR122, // o: everaged quantized energy, Q10 918 // (log2(qua_err)) 919 Word16 *ener_avg // o: averaged quantized energy, Q10 920 // (20*log10(qua_err)) 921 ) 922 { 923 Word16 av_pred_en; 924 Word16 i; 925 926 // do average in MR122 mode (log2() domain) 927 av_pred_en = 0; 928 for (i = 0; i < NPRED; i++) 929 { 930 av_pred_en = add (av_pred_en, st->past_qua_en_MR122[i]); 931 } 932 933 // av_pred_en = 0.25*av_pred_en 934 av_pred_en = mult (av_pred_en, 8192); 935 936 // if (av_pred_en < -14/(20Log10(2))) av_pred_en = .. 937 938 if (sub (av_pred_en, MIN_ENERGY_MR122) < 0) 939 { 940 av_pred_en = MIN_ENERGY_MR122; 941 } 942 *ener_avg_MR122 = av_pred_en; 943 944 // do average for other modes (20*log10() domain) 945 av_pred_en = 0; 946 for (i = 0; i < NPRED; i++) 947 { 948 av_pred_en = add (av_pred_en, st->past_qua_en[i]); 949 } 950 951 // av_pred_en = 0.25*av_pred_en 952 av_pred_en = mult (av_pred_en, 8192); 953 954 // if (av_pred_en < -14) av_pred_en = .. 955 956 if (sub (av_pred_en, MIN_ENERGY) < 0) 957 { 958 av_pred_en = MIN_ENERGY; 959 } 960 *ener_avg = av_pred_en; 961 } 962 963 ------------------------------------------------------------------------------ 964 RESOURCES USED [optional] 965 966 When the code is written for a specific target processor the 967 the resources used should be documented below. 968 969 HEAP MEMORY USED: x bytes 970 971 STACK MEMORY USED: x bytes 972 973 CLOCK CYCLES: (cycle count equation for this function) + (variable 974 used to represent cycle count for each subroutine 975 called) 976 where: (cycle count variable) = cycle count for [subroutine 977 name] 978 979 ------------------------------------------------------------------------------ 980 CAUTION [optional] 981 [State any special notes, constraints or cautions for users of this function] 982 983 ------------------------------------------------------------------------------ 984 */ 985 986 void gc_pred_average_limited( 987 gc_predState *st, /* i: State struct */ 988 Word16 *ener_avg_MR122, /* o: everaged quantized energy, Q10 */ 989 /* (log2(qua_err)) */ 990 Word16 *ener_avg, /* o: averaged quantized energy, Q10 */ 991 /* (20*log10(qua_err)) */ 992 Flag *pOverflow 993 ) 994 { 995 Word16 av_pred_en; 996 Word16 i; 997 998 /* do average in MR122 mode (log2() domain) */ 999 av_pred_en = 0; 1000 for (i = 0; i < NPRED; i++) 1001 { 1002 av_pred_en = 1003 add(av_pred_en, st->past_qua_en_MR122[i], pOverflow); 1004 } 1005 1006 /* av_pred_en = 0.25*av_pred_en (with sign-extension)*/ 1007 if (av_pred_en < 0) 1008 { 1009 av_pred_en = (av_pred_en >> 2) | 0xc000; 1010 } 1011 else 1012 { 1013 av_pred_en >>= 2; 1014 } 1015 1016 /* if (av_pred_en < -14/(20Log10(2))) av_pred_en = .. */ 1017 if (av_pred_en < MIN_ENERGY_MR122) 1018 { 1019 av_pred_en = MIN_ENERGY_MR122; 1020 } 1021 *ener_avg_MR122 = av_pred_en; 1022 1023 /* do average for other modes (20*log10() domain) */ 1024 av_pred_en = 0; 1025 for (i = 0; i < NPRED; i++) 1026 { 1027 av_pred_en = add(av_pred_en, st->past_qua_en[i], pOverflow); 1028 } 1029 1030 /* av_pred_en = 0.25*av_pred_en (with sign-extension)*/ 1031 if (av_pred_en < 0) 1032 { 1033 av_pred_en = (av_pred_en >> 2) | 0xc000; 1034 } 1035 else 1036 { 1037 av_pred_en >>= 2; 1038 } 1039 1040 /* if (av_pred_en < -14) av_pred_en = .. */ 1041 if (av_pred_en < MIN_ENERGY) 1042 { 1043 av_pred_en = MIN_ENERGY; 1044 } 1045 *ener_avg = av_pred_en; 1046 } 1047