123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313 |
- /*
- * DTMF decoder.
- *
- * Copyright by Andreas Eversberg (jolly@eversberg.eu)
- * based on different decoders such as ISDN4Linux
- *
- * This software may be used and distributed according to the terms
- * of the GNU General Public License, incorporated herein by reference.
- *
- */
- #include <linux/mISDNif.h>
- #include <linux/mISDNdsp.h>
- #include "core.h"
- #include "dsp.h"
- #define NCOEFF 8 /* number of frequencies to be analyzed */
- /* For DTMF recognition:
- * 2 * cos(2 * PI * k / N) precalculated for all k
- */
- static u64 cos2pik[NCOEFF] =
- {
- /* k << 15 (source: hfc-4s/8s documentation (www.colognechip.de)) */
- 55960, 53912, 51402, 48438, 38146, 32650, 26170, 18630
- };
- /* digit matrix */
- static char dtmf_matrix[4][4] =
- {
- {'1', '2', '3', 'A'},
- {'4', '5', '6', 'B'},
- {'7', '8', '9', 'C'},
- {'*', '0', '#', 'D'}
- };
- /* dtmf detection using goertzel algorithm
- * init function
- */
- void dsp_dtmf_goertzel_init(struct dsp *dsp)
- {
- dsp->dtmf.size = 0;
- dsp->dtmf.lastwhat = '\0';
- dsp->dtmf.lastdigit = '\0';
- dsp->dtmf.count = 0;
- }
- /* check for hardware or software features
- */
- void dsp_dtmf_hardware(struct dsp *dsp)
- {
- int hardware = 1;
- if (!dsp->dtmf.enable)
- return;
- if (!dsp->features.hfc_dtmf)
- hardware = 0;
- /* check for volume change */
- if (dsp->tx_volume) {
- if (dsp_debug & DEBUG_DSP_DTMF)
- printk(KERN_DEBUG "%s dsp %s cannot do hardware DTMF, "
- "because tx_volume is changed\n",
- __func__, dsp->name);
- hardware = 0;
- }
- if (dsp->rx_volume) {
- if (dsp_debug & DEBUG_DSP_DTMF)
- printk(KERN_DEBUG "%s dsp %s cannot do hardware DTMF, "
- "because rx_volume is changed\n",
- __func__, dsp->name);
- hardware = 0;
- }
- /* check if encryption is enabled */
- if (dsp->bf_enable) {
- if (dsp_debug & DEBUG_DSP_DTMF)
- printk(KERN_DEBUG "%s dsp %s cannot do hardware DTMF, "
- "because encryption is enabled\n",
- __func__, dsp->name);
- hardware = 0;
- }
- /* check if pipeline exists */
- if (dsp->pipeline.inuse) {
- if (dsp_debug & DEBUG_DSP_DTMF)
- printk(KERN_DEBUG "%s dsp %s cannot do hardware DTMF, "
- "because pipeline exists.\n",
- __func__, dsp->name);
- hardware = 0;
- }
- dsp->dtmf.hardware = hardware;
- dsp->dtmf.software = !hardware;
- }
- /*************************************************************
- * calculate the coefficients of the given sample and decode *
- *************************************************************/
- /* the given sample is decoded. if the sample is not long enough for a
- * complete frame, the decoding is finished and continued with the next
- * call of this function.
- *
- * the algorithm is very good for detection with a minimum of errors. i
- * tested it allot. it even works with very short tones (40ms). the only
- * disadvantage is, that it doesn't work good with different volumes of both
- * tones. this will happen, if accoustically coupled dialers are used.
- * it sometimes detects tones during speech, which is normal for decoders.
- * use sequences to given commands during calls.
- *
- * dtmf - points to a structure of the current dtmf state
- * spl and len - the sample
- * fmt - 0 = alaw, 1 = ulaw, 2 = coefficients from HFC DTMF hw-decoder
- */
- u8
- *dsp_dtmf_goertzel_decode(struct dsp *dsp, u8 *data, int len, int fmt)
- {
- u8 what;
- int size;
- signed short *buf;
- s32 sk, sk1, sk2;
- int k, n, i;
- s32 *hfccoeff;
- s32 result[NCOEFF], tresh, treshl;
- int lowgroup, highgroup;
- s64 cos2pik_;
- dsp->dtmf.digits[0] = '\0';
- /* Note: The function will loop until the buffer has not enough samples
- * left to decode a full frame.
- */
- again:
- /* convert samples */
- size = dsp->dtmf.size;
- buf = dsp->dtmf.buffer;
- switch (fmt) {
- case 0: /* alaw */
- case 1: /* ulaw */
- while (size < DSP_DTMF_NPOINTS && len) {
- buf[size++] = dsp_audio_law_to_s32[*data++];
- len--;
- }
- break;
- case 2: /* HFC coefficients */
- default:
- if (len < 64) {
- if (len > 0)
- printk(KERN_ERR "%s: coefficients have invalid "
- "size. (is=%d < must=%d)\n",
- __func__, len, 64);
- return dsp->dtmf.digits;
- }
- hfccoeff = (s32 *)data;
- for (k = 0; k < NCOEFF; k++) {
- sk2 = (*hfccoeff++) >> 4;
- sk = (*hfccoeff++) >> 4;
- if (sk > 32767 || sk < -32767 || sk2 > 32767
- || sk2 < -32767)
- printk(KERN_WARNING
- "DTMF-Detection overflow\n");
- /* compute |X(k)|**2 */
- result[k] =
- (sk * sk) -
- (((cos2pik[k] * sk) >> 15) * sk2) +
- (sk2 * sk2);
- }
- data += 64;
- len -= 64;
- goto coefficients;
- break;
- }
- dsp->dtmf.size = size;
- if (size < DSP_DTMF_NPOINTS)
- return dsp->dtmf.digits;
- dsp->dtmf.size = 0;
- /* now we have a full buffer of signed long samples - we do goertzel */
- for (k = 0; k < NCOEFF; k++) {
- sk = 0;
- sk1 = 0;
- sk2 = 0;
- buf = dsp->dtmf.buffer;
- cos2pik_ = cos2pik[k];
- for (n = 0; n < DSP_DTMF_NPOINTS; n++) {
- sk = ((cos2pik_ * sk1) >> 15) - sk2 + (*buf++);
- sk2 = sk1;
- sk1 = sk;
- }
- sk >>= 8;
- sk2 >>= 8;
- if (sk > 32767 || sk < -32767 || sk2 > 32767 || sk2 < -32767)
- printk(KERN_WARNING "DTMF-Detection overflow\n");
- /* compute |X(k)|**2 */
- result[k] =
- (sk * sk) -
- (((cos2pik[k] * sk) >> 15) * sk2) +
- (sk2 * sk2);
- }
- /* our (squared) coefficients have been calculated, we need to process
- * them.
- */
- coefficients:
- tresh = 0;
- for (i = 0; i < NCOEFF; i++) {
- if (result[i] < 0)
- result[i] = 0;
- if (result[i] > dsp->dtmf.treshold) {
- if (result[i] > tresh)
- tresh = result[i];
- }
- }
- if (tresh == 0) {
- what = 0;
- goto storedigit;
- }
- if (dsp_debug & DEBUG_DSP_DTMFCOEFF) {
- s32 tresh_100 = tresh/100;
- if (tresh_100 == 0) {
- tresh_100 = 1;
- printk(KERN_DEBUG
- "tresh(%d) too small set tresh/100 to 1\n",
- tresh);
- }
- printk(KERN_DEBUG "a %3d %3d %3d %3d %3d %3d %3d %3d"
- " tr:%3d r %3d %3d %3d %3d %3d %3d %3d %3d\n",
- result[0] / 10000, result[1] / 10000, result[2] / 10000,
- result[3] / 10000, result[4] / 10000, result[5] / 10000,
- result[6] / 10000, result[7] / 10000, tresh / 10000,
- result[0] / (tresh_100), result[1] / (tresh_100),
- result[2] / (tresh_100), result[3] / (tresh_100),
- result[4] / (tresh_100), result[5] / (tresh_100),
- result[6] / (tresh_100), result[7] / (tresh_100));
- }
- /* calc digit (lowgroup/highgroup) */
- lowgroup = -1;
- highgroup = -1;
- treshl = tresh >> 3; /* tones which are not on, must be below 9 dB */
- tresh = tresh >> 2; /* touchtones must match within 6 dB */
- for (i = 0; i < NCOEFF; i++) {
- if (result[i] < treshl)
- continue; /* ignore */
- if (result[i] < tresh) {
- lowgroup = -1;
- highgroup = -1;
- break; /* noise in between */
- }
- /* good level found. This is allowed only one time per group */
- if (i < NCOEFF / 2) {
- /* lowgroup */
- if (lowgroup >= 0) {
- /* Bad. Another tone found. */
- lowgroup = -1;
- break;
- } else
- lowgroup = i;
- } else {
- /* higroup */
- if (highgroup >= 0) {
- /* Bad. Another tone found. */
- highgroup = -1;
- break;
- } else
- highgroup = i - (NCOEFF / 2);
- }
- }
- /* get digit or null */
- what = 0;
- if (lowgroup >= 0 && highgroup >= 0)
- what = dtmf_matrix[lowgroup][highgroup];
- storedigit:
- if (what && (dsp_debug & DEBUG_DSP_DTMF))
- printk(KERN_DEBUG "DTMF what: %c\n", what);
- if (dsp->dtmf.lastwhat != what)
- dsp->dtmf.count = 0;
- /* the tone (or no tone) must remain 3 times without change */
- if (dsp->dtmf.count == 2) {
- if (dsp->dtmf.lastdigit != what) {
- dsp->dtmf.lastdigit = what;
- if (what) {
- if (dsp_debug & DEBUG_DSP_DTMF)
- printk(KERN_DEBUG "DTMF digit: %c\n",
- what);
- if ((strlen(dsp->dtmf.digits) + 1)
- < sizeof(dsp->dtmf.digits)) {
- dsp->dtmf.digits[strlen(
- dsp->dtmf.digits) + 1] = '\0';
- dsp->dtmf.digits[strlen(
- dsp->dtmf.digits)] = what;
- }
- }
- }
- } else
- dsp->dtmf.count++;
- dsp->dtmf.lastwhat = what;
- goto again;
- }
|