2019-05-28 09:57:05 -07:00
// SPDX-License-Identifier: GPL-2.0-only
2008-10-06 21:41:46 -07:00
/*
* SpanDSP - a series of DSP components for telephony
*
* echo . c - A line echo canceller . This code is being developed
* against and partially complies with G168 .
*
* Written by Steve Underwood < steveu @ coppice . org >
* and David Rowe < david_at_rowetel_dot_com >
*
* Copyright ( C ) 2001 , 2003 Steve Underwood , 2007 David Rowe
*
* Based on a bit from here , a bit from there , eye of toad , ear of
* bat , 15 years of failed attempts by David and a few fried brain
* cells .
*
* All rights reserved .
*/
/*! \file */
/* Implementation Notes
David Rowe
April 2007
This code started life as Steve ' s NLMS algorithm with a tap
rotation algorithm to handle divergence during double talk . I
added a Geigel Double Talk Detector ( DTD ) [ 2 ] and performed some
G168 tests . However I had trouble meeting the G168 requirements ,
especially for double talk - there were always cases where my DTD
failed , for example where near end speech was under the 6 dB
threshold required for declaring double talk .
So I tried a two path algorithm [ 1 ] , which has so far given better
results . The original tap rotation / Geigel algorithm is available
in SVN http : //svn.rowetel.com/software/oslec/tags/before_16bit.
It ' s probably possible to make it work if some one wants to put some
serious work into it .
At present no special treatment is provided for tones , which
generally cause NLMS algorithms to diverge . Initial runs of a
subset of the G168 tests for tones ( e . g . / echo_test 6 ) show the
current algorithm is passing OK , which is kind of surprising . The
full set of tests needs to be performed to confirm this result .
One other interesting change is that I have managed to get the NLMS
code to work with 16 bit coefficients , rather than the original 32
bit coefficents . This reduces the MIPs and storage required .
I evaulated the 16 bit port using g168_tests . sh and listening tests
on 4 real - world samples .
I also attempted the implementation of a block based NLMS update
[ 2 ] but although this passes g168_tests . sh it didn ' t converge well
on the real - world samples . I have no idea why , perhaps a scaling
problem . The block based code is also available in SVN
http : //svn.rowetel.com/software/oslec/tags/before_16bit. If this
code can be debugged , it will lead to further reduction in MIPS , as
the block update code maps nicely onto DSP instruction sets ( it ' s a
dot product ) compared to the current sample - by - sample update .
Steve also has some nice notes on echo cancellers in echo . h
References :
[ 1 ] Ochiai , Areseki , and Ogihara , " Echo Canceller with Two Echo
Path Models " , IEEE Transactions on communications, COM-25,
No . 6 , June
1977.
2020-07-13 12:44:53 +02:00
https : //www.rowetel.com/images/echo/dual_path_paper.pdf
2008-10-06 21:41:46 -07:00
[ 2 ] The classic , very useful paper that tells you how to
actually build a real world echo canceller :
2009-08-10 10:45:25 -07:00
Messerschmitt , Hedberg , Cole , Haoui , Winship , " Digital Voice
Echo Canceller with a TMS320020 ,
2020-07-13 12:44:53 +02:00
https : //www.rowetel.com/images/echo/spra129.pdf
2008-10-06 21:41:46 -07:00
[ 3 ] I have written a series of blog posts on this work , here is
Part 1 : http : //www.rowetel.com/blog/?p=18
[ 4 ] The source code http : //svn.rowetel.com/software/oslec/
[ 5 ] A nice reference on LMS filters :
2020-07-13 12:44:53 +02:00
https : //en.wikipedia.org/wiki/Least_mean_squares_filter
2008-10-06 21:41:46 -07:00
Credits :
Thanks to Steve Underwood , Jean - Marc Valin , and Ramakrishnan
Muthukrishnan for their suggestions and email discussions . Thanks
also to those people who collected echo samples for me such as
Mark , Pawel , and Pavel .
*/
2009-08-10 10:45:25 -07:00
# include <linux/kernel.h>
2008-10-06 21:41:46 -07:00
# include <linux/module.h>
# include <linux/slab.h>
# include "echo.h"
2009-08-10 10:45:25 -07:00
# define MIN_TX_POWER_FOR_ADAPTION 64
# define MIN_RX_POWER_FOR_ADAPTION 64
# define DTD_HANGOVER 600 /* 600 samples, or 75ms */
# define DC_LOG2BETA 3 /* log2() of DC filter Beta */
2008-10-06 21:41:46 -07:00
/* adapting coeffs using the traditional stochastic descent (N)LMS algorithm */
2011-05-14 11:31:42 +05:30
static inline void lms_adapt_bg ( struct oslec_state * ec , int clean , int shift )
2008-10-06 21:41:46 -07:00
{
2008-10-20 19:01:31 -04:00
int i ;
int offset1 ;
int offset2 ;
int factor ;
int exp ;
if ( shift > 0 )
factor = clean < < shift ;
else
factor = clean > > - shift ;
/* Update the FIR taps */
offset2 = ec - > curr_pos ;
offset1 = ec - > taps - offset2 ;
for ( i = ec - > taps - 1 ; i > = offset1 ; i - - ) {
exp = ( ec - > fir_state_bg . history [ i - offset1 ] * factor ) ;
ec - > fir_taps16 [ 1 ] [ i ] + = ( int16_t ) ( ( exp + ( 1 < < 14 ) ) > > 15 ) ;
}
for ( ; i > = 0 ; i - - ) {
exp = ( ec - > fir_state_bg . history [ i + offset2 ] * factor ) ;
ec - > fir_taps16 [ 1 ] [ i ] + = ( int16_t ) ( ( exp + ( 1 < < 14 ) ) > > 15 ) ;
}
2008-10-06 21:41:46 -07:00
}
2009-08-25 22:07:56 -07:00
static inline int top_bit ( unsigned int bits )
2009-08-23 10:57:53 +09:30
{
if ( bits = = 0 )
2009-08-25 22:07:56 -07:00
return - 1 ;
else
2011-05-14 11:31:42 +05:30
return ( int ) fls ( ( int32_t ) bits ) - 1 ;
2009-08-23 10:57:53 +09:30
}
2008-10-12 07:17:26 +02:00
struct oslec_state * oslec_create ( int len , int adaption_mode )
2008-10-06 21:41:46 -07:00
{
2008-10-20 19:01:31 -04:00
struct oslec_state * ec ;
int i ;
2012-12-22 17:12:26 +01:00
const int16_t * history ;
2008-10-20 19:01:31 -04:00
ec = kzalloc ( sizeof ( * ec ) , GFP_KERNEL ) ;
if ( ! ec )
return NULL ;
ec - > taps = len ;
ec - > log2taps = top_bit ( len ) ;
ec - > curr_pos = ec - > taps - 1 ;
2012-12-22 17:12:26 +01:00
ec - > fir_taps16 [ 0 ] =
kcalloc ( ec - > taps , sizeof ( int16_t ) , GFP_KERNEL ) ;
if ( ! ec - > fir_taps16 [ 0 ] )
goto error_oom_0 ;
ec - > fir_taps16 [ 1 ] =
kcalloc ( ec - > taps , sizeof ( int16_t ) , GFP_KERNEL ) ;
if ( ! ec - > fir_taps16 [ 1 ] )
goto error_oom_1 ;
2008-10-20 19:01:31 -04:00
2012-12-22 17:12:26 +01:00
history = fir16_create ( & ec - > fir_state , ec - > fir_taps16 [ 0 ] , ec - > taps ) ;
if ( ! history )
goto error_state ;
history = fir16_create ( & ec - > fir_state_bg , ec - > fir_taps16 [ 1 ] , ec - > taps ) ;
if ( ! history )
goto error_state_bg ;
2008-10-20 19:01:31 -04:00
2009-03-12 03:32:45 +03:00
for ( i = 0 ; i < 5 ; i + + )
2008-10-20 19:01:31 -04:00
ec - > xvtx [ i ] = ec - > yvtx [ i ] = ec - > xvrx [ i ] = ec - > yvrx [ i ] = 0 ;
ec - > cng_level = 1000 ;
oslec_adaption_mode ( ec , adaption_mode ) ;
ec - > snapshot = kcalloc ( ec - > taps , sizeof ( int16_t ) , GFP_KERNEL ) ;
if ( ! ec - > snapshot )
2012-12-22 17:12:26 +01:00
goto error_snap ;
2008-10-20 19:01:31 -04:00
ec - > cond_met = 0 ;
2013-05-05 23:38:24 -07:00
ec - > pstates = 0 ;
ec - > ltxacc = ec - > lrxacc = ec - > lcleanacc = ec - > lclean_bgacc = 0 ;
ec - > ltx = ec - > lrx = ec - > lclean = ec - > lclean_bg = 0 ;
2008-10-20 19:01:31 -04:00
ec - > tx_1 = ec - > tx_2 = ec - > rx_1 = ec - > rx_2 = 0 ;
2013-05-05 23:38:24 -07:00
ec - > lbgn = ec - > lbgn_acc = 0 ;
ec - > lbgn_upper = 200 ;
ec - > lbgn_upper_acc = ec - > lbgn_upper < < 13 ;
2008-10-20 19:01:31 -04:00
return ec ;
2012-12-22 17:12:26 +01:00
error_snap :
fir16_free ( & ec - > fir_state_bg ) ;
error_state_bg :
fir16_free ( & ec - > fir_state ) ;
error_state :
kfree ( ec - > fir_taps16 [ 1 ] ) ;
error_oom_1 :
kfree ( ec - > fir_taps16 [ 0 ] ) ;
error_oom_0 :
2008-10-20 19:01:31 -04:00
kfree ( ec ) ;
return NULL ;
2008-10-06 21:41:46 -07:00
}
2008-10-12 07:17:26 +02:00
EXPORT_SYMBOL_GPL ( oslec_create ) ;
2008-10-06 21:41:46 -07:00
2008-10-12 07:17:26 +02:00
void oslec_free ( struct oslec_state * ec )
2008-10-06 21:41:46 -07:00
{
int i ;
fir16_free ( & ec - > fir_state ) ;
fir16_free ( & ec - > fir_state_bg ) ;
2008-10-20 19:01:31 -04:00
for ( i = 0 ; i < 2 ; i + + )
2008-10-06 21:41:46 -07:00
kfree ( ec - > fir_taps16 [ i ] ) ;
kfree ( ec - > snapshot ) ;
kfree ( ec ) ;
}
2008-10-12 07:17:26 +02:00
EXPORT_SYMBOL_GPL ( oslec_free ) ;
2008-10-06 21:41:46 -07:00
2008-10-12 07:17:26 +02:00
void oslec_adaption_mode ( struct oslec_state * ec , int adaption_mode )
2008-10-06 21:41:46 -07:00
{
2008-10-20 19:01:31 -04:00
ec - > adaption_mode = adaption_mode ;
2008-10-06 21:41:46 -07:00
}
2008-10-12 07:17:26 +02:00
EXPORT_SYMBOL_GPL ( oslec_adaption_mode ) ;
2008-10-06 21:41:46 -07:00
2008-10-12 07:17:26 +02:00
void oslec_flush ( struct oslec_state * ec )
2008-10-06 21:41:46 -07:00
{
2008-10-20 19:01:31 -04:00
int i ;
2008-10-06 21:41:46 -07:00
2013-05-05 23:38:24 -07:00
ec - > ltxacc = ec - > lrxacc = ec - > lcleanacc = ec - > lclean_bgacc = 0 ;
ec - > ltx = ec - > lrx = ec - > lclean = ec - > lclean_bg = 0 ;
2008-10-20 19:01:31 -04:00
ec - > tx_1 = ec - > tx_2 = ec - > rx_1 = ec - > rx_2 = 0 ;
2008-10-06 21:41:46 -07:00
2013-05-05 23:38:24 -07:00
ec - > lbgn = ec - > lbgn_acc = 0 ;
ec - > lbgn_upper = 200 ;
ec - > lbgn_upper_acc = ec - > lbgn_upper < < 13 ;
2008-10-06 21:41:46 -07:00
2008-10-20 19:01:31 -04:00
ec - > nonupdate_dwell = 0 ;
2008-10-06 21:41:46 -07:00
2008-10-20 19:01:31 -04:00
fir16_flush ( & ec - > fir_state ) ;
fir16_flush ( & ec - > fir_state_bg ) ;
ec - > fir_state . curr_pos = ec - > taps - 1 ;
ec - > fir_state_bg . curr_pos = ec - > taps - 1 ;
for ( i = 0 ; i < 2 ; i + + )
memset ( ec - > fir_taps16 [ i ] , 0 , ec - > taps * sizeof ( int16_t ) ) ;
2008-10-06 21:41:46 -07:00
2008-10-20 19:01:31 -04:00
ec - > curr_pos = ec - > taps - 1 ;
2013-05-05 23:38:24 -07:00
ec - > pstates = 0 ;
2008-10-06 21:41:46 -07:00
}
2008-10-12 07:17:26 +02:00
EXPORT_SYMBOL_GPL ( oslec_flush ) ;
2008-10-06 21:41:46 -07:00
2008-10-20 19:01:31 -04:00
void oslec_snapshot ( struct oslec_state * ec )
{
memcpy ( ec - > snapshot , ec - > fir_taps16 [ 0 ] , ec - > taps * sizeof ( int16_t ) ) ;
2008-10-06 21:41:46 -07:00
}
2008-10-12 07:17:26 +02:00
EXPORT_SYMBOL_GPL ( oslec_snapshot ) ;
2008-10-06 21:41:46 -07:00
2009-08-10 10:45:25 -07:00
/* Dual Path Echo Canceller */
2008-10-06 21:41:46 -07:00
2008-10-12 07:17:26 +02:00
int16_t oslec_update ( struct oslec_state * ec , int16_t tx , int16_t rx )
2008-10-06 21:41:46 -07:00
{
2008-10-20 19:01:31 -04:00
int32_t echo_value ;
int clean_bg ;
2012-06-27 22:28:55 +02:00
int tmp ;
int tmp1 ;
2008-10-20 19:01:31 -04:00
2009-08-10 10:45:25 -07:00
/*
* Input scaling was found be required to prevent problems when tx
* starts clipping . Another possible way to handle this would be the
* filter coefficent scaling .
*/
2008-10-20 19:01:31 -04:00
ec - > tx = tx ;
ec - > rx = rx ;
tx > > = 1 ;
rx > > = 1 ;
/*
2009-08-10 10:45:25 -07:00
* Filter DC , 3 dB point is 160 Hz ( I think ) , note 32 bit precision
* required otherwise values do not track down to 0. Zero at DC , Pole
2009-08-23 10:57:53 +09:30
* at ( 1 - Beta ) on real axis . Some chip sets ( like Si labs ) don ' t
2009-08-10 10:45:25 -07:00
* need this , but something like a $ 10 X100P card does . Any DC really
* slows down convergence .
*
* Note : removes some low frequency from the signal , this reduces the
* speech quality when listening to samples through headphones but may
* not be obvious through a telephone handset .
*
* Note that the 3 dB frequency in radians is approx Beta , e . g . for Beta
* = 2 ^ ( - 3 ) = 0.125 , 3 dB freq is 0.125 rads = 159 Hz .
2008-10-20 19:01:31 -04:00
*/
if ( ec - > adaption_mode & ECHO_CAN_USE_RX_HPF ) {
tmp = rx < < 15 ;
2009-08-23 10:57:53 +09:30
2009-08-10 10:45:25 -07:00
/*
* Make sure the gain of the HPF is 1.0 . This can still
* saturate a little under impulse conditions , and it might
* roll to 32768 and need clipping on sustained peak level
* signals . However , the scale of such clipping is small , and
* the error due to any saturation should not markedly affect
* the downstream processing .
*/
2008-10-20 19:01:31 -04:00
tmp - = ( tmp > > 4 ) ;
2009-08-23 10:57:53 +09:30
2008-10-20 19:01:31 -04:00
ec - > rx_1 + = - ( ec - > rx_1 > > DC_LOG2BETA ) + tmp - ec - > rx_2 ;
2009-08-10 10:45:25 -07:00
/*
* hard limit filter to prevent clipping . Note that at this
* stage rx should be limited to + / - 16383 due to right shift
* above
*/
2008-10-20 19:01:31 -04:00
tmp1 = ec - > rx_1 > > 15 ;
if ( tmp1 > 16383 )
tmp1 = 16383 ;
if ( tmp1 < - 16383 )
tmp1 = - 16383 ;
rx = tmp1 ;
ec - > rx_2 = tmp ;
}
2008-10-06 21:41:46 -07:00
2008-10-20 19:01:31 -04:00
/* Block average of power in the filter states. Used for
adaption power calculation . */
2008-10-06 21:41:46 -07:00
2008-10-20 19:01:31 -04:00
{
int new , old ;
/* efficient "out with the old and in with the new" algorithm so
we don ' t have to recalculate over the whole block of
samples . */
2011-07-01 21:55:38 +12:00
new = ( int ) tx * ( int ) tx ;
2008-10-20 19:01:31 -04:00
old = ( int ) ec - > fir_state . history [ ec - > fir_state . curr_pos ] *
( int ) ec - > fir_state . history [ ec - > fir_state . curr_pos ] ;
2013-05-05 23:38:24 -07:00
ec - > pstates + =
2011-05-14 11:31:42 +05:30
( ( new - old ) + ( 1 < < ( ec - > log2taps - 1 ) ) ) > > ec - > log2taps ;
2013-05-05 23:38:24 -07:00
if ( ec - > pstates < 0 )
ec - > pstates = 0 ;
2008-10-20 19:01:31 -04:00
}
2008-10-06 21:41:46 -07:00
2008-10-20 19:01:31 -04:00
/* Calculate short term average levels using simple single pole IIRs */
2008-10-06 21:41:46 -07:00
2013-05-05 23:38:24 -07:00
ec - > ltxacc + = abs ( tx ) - ec - > ltx ;
ec - > ltx = ( ec - > ltxacc + ( 1 < < 4 ) ) > > 5 ;
ec - > lrxacc + = abs ( rx ) - ec - > lrx ;
ec - > lrx = ( ec - > lrxacc + ( 1 < < 4 ) ) > > 5 ;
2008-10-06 21:41:46 -07:00
2009-08-10 10:45:25 -07:00
/* Foreground filter */
2008-10-06 21:41:46 -07:00
2008-10-20 19:01:31 -04:00
ec - > fir_state . coeffs = ec - > fir_taps16 [ 0 ] ;
echo_value = fir16 ( & ec - > fir_state , tx ) ;
ec - > clean = rx - echo_value ;
2013-05-05 23:38:24 -07:00
ec - > lcleanacc + = abs ( ec - > clean ) - ec - > lclean ;
ec - > lclean = ( ec - > lcleanacc + ( 1 < < 4 ) ) > > 5 ;
2008-10-06 21:41:46 -07:00
2009-08-10 10:45:25 -07:00
/* Background filter */
2008-10-06 21:41:46 -07:00
2008-10-20 19:01:31 -04:00
echo_value = fir16 ( & ec - > fir_state_bg , tx ) ;
clean_bg = rx - echo_value ;
2013-05-05 23:38:24 -07:00
ec - > lclean_bgacc + = abs ( clean_bg ) - ec - > lclean_bg ;
ec - > lclean_bg = ( ec - > lclean_bgacc + ( 1 < < 4 ) ) > > 5 ;
2008-10-06 21:41:46 -07:00
2009-08-10 10:45:25 -07:00
/* Background Filter adaption */
2008-10-06 21:41:46 -07:00
2008-10-20 19:01:31 -04:00
/* Almost always adap bg filter, just simple DT and energy
detection to minimise adaption in cases of strong double talk .
However this is not critical for the dual path algorithm .
*/
ec - > factor = 0 ;
ec - > shift = 0 ;
2018-09-14 23:43:37 -07:00
if ( ! ec - > nonupdate_dwell ) {
2013-05-05 23:38:24 -07:00
int p , logp , shift ;
2008-10-20 19:01:31 -04:00
/* Determine:
f = Beta * clean_bg_rx / P - - - - - - ( 1 )
where P is the total power in the filter states .
The Boffins have shown that if we obey ( 1 ) we converge
quickly and avoid instability .
The correct factor f must be in Q30 , as this is the fixed
point format required by the lms_adapt_bg ( ) function ,
therefore the scaled version of ( 1 ) is :
( 2 ^ 30 ) * f = ( 2 ^ 30 ) * Beta * clean_bg_rx / P
2009-08-23 10:57:53 +09:30
factor = ( 2 ^ 30 ) * Beta * clean_bg_rx / P - - - - - ( 2 )
2008-10-20 19:01:31 -04:00
We have chosen Beta = 0.25 by experiment , so :
2009-08-23 10:57:53 +09:30
factor = ( 2 ^ 30 ) * ( 2 ^ - 2 ) * clean_bg_rx / P
2008-10-20 19:01:31 -04:00
2011-05-14 11:31:42 +05:30
( 30 - 2 - log2 ( P ) )
2009-08-23 10:57:53 +09:30
factor = clean_bg_rx 2 - - - - - ( 3 )
2008-10-20 19:01:31 -04:00
To avoid a divide we approximate log2 ( P ) as top_bit ( P ) ,
which returns the position of the highest non - zero bit in
P . This approximation introduces an error as large as a
factor of 2 , but the algorithm seems to handle it OK .
Come to think of it a divide may not be a big deal on a
modern DSP , so its probably worth checking out the cycles
for a divide versus a top_bit ( ) implementation .
*/
2013-05-05 23:38:24 -07:00
p = MIN_TX_POWER_FOR_ADAPTION + ec - > pstates ;
logp = top_bit ( p ) + ec - > log2taps ;
shift = 30 - 2 - logp ;
2008-10-20 19:01:31 -04:00
ec - > shift = shift ;
lms_adapt_bg ( ec , clean_bg , shift ) ;
2008-10-06 21:41:46 -07:00
}
2008-10-20 19:01:31 -04:00
/* very simple DTD to make sure we dont try and adapt with strong
near end speech */
ec - > adapt = 0 ;
2013-05-05 23:38:24 -07:00
if ( ( ec - > lrx > MIN_RX_POWER_FOR_ADAPTION ) & & ( ec - > lrx > ec - > ltx ) )
2008-10-20 19:01:31 -04:00
ec - > nonupdate_dwell = DTD_HANGOVER ;
if ( ec - > nonupdate_dwell )
ec - > nonupdate_dwell - - ;
2009-08-10 10:45:25 -07:00
/* Transfer logic */
2008-10-20 19:01:31 -04:00
/* These conditions are from the dual path paper [1], I messed with
them a bit to improve performance . */
if ( ( ec - > adaption_mode & ECHO_CAN_USE_ADAPTION ) & &
( ec - > nonupdate_dwell = = 0 ) & &
2009-03-12 03:32:45 +03:00
/* (ec->Lclean_bg < 0.875*ec->Lclean) */
2013-05-05 23:38:24 -07:00
( 8 * ec - > lclean_bg < 7 * ec - > lclean ) & &
2009-03-12 03:32:45 +03:00
/* (ec->Lclean_bg < 0.125*ec->Ltx) */
2013-05-05 23:38:24 -07:00
( 8 * ec - > lclean_bg < ec - > ltx ) ) {
2008-10-20 19:01:31 -04:00
if ( ec - > cond_met = = 6 ) {
2009-08-10 10:45:25 -07:00
/*
* BG filter has had better results for 6 consecutive
* samples
*/
2008-10-20 19:01:31 -04:00
ec - > adapt = 1 ;
memcpy ( ec - > fir_taps16 [ 0 ] , ec - > fir_taps16 [ 1 ] ,
2011-05-14 11:31:42 +05:30
ec - > taps * sizeof ( int16_t ) ) ;
2008-10-20 19:01:31 -04:00
} else
ec - > cond_met + + ;
} else
ec - > cond_met = 0 ;
2009-08-10 10:45:25 -07:00
/* Non-Linear Processing */
2008-10-20 19:01:31 -04:00
ec - > clean_nlp = ec - > clean ;
if ( ec - > adaption_mode & ECHO_CAN_USE_NLP ) {
2009-08-10 10:45:25 -07:00
/*
* Non - linear processor - a fancy way to say " zap small
* signals , to avoid residual echo due to ( uLaw / ALaw )
* non - linearity in the channel . " .
*/
2008-10-20 19:01:31 -04:00
2013-05-05 23:38:24 -07:00
if ( ( 16 * ec - > lclean < ec - > ltx ) ) {
2009-08-10 10:45:25 -07:00
/*
* Our e / c has improved echo by at least 24 dB ( each
* factor of 2 is 6 dB , so 2 * 2 * 2 * 2 = 16 is the same as
* 6 + 6 + 6 + 6 = 24 dB )
*/
2008-10-20 19:01:31 -04:00
if ( ec - > adaption_mode & ECHO_CAN_USE_CNG ) {
2013-05-05 23:38:24 -07:00
ec - > cng_level = ec - > lbgn ;
2008-10-20 19:01:31 -04:00
2009-08-10 10:45:25 -07:00
/*
* Very elementary comfort noise generation .
* Just random numbers rolled off very vaguely
* Hoth - like . DR : This noise doesn ' t sound
* quite right to me - I suspect there are some
2011-03-01 23:58:54 +01:00
* overflow issues in the filtering as it ' s too
2009-08-10 10:45:25 -07:00
* " crackly " .
* TODO : debug this , maybe just play noise at
* high level or look at spectrum .
2008-10-20 19:01:31 -04:00
*/
ec - > cng_rndnum =
1664525U * ec - > cng_rndnum + 1013904223U ;
ec - > cng_filter =
( ( ec - > cng_rndnum & 0xFFFF ) - 32768 +
5 * ec - > cng_filter ) > > 3 ;
ec - > clean_nlp =
( ec - > cng_filter * ec - > cng_level * 8 ) > > 14 ;
} else if ( ec - > adaption_mode & ECHO_CAN_USE_CLIP ) {
/* This sounds much better than CNG */
2013-05-05 23:38:24 -07:00
if ( ec - > clean_nlp > ec - > lbgn )
ec - > clean_nlp = ec - > lbgn ;
if ( ec - > clean_nlp < - ec - > lbgn )
ec - > clean_nlp = - ec - > lbgn ;
2008-10-20 19:01:31 -04:00
} else {
2009-08-10 10:45:25 -07:00
/*
* just mute the residual , doesn ' t sound very
* good , used mainly in G168 tests
*/
2008-10-20 19:01:31 -04:00
ec - > clean_nlp = 0 ;
}
} else {
2009-08-10 10:45:25 -07:00
/*
* Background noise estimator . I tried a few
* algorithms here without much luck . This very simple
* one seems to work best , we just average the level
* using a slow ( 1 sec time const ) filter if the
* current level is less than a ( experimentally
* derived ) constant . This means we dont include high
* level signals like near end speech . When combined
* with CNG or especially CLIP seems to work OK .
2008-10-20 19:01:31 -04:00
*/
2013-05-05 23:38:24 -07:00
if ( ec - > lclean < 40 ) {
ec - > lbgn_acc + = abs ( ec - > clean ) - ec - > lbgn ;
ec - > lbgn = ( ec - > lbgn_acc + ( 1 < < 11 ) ) > > 12 ;
2008-10-20 19:01:31 -04:00
}
}
}
/* Roll around the taps buffer */
if ( ec - > curr_pos < = 0 )
ec - > curr_pos = ec - > taps ;
ec - > curr_pos - - ;
if ( ec - > adaption_mode & ECHO_CAN_DISABLE )
ec - > clean_nlp = rx ;
/* Output scaled back up again to match input scaling */
return ( int16_t ) ec - > clean_nlp < < 1 ;
2008-10-06 21:41:46 -07:00
}
2008-10-12 07:17:26 +02:00
EXPORT_SYMBOL_GPL ( oslec_update ) ;
2008-10-06 21:41:46 -07:00
2010-05-12 13:03:13 +05:30
/* This function is separated from the echo canceller is it is usually called
2008-10-06 21:41:46 -07:00
as part of the tx process . See rx HP ( DC blocking ) filter above , it ' s
the same design .
Some soft phones send speech signals with a lot of low frequency
energy , e . g . down to 20 Hz . This can make the hybrid non - linear
which causes the echo canceller to fall over . This filter can help
by removing any low frequency before it gets to the tx port of the
hybrid .
It can also help by removing and DC in the tx signal . DC is bad
for LMS algorithms .
2009-08-10 10:45:25 -07:00
This is one of the classic DC removal filters , adjusted to provide
sufficient bass rolloff to meet the above requirement to protect hybrids
from things that upset them . The difference between successive samples
produces a lousy HPF , and then a suitably placed pole flattens things out .
The final result is a nicely rolled off bass end . The filtering is
implemented with extended fractional precision , which noise shapes things ,
giving very clean DC removal .
2008-10-06 21:41:46 -07:00
*/
2011-07-01 21:55:38 +12:00
int16_t oslec_hpf_tx ( struct oslec_state * ec , int16_t tx )
2008-10-20 19:01:31 -04:00
{
2012-06-27 22:28:55 +02:00
int tmp ;
int tmp1 ;
2008-10-06 21:41:46 -07:00
2008-10-20 19:01:31 -04:00
if ( ec - > adaption_mode & ECHO_CAN_USE_TX_HPF ) {
tmp = tx < < 15 ;
2009-08-23 10:57:53 +09:30
2009-08-10 10:45:25 -07:00
/*
* Make sure the gain of the HPF is 1.0 . The first can still
* saturate a little under impulse conditions , and it might
* roll to 32768 and need clipping on sustained peak level
* signals . However , the scale of such clipping is small , and
* the error due to any saturation should not markedly affect
* the downstream processing .
*/
2008-10-20 19:01:31 -04:00
tmp - = ( tmp > > 4 ) ;
2009-08-23 10:57:53 +09:30
2008-10-20 19:01:31 -04:00
ec - > tx_1 + = - ( ec - > tx_1 > > DC_LOG2BETA ) + tmp - ec - > tx_2 ;
tmp1 = ec - > tx_1 > > 15 ;
if ( tmp1 > 32767 )
tmp1 = 32767 ;
if ( tmp1 < - 32767 )
tmp1 = - 32767 ;
tx = tmp1 ;
ec - > tx_2 = tmp ;
}
return tx ;
2008-10-06 21:41:46 -07:00
}
2008-10-12 07:17:26 +02:00
EXPORT_SYMBOL_GPL ( oslec_hpf_tx ) ;
2008-10-12 06:55:40 +02:00
MODULE_LICENSE ( " GPL " ) ;
MODULE_AUTHOR ( " David Rowe " ) ;
MODULE_DESCRIPTION ( " Open Source Line Echo Canceller " ) ;
MODULE_VERSION ( " 0.3.0 " ) ;