123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319 |
- /*************************************************************************
- *
- * GSM AMR-NB speech codec R98 Version 7.6.0 December 12, 2001
- * R99 Version 3.3.0
- * REL-4 Version 4.1.0
- *
- ********************************************************************************
- *
- * File : c_g_aver.c
- * Purpose :
- *
- ********************************************************************************
- */
- /*
- ********************************************************************************
- * MODULE INCLUDE FILE AND VERSION ID
- ********************************************************************************
- */
- #include "c_g_aver.h"
- const char c_g_aver_id[] = "@(#)$Id $" c_g_aver_h;
- #include <stdlib.h>
- #include <stdio.h>
- #include "typedef.h"
- #include "mode.h"
- #include "basic_op.h"
- #include "count.h"
- #include "cnst.h"
- #include "set_zero.h"
- /*
- ********************************************************************************
- * LOCAL VARIABLES AND TABLES
- ********************************************************************************
- */
- /*-----------------------------------------------------------------*
- * Decoder constant parameters (defined in "cnst.h") *
- *-----------------------------------------------------------------*
- * L_FRAME : Frame size. *
- * L_SUBFR : Sub-frame size. *
- *-----------------------------------------------------------------*/
- /*
- ********************************************************************************
- * PUBLIC PROGRAM CODE
- ********************************************************************************
- */
- /*
- **************************************************************************
- *
- * Function : Cb_gain_average_init
- * Purpose : Allocates and initializes state memory
- *
- **************************************************************************
- */
- Word16 Cb_gain_average_init (Cb_gain_averageState **state)
- {
- Cb_gain_averageState* s;
-
- if (state == (Cb_gain_averageState **) NULL){
- wfprintf(stderr, "Cb_gain_average_init: invalid parameter\n");
- return -1;
- }
- *state = NULL;
-
- /* allocate memory */
- if ((s= (Cb_gain_averageState *) wmalloc(sizeof(Cb_gain_averageState))) == NULL){
- wfprintf(stderr, "Cb_gain_average_init: can not malloc state structure\n");
- return -1;
- }
-
- Cb_gain_average_reset(s);
- *state = s;
-
- return 0;
- }
- /*
- **************************************************************************
- *
- * Function : Cb_gain_average_reset
- * Purpose : Resets state memory
- *
- **************************************************************************
- */
- Word16 Cb_gain_average_reset (Cb_gain_averageState *state)
- {
- if (state == (Cb_gain_averageState *) NULL){
- wfprintf(stderr, "Cb_gain_average_reset: invalid parameter\n");
- return -1;
- }
- /* Static vectors to zero */
- Set_zero (state->cbGainHistory, L_CBGAINHIST);
- /* Initialize hangover handling */
- state->hangVar = 0;
- state->hangCount= 0;
-
- return 0;
- }
- /*
- **************************************************************************
- *
- * Function : Cb_gain_average_exit
- * Purpose : The memory used for state memory is freed
- *
- **************************************************************************
- */
- void Cb_gain_average_exit (Cb_gain_averageState **state)
- {
- if (state == NULL || *state == NULL)
- return;
- /* deallocate memory */
- wfree(*state);
- *state = NULL;
-
- return;
- }
- /*
- **************************************************************************
- *
- * Function : Cb_gain_average
- * Purpose :
- * Returns : The mix cb gains for MR475, MR515, MR59, MR67, MR102; gain_code other modes
- *
- **************************************************************************
- */
- Word16 Cb_gain_average (
- Cb_gain_averageState *st, /* i/o : State variables for CB gain avergeing */
- enum Mode mode, /* i : AMR mode */
- Word16 gain_code, /* i : CB gain Q1 */
- Word16 lsp[], /* i : The LSP for the current frame Q15 */
- Word16 lspAver[], /* i : The average of LSP for 8 frames Q15 */
- Word16 bfi, /* i : bad frame indication flag */
- Word16 prev_bf, /* i : previous bad frame indication flag */
- Word16 pdfi, /* i : potential degraded bad frame ind flag */
- Word16 prev_pdf, /* i : prev pot. degraded bad frame ind flag */
- Word16 inBackgroundNoise, /* i : background noise decision */
- Word16 voicedHangover /* i : # of frames after last voiced frame */
- )
- {
- /*---------------------------------------------------------*
- * Compute mixed cb gain, used to make cb gain more *
- * smooth in background noise for modes 5.15, 5.9 and 6.7 *
- * states that needs to be updated by all *
- *---------------------------------------------------------*/
- Word16 i;
- Word16 cbGainMix, diff, tmp_diff, bgMix, cbGainMean;
- Word32 L_sum;
- Word16 tmp[M], tmp1, tmp2, shift1, shift2, shift;
- /* set correct cbGainMix for MR74, MR795, MR122 */
- cbGainMix = gain_code; move16 ();
-
- /*-------------------------------------------------------*
- * Store list of CB gain needed in the CB gain *
- * averaging *
- *-------------------------------------------------------*/
- for (i = 0; i < (L_CBGAINHIST-1); i++)
- {
- st->cbGainHistory[i] = st->cbGainHistory[i+1]; move16 ();
- }
- st->cbGainHistory[L_CBGAINHIST-1] = gain_code; move16 ();
-
- /* compute lsp difference */
- for (i = 0; i < M; i++) {
- tmp1 = abs_s_ex(sub_ex(lspAver[i], lsp[i])); /* Q15 */
- shift1 = sub_ex(norm_s_ex(tmp1), 1); /* Qn */
- tmp1 = shl_ex(tmp1, shift1); /* Q15+Qn */
- shift2 = norm_s_ex(lspAver[i]); /* Qm */
- tmp2 = shl_ex(lspAver[i], shift2); /* Q15+Qm */
- tmp[i] = div_s(tmp1, tmp2); /* Q15+(Q15+Qn)-(Q15+Qm) */
- move16 ();
- shift = sub_ex(add_ex(2, shift1), shift2);
- test ();
- if (shift >= 0)
- {
- tmp[i] = shr_ex(tmp[i], shift); move16 (); /* Q15+Qn-Qm-Qx=Q13 */
- }
- else
- {
- tmp[i] = shl_ex(tmp[i], negate_ex(shift)); move16 (); /* Q15+Qn-Qm-Qx=Q13 */
- }
- }
-
- diff = tmp[0]; move16 ();
- for (i = 1; i < M; i++) {
- diff = add_ex(diff, tmp[i]); /* Q13 */
- }
- /* Compute hangover */
- test ();
- if (sub_ex(diff, 5325) > 0) /* 0.65 in Q11 */
- {
- st->hangVar = add_ex(st->hangVar, 1);
- }
- else
- {
- st->hangVar = 0; move16 ();
- }
- test ();
- if (sub_ex(st->hangVar, 10) > 0)
- {
- st->hangCount = 0; /* Speech period, reset hangover variable */ move16 ();
- }
- /* Compute mix constant (bgMix) */
- bgMix = 8192; /* 1 in Q13 */ move16 ();
- test ();
- if ((sub_ex(mode, MR67) <= 0) || (sub_ex(mode, MR102) == 0))
- /* MR475, MR515, MR59, MR67, MR102 */
- {
- /* if errors and presumed noise make smoothing probability stronger */
- test (); test (); test (); test (); test (); test(); test (); test (); test ();
- if (((((pdfi != 0) && (prev_pdf != 0)) || (bfi != 0) || (prev_bf != 0)) &&
- (sub_ex(voicedHangover, 1) > 0) && (inBackgroundNoise != 0) &&
- ((sub_ex(mode, MR475) == 0) ||
- (sub_ex(mode, MR515) == 0) ||
- (sub_ex(mode, MR59) == 0)) ))
- {
- /* bgMix = min(0.25, max(0.0, diff-0.55)) / 0.25; */
- tmp_diff = sub_ex(diff, 4506); /* 0.55 in Q13 */
- /* max(0.0, diff-0.55) */
- test ();
- if (tmp_diff > 0)
- {
- tmp1 = tmp_diff; move16 ();
- }
- else
- {
- tmp1 = 0; move16 ();
- }
- /* min(0.25, tmp1) */
- test ();
- if (sub_ex(2048, tmp1) < 0)
- {
- bgMix = 8192; move16 ();
- }
- else
- {
- bgMix = shl_ex(tmp1, 2);
- }
- }
- else
- {
- /* bgMix = min(0.25, max(0.0, diff-0.40)) / 0.25; */
- tmp_diff = sub_ex(diff, 3277); /* 0.4 in Q13 */
- /* max(0.0, diff-0.40) */
- test ();
- if (tmp_diff > 0)
- {
- tmp1 = tmp_diff; move16 ();
- }
- else
- {
- tmp1 = 0; move16 ();
- }
- /* min(0.25, tmp1) */
- test ();
- if (sub_ex(2048, tmp1) < 0)
- {
- bgMix = 8192; move16 ();
- }
- else
- {
- bgMix = shl_ex(tmp1, 2);
- }
- }
- test (); test ();
- if ((sub_ex(st->hangCount, 40) < 0) || (sub_ex(diff, 5325) > 0)) /* 0.65 in Q13 */
- {
- bgMix = 8192; /* disable mix if too short time since */ move16 ();
- }
- /* Smoothen the cb gain trajectory */
- /* smoothing depends on mix constant bgMix */
- L_sum = L_mult_ex(6554, st->cbGainHistory[2]); /* 0.2 in Q15; L_sum in Q17 */
- for (i = 3; i < L_CBGAINHIST; i++)
- {
- L_sum = L_mac_ex(L_sum, 6554, st->cbGainHistory[i]);
- }
- cbGainMean = round_ex(L_sum); /* Q1 */
-
- /* more smoothing in error and bg noise (NB no DFI used here) */
- test (); test (); test (); test (); test(); test();
- if (((bfi != 0) || (prev_bf != 0)) && (inBackgroundNoise != 0) &&
- ((sub_ex(mode, MR475) == 0) ||
- (sub_ex(mode, MR515) == 0) ||
- (sub_ex(mode, MR59) == 0)) )
- {
- L_sum = L_mult_ex(4681, st->cbGainHistory[0]); /* 0.143 in Q15; L_sum in Q17 */
- for (i = 1; i < L_CBGAINHIST; i++)
- {
- L_sum = L_mac_ex(L_sum, 4681, st->cbGainHistory[i]);
- }
- cbGainMean = round_ex(L_sum); /* Q1 */
- }
-
- /* cbGainMix = bgMix*cbGainMix + (1-bgMix)*cbGainMean; */
- L_sum = L_mult_ex(bgMix, cbGainMix); /* L_sum in Q15 */
- L_sum = L_mac_ex(L_sum, 8192, cbGainMean);
- L_sum = L_msu_ex(L_sum, bgMix, cbGainMean);
- cbGainMix = round_ex(L_shl_ex(L_sum, 2)); /* Q1 */
- }
-
- st->hangCount = add_ex(st->hangCount, 1);
- return cbGainMix;
- }
|