2010-08-28 05:59:27 +08:00
|
|
|
/*
|
|
|
|
* SpanDSP - a series of DSP components for telephony
|
|
|
|
*
|
|
|
|
* echo.c - An echo cancellor, suitable for electrical and acoustic
|
|
|
|
* cancellation. This code does not currently comply with
|
|
|
|
* any relevant standards (e.g. G.164/5/7/8). One day....
|
|
|
|
*
|
|
|
|
* Written by Steve Underwood <steveu@coppice.org>
|
|
|
|
* Various optimizations and improvements by Mark Spencer <markster@digium.com>
|
|
|
|
*
|
|
|
|
* Copyright (C) 2001 Steve Underwood
|
|
|
|
*
|
|
|
|
* Based on a bit from here, a bit from there, eye of toad,
|
|
|
|
* ear of bat, etc - plus, of course, my own 2 cents.
|
|
|
|
*
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* See http://www.asterisk.org for more information about
|
|
|
|
* the Asterisk project. Please do not directly contact
|
|
|
|
* any of the maintainers of this project for assistance;
|
|
|
|
* the project provides a web site, mailing lists and IRC
|
|
|
|
* channels for your use.
|
|
|
|
*
|
|
|
|
* This program is free software, distributed under the terms of
|
|
|
|
* the GNU General Public License Version 2 as published by the
|
|
|
|
* Free Software Foundation. See the LICENSE file included with
|
|
|
|
* this program for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* TODO:
|
|
|
|
Finish the echo suppressor option, however nasty suppression may be
|
|
|
|
Add an option to reintroduce side tone at -24dB under appropriate conditions.
|
|
|
|
Improve double talk detector (iterative!)
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/errno.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/ctype.h>
|
|
|
|
#include <linux/moduleparam.h>
|
|
|
|
|
|
|
|
#include <dahdi/kernel.h>
|
|
|
|
|
|
|
|
static int debug;
|
|
|
|
|
|
|
|
#include "arith.h"
|
|
|
|
|
|
|
|
#ifndef NULL
|
|
|
|
#define NULL 0
|
|
|
|
#endif
|
|
|
|
#ifndef FALSE
|
|
|
|
#define FALSE 0
|
|
|
|
#endif
|
|
|
|
#ifndef TRUE
|
|
|
|
#define TRUE (!FALSE)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define USE_SHORTS
|
|
|
|
|
|
|
|
#define NONUPDATE_DWELL_TIME 600 /* 600 samples, or 75ms */
|
|
|
|
|
|
|
|
/* Original parameters :
|
|
|
|
#define MIN_TX_POWER_FOR_ADAPTION 256
|
|
|
|
#define MIN_RX_POWER_FOR_ADAPTION 128
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define MIN_TX_POWER_FOR_ADAPTION 256
|
|
|
|
#define MIN_RX_POWER_FOR_ADAPTION 64
|
|
|
|
|
|
|
|
/* Better ones found by Jim
|
|
|
|
#define MIN_TX_POWER_FOR_ADAPTION 128
|
|
|
|
#define MIN_RX_POWER_FOR_ADAPTION 64
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int echo_can_create(struct dahdi_chan *chan, struct dahdi_echocanparams *ecp,
|
|
|
|
struct dahdi_echocanparam *p, struct dahdi_echocan_state **ec);
|
|
|
|
static void echo_can_free(struct dahdi_chan *chan, struct dahdi_echocan_state *ec);
|
|
|
|
static void echo_can_process(struct dahdi_echocan_state *ec, short *isig, const short *iref, u32 size);
|
|
|
|
static int echo_can_traintap(struct dahdi_echocan_state *ec, int pos, short val);
|
|
|
|
static void echocan_NLP_toggle(struct dahdi_echocan_state *ec, unsigned int enable);
|
2010-12-10 04:19:26 +08:00
|
|
|
static const char *name = "SEC";
|
|
|
|
static const char *ec_name(const struct dahdi_chan *chan) { return name; }
|
2010-08-28 05:59:27 +08:00
|
|
|
|
|
|
|
static const struct dahdi_echocan_factory my_factory = {
|
2010-12-10 04:19:26 +08:00
|
|
|
.get_name = ec_name,
|
2010-08-28 05:59:27 +08:00
|
|
|
.owner = THIS_MODULE,
|
|
|
|
.echocan_create = echo_can_create,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct dahdi_echocan_features my_features = {
|
|
|
|
.NLP_toggle = 1,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct dahdi_echocan_ops my_ops = {
|
|
|
|
.echocan_free = echo_can_free,
|
|
|
|
.echocan_process = echo_can_process,
|
|
|
|
.echocan_traintap = echo_can_traintap,
|
|
|
|
.echocan_NLP_toggle = echocan_NLP_toggle,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ec_pvt {
|
|
|
|
struct dahdi_echocan_state dahdi;
|
|
|
|
int tx_power;
|
|
|
|
int rx_power;
|
|
|
|
int clean_rx_power;
|
|
|
|
|
|
|
|
int rx_power_threshold;
|
|
|
|
int nonupdate_dwell;
|
|
|
|
|
|
|
|
int16_t *tx_history; /* Last N tx samples */
|
|
|
|
int32_t *fir_taps; /* Echo FIR taps */
|
|
|
|
int16_t *fir_taps_short; /* Echo FIR taps, shorts instead of ints */
|
|
|
|
|
|
|
|
int curr_pos;
|
|
|
|
|
|
|
|
int taps;
|
|
|
|
int tap_mask;
|
|
|
|
int use_nlp;
|
|
|
|
int use_suppressor;
|
|
|
|
|
|
|
|
int32_t supp_test1;
|
|
|
|
int32_t supp_test2;
|
|
|
|
int32_t supp1;
|
|
|
|
int32_t supp2;
|
|
|
|
|
|
|
|
int32_t latest_correction; /* Indication of the magnitude of the latest
|
|
|
|
adaption, or a code to indicate why adaption
|
|
|
|
was skipped, for test purposes */
|
|
|
|
};
|
|
|
|
|
|
|
|
#define dahdi_to_pvt(a) container_of(a, struct ec_pvt, dahdi)
|
|
|
|
|
|
|
|
static int echo_can_create(struct dahdi_chan *chan, struct dahdi_echocanparams *ecp,
|
|
|
|
struct dahdi_echocanparam *p, struct dahdi_echocan_state **ec)
|
|
|
|
{
|
|
|
|
struct ec_pvt *pvt;
|
|
|
|
size_t size;
|
|
|
|
|
|
|
|
if (ecp->param_count > 0) {
|
|
|
|
printk(KERN_WARNING "SEC does not support parameters; failing request\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
size = sizeof(*pvt) + ecp->tap_length * sizeof(int32_t) + ecp->tap_length * 3 * sizeof(int16_t);
|
|
|
|
|
|
|
|
pvt = kzalloc(size, GFP_KERNEL);
|
|
|
|
if (!pvt)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
pvt->dahdi.ops = &my_ops;
|
|
|
|
pvt->dahdi.features = my_features;
|
|
|
|
|
|
|
|
pvt->taps = ecp->tap_length;
|
|
|
|
pvt->tap_mask = ecp->tap_length - 1;
|
|
|
|
pvt->tx_history = (int16_t *) (pvt + sizeof(*pvt));
|
|
|
|
pvt->fir_taps = (int32_t *) (pvt + sizeof(*pvt) +
|
|
|
|
ecp->tap_length * 2 * sizeof(int16_t));
|
|
|
|
pvt->fir_taps_short = (int16_t *) (pvt + sizeof(*pvt) +
|
|
|
|
ecp->tap_length * sizeof(int32_t) +
|
|
|
|
ecp->tap_length * 2 * sizeof(int16_t));
|
|
|
|
pvt->rx_power_threshold = 10000000;
|
|
|
|
pvt->use_suppressor = FALSE;
|
|
|
|
/* Non-linear processor - a fancy way to say "zap small signals, to avoid
|
|
|
|
accumulating noise". */
|
|
|
|
pvt->use_nlp = TRUE;
|
|
|
|
|
|
|
|
*ec = &pvt->dahdi;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void echo_can_free(struct dahdi_chan *chan, struct dahdi_echocan_state *ec)
|
|
|
|
{
|
|
|
|
struct ec_pvt *pvt = dahdi_to_pvt(ec);
|
|
|
|
|
|
|
|
kfree(pvt);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int16_t sample_update(struct ec_pvt *pvt, int16_t tx, int16_t rx)
|
|
|
|
{
|
|
|
|
int32_t echo_value;
|
|
|
|
int clean_rx;
|
|
|
|
int nsuppr;
|
|
|
|
|
|
|
|
pvt->tx_history[pvt->curr_pos] = tx;
|
|
|
|
pvt->tx_history[pvt->curr_pos + pvt->taps] = tx;
|
|
|
|
|
|
|
|
/* Evaluate the echo - i.e. apply the FIR filter */
|
|
|
|
/* Assume the gain of the FIR does not exceed unity. Exceeding unity
|
|
|
|
would seem like a rather poor thing for an echo cancellor to do :)
|
|
|
|
This means we can compute the result with a total disregard for
|
|
|
|
overflows. 16bits x 16bits -> 31bits, so no overflow can occur in
|
|
|
|
any multiply. While accumulating we may overflow and underflow the
|
|
|
|
32 bit scale often. However, if the gain does not exceed unity,
|
|
|
|
everything should work itself out, and the final result will be
|
|
|
|
OK, without any saturation logic. */
|
|
|
|
/* Overflow is very much possible here, and we do nothing about it because
|
|
|
|
of the compute costs */
|
|
|
|
/* 16 bit coeffs for the LMS give lousy results (maths good, actual sound
|
|
|
|
bad!), but 32 bit coeffs require some shifting. On balance 32 bit seems
|
|
|
|
best */
|
|
|
|
#ifdef USE_SHORTS
|
|
|
|
echo_value = CONVOLVE2(pvt->fir_taps_short, pvt->tx_history + pvt->curr_pos, pvt->taps);
|
|
|
|
#else
|
|
|
|
echo_value = CONVOLVE(pvt->fir_taps, pvt->tx_history + pvt->curr_pos, pvt->taps);
|
|
|
|
#endif
|
|
|
|
echo_value >>= 16;
|
|
|
|
|
|
|
|
/* And the answer is..... */
|
|
|
|
clean_rx = rx - echo_value;
|
|
|
|
|
|
|
|
/* That was the easy part. Now we need to adapt! */
|
|
|
|
if (pvt->nonupdate_dwell > 0)
|
|
|
|
pvt->nonupdate_dwell--;
|
|
|
|
|
|
|
|
/* If there is very little being transmitted, any attempt to train is
|
|
|
|
futile. We would either be training on the far end's noise or signal,
|
|
|
|
the channel's own noise, or our noise. Either way, this is hardly good
|
|
|
|
training, so don't do it (avoid trouble). */
|
|
|
|
/* If the received power is very low, either we are sending very little or
|
|
|
|
we are already well adapted. There is little point in trying to improve
|
|
|
|
the adaption under these circumstanceson, so don't do it (reduce the
|
|
|
|
compute load). */
|
|
|
|
if (pvt->tx_power > MIN_TX_POWER_FOR_ADAPTION && pvt->rx_power > MIN_RX_POWER_FOR_ADAPTION) {
|
|
|
|
/* This is a really crude piece of decision logic, but it does OK
|
|
|
|
for now. */
|
|
|
|
if (pvt->tx_power > pvt->rx_power << 1) {
|
|
|
|
/* There is no far-end speech detected */
|
|
|
|
if (pvt->nonupdate_dwell == 0) {
|
|
|
|
/* ... and we are not in the dwell time from previous speech. */
|
|
|
|
/* nsuppr = saturate((clean_rx << 16)/pvt->tx_power); */
|
|
|
|
nsuppr = (clean_rx << 16) / pvt->tx_power;
|
|
|
|
nsuppr >>= 4;
|
|
|
|
if (nsuppr > 512)
|
|
|
|
nsuppr = 512;
|
|
|
|
if (nsuppr < -512)
|
|
|
|
nsuppr = -512;
|
|
|
|
|
|
|
|
/* Update the FIR taps */
|
|
|
|
pvt->latest_correction = 0;
|
|
|
|
#ifdef USE_SHORTS
|
|
|
|
UPDATE2(pvt->fir_taps, pvt->fir_taps_short, pvt->tx_history + pvt->curr_pos, nsuppr, pvt->taps);
|
|
|
|
#else
|
|
|
|
UPDATE(pvt->fir_taps, pvt->fir_taps_short, pvt->tx_history + pvt->curr_pos, nsuppr, pvt->taps);
|
|
|
|
#endif
|
|
|
|
} else {
|
|
|
|
pvt->latest_correction = -3;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
pvt->nonupdate_dwell = NONUPDATE_DWELL_TIME;
|
|
|
|
pvt->latest_correction = -2;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
pvt->nonupdate_dwell = 0;
|
|
|
|
pvt->latest_correction = -1;
|
|
|
|
}
|
|
|
|
/* Calculate short term power levels using very simple single pole IIRs */
|
|
|
|
/* TODO: Is the nasty modulus approach the fastest, or would a real
|
|
|
|
tx*tx power calculation actually be faster? */
|
|
|
|
pvt->tx_power += ((abs(tx) - pvt->tx_power) >> 5);
|
|
|
|
pvt->rx_power += ((abs(rx) - pvt->rx_power) >> 5);
|
|
|
|
pvt->clean_rx_power += ((abs(clean_rx) - pvt->clean_rx_power) >> 5);
|
|
|
|
|
|
|
|
#if defined(XYZZY)
|
|
|
|
if (pvt->use_suppressor) {
|
|
|
|
pvt->supp_test1 += (pvt->tx_history[pvt->curr_pos] - pvt->tx_history[(pvt->curr_pos - 7) & pvt->tap_mask]);
|
|
|
|
pvt->supp_test2 += (pvt->tx_history[(pvt->curr_pos - 24) & pvt->tap_mask] - pvt->tx_history[(pvt->curr_pos - 31) & pvt->tap_mask]);
|
|
|
|
if (pvt->supp_test1 > 42 && pvt->supp_test2 > 42)
|
|
|
|
supp_change = 25;
|
|
|
|
else
|
|
|
|
supp_change = 50;
|
|
|
|
supp = supp_change + k1*pvt->supp1 + k2*pvt->supp2;
|
|
|
|
pvt->supp2 = pvt->supp1;
|
|
|
|
pvt->supp1 = supp;
|
|
|
|
clean_rx *= (1 - supp);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (pvt->use_nlp && pvt->rx_power < 32)
|
|
|
|
clean_rx = 0;
|
|
|
|
|
|
|
|
/* Roll around the rolling buffer */
|
|
|
|
pvt->curr_pos = (pvt->curr_pos - 1) & pvt->tap_mask;
|
|
|
|
|
|
|
|
return clean_rx;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void echo_can_process(struct dahdi_echocan_state *ec, short *isig, const short *iref, u32 size)
|
|
|
|
{
|
|
|
|
struct ec_pvt *pvt = dahdi_to_pvt(ec);
|
|
|
|
u32 x;
|
|
|
|
short result;
|
|
|
|
|
|
|
|
for (x = 0; x < size; x++) {
|
|
|
|
result = sample_update(pvt, *iref, *isig);
|
|
|
|
*isig++ = result;
|
|
|
|
++iref;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int echo_can_traintap(struct dahdi_echocan_state *ec, int pos, short val)
|
|
|
|
{
|
|
|
|
struct ec_pvt *pvt = dahdi_to_pvt(ec);
|
|
|
|
|
|
|
|
/* Reset hang counter to avoid adjustments after
|
|
|
|
initial forced training */
|
|
|
|
pvt->nonupdate_dwell = pvt->taps << 1;
|
|
|
|
if (pos >= pvt->taps)
|
|
|
|
return 1;
|
|
|
|
pvt->fir_taps[pos] = val << 17;
|
|
|
|
pvt->fir_taps_short[pos] = val << 1;
|
|
|
|
if (++pos >= pvt->taps)
|
|
|
|
return 1;
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void echocan_NLP_toggle(struct dahdi_echocan_state *ec, unsigned int enable)
|
|
|
|
{
|
|
|
|
struct ec_pvt *pvt = dahdi_to_pvt(ec);
|
|
|
|
|
|
|
|
pvt->use_nlp = enable ? 1 : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __init mod_init(void)
|
|
|
|
{
|
|
|
|
if (dahdi_register_echocan_factory(&my_factory)) {
|
|
|
|
module_printk(KERN_ERR, "could not register with DAHDI core\n");
|
|
|
|
|
|
|
|
return -EPERM;
|
|
|
|
}
|
|
|
|
|
2010-12-10 04:19:26 +08:00
|
|
|
module_printk(KERN_NOTICE, "Registered echo canceler '%s'\n",
|
|
|
|
my_factory.get_name(NULL));
|
2010-08-28 05:59:27 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __exit mod_exit(void)
|
|
|
|
{
|
|
|
|
dahdi_unregister_echocan_factory(&my_factory);
|
|
|
|
}
|
|
|
|
|
|
|
|
module_param(debug, int, S_IRUGO | S_IWUSR);
|
|
|
|
|
|
|
|
MODULE_DESCRIPTION("DAHDI 'SEC' Echo Canceler");
|
|
|
|
MODULE_AUTHOR("Steve Underwood <steveu@coppice.org>");
|
|
|
|
MODULE_LICENSE("GPL");
|
|
|
|
|
|
|
|
module_init(mod_init);
|
|
|
|
module_exit(mod_exit);
|