diff options
author | Reynaldo H. Verdejo Pinochet <reynaldo@opendot.cl> | 2009-11-04 19:29:29 +0000 |
---|---|---|
committer | Reynaldo H. Verdejo Pinochet <reynaldo@opendot.cl> | 2009-11-04 19:29:29 +0000 |
commit | 0c50f8e6ccda2ff0b265d36412fae103b276af2e (patch) | |
tree | b9f3d071e6c48f011ecc1ba3229b71ceeb60d95a /libavcodec | |
parent | 223217746c5beaa6110ce9ade2a582fc10d3d630 (diff) | |
download | ffmpeg-0c50f8e6ccda2ff0b265d36412fae103b276af2e.tar.gz |
Implement ff_scale_vector_to_given_sum_of_squares()
to aid generic gain control routines. Changes
for qcelp are included. Patch Collin McQuillan.
Originally committed as revision 20450 to svn://svn.ffmpeg.org/ffmpeg/trunk
Diffstat (limited to 'libavcodec')
-rw-r--r-- | libavcodec/acelp_vectors.c | 12 | ||||
-rw-r--r-- | libavcodec/acelp_vectors.h | 18 | ||||
-rw-r--r-- | libavcodec/qcelpdec.c | 39 |
3 files changed, 36 insertions, 33 deletions
diff --git a/libavcodec/acelp_vectors.c b/libavcodec/acelp_vectors.c index 2d9aa1ad79..e6979205b9 100644 --- a/libavcodec/acelp_vectors.c +++ b/libavcodec/acelp_vectors.c @@ -22,6 +22,7 @@ #include <inttypes.h> #include "avcodec.h" +#include "celp_math.h" #include "acelp_vectors.h" #include "celp_math.h" @@ -177,3 +178,14 @@ void ff_adaptative_gain_control(float *buf_out, float speech_energ, *gain_mem = mem; } + +void ff_scale_vector_to_given_sum_of_squares(float *out, const float *in, + float sum_of_squares, const int n) +{ + int i; + float scalefactor = ff_dot_productf(in, in, n); + if (scalefactor) + scalefactor = sqrt(sum_of_squares / scalefactor); + for (i = 0; i < n; i++) + out[i] = in[i] * scalefactor; +} diff --git a/libavcodec/acelp_vectors.h b/libavcodec/acelp_vectors.h index 58cd84c841..a711160d4c 100644 --- a/libavcodec/acelp_vectors.h +++ b/libavcodec/acelp_vectors.h @@ -176,4 +176,22 @@ void ff_weighted_vector_sumf(float *out, const float *in_a, const float *in_b, void ff_adaptative_gain_control(float *buf_out, float speech_energ, int size, float alpha, float *gain_mem); +/** + * Set the sum of squares of a signal by scaling + * + * @param out output samples + * @param in input samples + * @param sum_of_squares new sum of squares + * @param n number of samples + * + * @note If the input is zero (or its energy underflows), the output is zero. + * This is the behavior of AGC in the AMR reference decoder. The QCELP + * reference decoder seems to have undefined behavior. + * + * TIA/EIA/IS-733 2.4.8.3-2/3/4/5, 2.4.8.6 + * 3GPP TS 26.090 6.1 (6) + */ +void ff_scale_vector_to_given_sum_of_squares(float *out, const float *in, + float sum_of_squares, const int n); + #endif /* AVCODEC_ACELP_VECTORS_H */ diff --git a/libavcodec/qcelpdec.c b/libavcodec/qcelpdec.c index 0df30daf34..d655529c16 100644 --- a/libavcodec/qcelpdec.c +++ b/libavcodec/qcelpdec.c @@ -406,31 +406,6 @@ static void compute_svector(QCELPContext *q, const float *gain, } /** - * Compute the gain control - * - * @param v_in gain-controlled vector - * @param v_ref vector to control gain of - * - * @return gain control - * - * FIXME: If v_ref is a zero vector, it energy is zero - * and the behavior of the gain control is - * undefined in the specs. - * - * TIA/EIA/IS-733 2.4.8.3-2/3/4/5, 2.4.8.6 - */ -static float compute_gain_ctrl(const float *v_ref, const float *v_in, const int len) -{ - float scalefactor = ff_dot_productf(v_in, v_in, len); - - if(scalefactor) - scalefactor = sqrt(ff_dot_productf(v_ref, v_ref, len) / scalefactor); - else - av_log_missing_feature(NULL, "Zero energy for gain control", 1); - return scalefactor; -} - -/** * Apply generic gain control. * * @param v_out output vector @@ -442,15 +417,13 @@ static float compute_gain_ctrl(const float *v_ref, const float *v_in, const int static void apply_gain_ctrl(float *v_out, const float *v_ref, const float *v_in) { - int i, j, len; - float scalefactor; + int i; - for(i=0, j=0; i<4; i++) - { - scalefactor = compute_gain_ctrl(v_ref + j, v_in + j, 40); - for(len=j+40; j<len; j++) - v_out[j] = scalefactor * v_in[j]; - } + for (i = 0; i < 160; i += 40) + ff_scale_vector_to_given_sum_of_squares(v_out + i, v_in + i, + ff_dot_productf(v_ref + i, + v_ref + i, 40), + 40); } /** |