162306a36Sopenharmony_ci/*
262306a36Sopenharmony_ci * DTMF decoder.
362306a36Sopenharmony_ci *
462306a36Sopenharmony_ci * Copyright            by Andreas Eversberg (jolly@eversberg.eu)
562306a36Sopenharmony_ci *			based on different decoders such as ISDN4Linux
662306a36Sopenharmony_ci *
762306a36Sopenharmony_ci * This software may be used and distributed according to the terms
862306a36Sopenharmony_ci * of the GNU General Public License, incorporated herein by reference.
962306a36Sopenharmony_ci *
1062306a36Sopenharmony_ci */
1162306a36Sopenharmony_ci
1262306a36Sopenharmony_ci#include <linux/mISDNif.h>
1362306a36Sopenharmony_ci#include <linux/mISDNdsp.h>
1462306a36Sopenharmony_ci#include "core.h"
1562306a36Sopenharmony_ci#include "dsp.h"
1662306a36Sopenharmony_ci
1762306a36Sopenharmony_ci#define NCOEFF            8     /* number of frequencies to be analyzed */
1862306a36Sopenharmony_ci
1962306a36Sopenharmony_ci/* For DTMF recognition:
2062306a36Sopenharmony_ci * 2 * cos(2 * PI * k / N) precalculated for all k
2162306a36Sopenharmony_ci */
2262306a36Sopenharmony_cistatic u64 cos2pik[NCOEFF] =
2362306a36Sopenharmony_ci{
2462306a36Sopenharmony_ci	/* k << 15 (source: hfc-4s/8s documentation (www.colognechip.de)) */
2562306a36Sopenharmony_ci	55960, 53912, 51402, 48438, 38146, 32650, 26170, 18630
2662306a36Sopenharmony_ci};
2762306a36Sopenharmony_ci
2862306a36Sopenharmony_ci/* digit matrix */
2962306a36Sopenharmony_cistatic char dtmf_matrix[4][4] =
3062306a36Sopenharmony_ci{
3162306a36Sopenharmony_ci	{'1', '2', '3', 'A'},
3262306a36Sopenharmony_ci	{'4', '5', '6', 'B'},
3362306a36Sopenharmony_ci	{'7', '8', '9', 'C'},
3462306a36Sopenharmony_ci	{'*', '0', '#', 'D'}
3562306a36Sopenharmony_ci};
3662306a36Sopenharmony_ci
3762306a36Sopenharmony_ci/* dtmf detection using goertzel algorithm
3862306a36Sopenharmony_ci * init function
3962306a36Sopenharmony_ci */
4062306a36Sopenharmony_civoid dsp_dtmf_goertzel_init(struct dsp *dsp)
4162306a36Sopenharmony_ci{
4262306a36Sopenharmony_ci	dsp->dtmf.size = 0;
4362306a36Sopenharmony_ci	dsp->dtmf.lastwhat = '\0';
4462306a36Sopenharmony_ci	dsp->dtmf.lastdigit = '\0';
4562306a36Sopenharmony_ci	dsp->dtmf.count = 0;
4662306a36Sopenharmony_ci}
4762306a36Sopenharmony_ci
4862306a36Sopenharmony_ci/* check for hardware or software features
4962306a36Sopenharmony_ci */
5062306a36Sopenharmony_civoid dsp_dtmf_hardware(struct dsp *dsp)
5162306a36Sopenharmony_ci{
5262306a36Sopenharmony_ci	int hardware = 1;
5362306a36Sopenharmony_ci
5462306a36Sopenharmony_ci	if (!dsp->dtmf.enable)
5562306a36Sopenharmony_ci		return;
5662306a36Sopenharmony_ci
5762306a36Sopenharmony_ci	if (!dsp->features.hfc_dtmf)
5862306a36Sopenharmony_ci		hardware = 0;
5962306a36Sopenharmony_ci
6062306a36Sopenharmony_ci	/* check for volume change */
6162306a36Sopenharmony_ci	if (dsp->tx_volume) {
6262306a36Sopenharmony_ci		if (dsp_debug & DEBUG_DSP_DTMF)
6362306a36Sopenharmony_ci			printk(KERN_DEBUG "%s dsp %s cannot do hardware DTMF, "
6462306a36Sopenharmony_ci			       "because tx_volume is changed\n",
6562306a36Sopenharmony_ci			       __func__, dsp->name);
6662306a36Sopenharmony_ci		hardware = 0;
6762306a36Sopenharmony_ci	}
6862306a36Sopenharmony_ci	if (dsp->rx_volume) {
6962306a36Sopenharmony_ci		if (dsp_debug & DEBUG_DSP_DTMF)
7062306a36Sopenharmony_ci			printk(KERN_DEBUG "%s dsp %s cannot do hardware DTMF, "
7162306a36Sopenharmony_ci			       "because rx_volume is changed\n",
7262306a36Sopenharmony_ci			       __func__, dsp->name);
7362306a36Sopenharmony_ci		hardware = 0;
7462306a36Sopenharmony_ci	}
7562306a36Sopenharmony_ci	/* check if encryption is enabled */
7662306a36Sopenharmony_ci	if (dsp->bf_enable) {
7762306a36Sopenharmony_ci		if (dsp_debug & DEBUG_DSP_DTMF)
7862306a36Sopenharmony_ci			printk(KERN_DEBUG "%s dsp %s cannot do hardware DTMF, "
7962306a36Sopenharmony_ci			       "because encryption is enabled\n",
8062306a36Sopenharmony_ci			       __func__, dsp->name);
8162306a36Sopenharmony_ci		hardware = 0;
8262306a36Sopenharmony_ci	}
8362306a36Sopenharmony_ci	/* check if pipeline exists */
8462306a36Sopenharmony_ci	if (dsp->pipeline.inuse) {
8562306a36Sopenharmony_ci		if (dsp_debug & DEBUG_DSP_DTMF)
8662306a36Sopenharmony_ci			printk(KERN_DEBUG "%s dsp %s cannot do hardware DTMF, "
8762306a36Sopenharmony_ci			       "because pipeline exists.\n",
8862306a36Sopenharmony_ci			       __func__, dsp->name);
8962306a36Sopenharmony_ci		hardware = 0;
9062306a36Sopenharmony_ci	}
9162306a36Sopenharmony_ci
9262306a36Sopenharmony_ci	dsp->dtmf.hardware = hardware;
9362306a36Sopenharmony_ci	dsp->dtmf.software = !hardware;
9462306a36Sopenharmony_ci}
9562306a36Sopenharmony_ci
9662306a36Sopenharmony_ci
9762306a36Sopenharmony_ci/*************************************************************
9862306a36Sopenharmony_ci * calculate the coefficients of the given sample and decode *
9962306a36Sopenharmony_ci *************************************************************/
10062306a36Sopenharmony_ci
10162306a36Sopenharmony_ci/* the given sample is decoded. if the sample is not long enough for a
10262306a36Sopenharmony_ci * complete frame, the decoding is finished and continued with the next
10362306a36Sopenharmony_ci * call of this function.
10462306a36Sopenharmony_ci *
10562306a36Sopenharmony_ci * the algorithm is very good for detection with a minimum of errors. i
10662306a36Sopenharmony_ci * tested it allot. it even works with very short tones (40ms). the only
10762306a36Sopenharmony_ci * disadvantage is, that it doesn't work good with different volumes of both
10862306a36Sopenharmony_ci * tones. this will happen, if accoustically coupled dialers are used.
10962306a36Sopenharmony_ci * it sometimes detects tones during speech, which is normal for decoders.
11062306a36Sopenharmony_ci * use sequences to given commands during calls.
11162306a36Sopenharmony_ci *
11262306a36Sopenharmony_ci * dtmf - points to a structure of the current dtmf state
11362306a36Sopenharmony_ci * spl and len - the sample
11462306a36Sopenharmony_ci * fmt - 0 = alaw, 1 = ulaw, 2 = coefficients from HFC DTMF hw-decoder
11562306a36Sopenharmony_ci */
11662306a36Sopenharmony_ci
11762306a36Sopenharmony_ciu8
11862306a36Sopenharmony_ci*dsp_dtmf_goertzel_decode(struct dsp *dsp, u8 *data, int len, int fmt)
11962306a36Sopenharmony_ci{
12062306a36Sopenharmony_ci	u8 what;
12162306a36Sopenharmony_ci	int size;
12262306a36Sopenharmony_ci	signed short *buf;
12362306a36Sopenharmony_ci	s32 sk, sk1, sk2;
12462306a36Sopenharmony_ci	int k, n, i;
12562306a36Sopenharmony_ci	s32 *hfccoeff;
12662306a36Sopenharmony_ci	s32 result[NCOEFF], tresh, treshl;
12762306a36Sopenharmony_ci	int lowgroup, highgroup;
12862306a36Sopenharmony_ci	s64 cos2pik_;
12962306a36Sopenharmony_ci
13062306a36Sopenharmony_ci	dsp->dtmf.digits[0] = '\0';
13162306a36Sopenharmony_ci
13262306a36Sopenharmony_ci	/* Note: The function will loop until the buffer has not enough samples
13362306a36Sopenharmony_ci	 * left to decode a full frame.
13462306a36Sopenharmony_ci	 */
13562306a36Sopenharmony_ciagain:
13662306a36Sopenharmony_ci	/* convert samples */
13762306a36Sopenharmony_ci	size = dsp->dtmf.size;
13862306a36Sopenharmony_ci	buf = dsp->dtmf.buffer;
13962306a36Sopenharmony_ci	switch (fmt) {
14062306a36Sopenharmony_ci	case 0: /* alaw */
14162306a36Sopenharmony_ci	case 1: /* ulaw */
14262306a36Sopenharmony_ci		while (size < DSP_DTMF_NPOINTS && len) {
14362306a36Sopenharmony_ci			buf[size++] = dsp_audio_law_to_s32[*data++];
14462306a36Sopenharmony_ci			len--;
14562306a36Sopenharmony_ci		}
14662306a36Sopenharmony_ci		break;
14762306a36Sopenharmony_ci
14862306a36Sopenharmony_ci	case 2: /* HFC coefficients */
14962306a36Sopenharmony_ci	default:
15062306a36Sopenharmony_ci		if (len < 64) {
15162306a36Sopenharmony_ci			if (len > 0)
15262306a36Sopenharmony_ci				printk(KERN_ERR "%s: coefficients have invalid "
15362306a36Sopenharmony_ci				       "size. (is=%d < must=%d)\n",
15462306a36Sopenharmony_ci				       __func__, len, 64);
15562306a36Sopenharmony_ci			return dsp->dtmf.digits;
15662306a36Sopenharmony_ci		}
15762306a36Sopenharmony_ci		hfccoeff = (s32 *)data;
15862306a36Sopenharmony_ci		for (k = 0; k < NCOEFF; k++) {
15962306a36Sopenharmony_ci			sk2 = (*hfccoeff++) >> 4;
16062306a36Sopenharmony_ci			sk = (*hfccoeff++) >> 4;
16162306a36Sopenharmony_ci			if (sk > 32767 || sk < -32767 || sk2 > 32767
16262306a36Sopenharmony_ci			    || sk2 < -32767)
16362306a36Sopenharmony_ci				printk(KERN_WARNING
16462306a36Sopenharmony_ci				       "DTMF-Detection overflow\n");
16562306a36Sopenharmony_ci			/* compute |X(k)|**2 */
16662306a36Sopenharmony_ci			result[k] =
16762306a36Sopenharmony_ci				(sk * sk) -
16862306a36Sopenharmony_ci				(((cos2pik[k] * sk) >> 15) * sk2) +
16962306a36Sopenharmony_ci				(sk2 * sk2);
17062306a36Sopenharmony_ci		}
17162306a36Sopenharmony_ci		data += 64;
17262306a36Sopenharmony_ci		len -= 64;
17362306a36Sopenharmony_ci		goto coefficients;
17462306a36Sopenharmony_ci		break;
17562306a36Sopenharmony_ci	}
17662306a36Sopenharmony_ci	dsp->dtmf.size = size;
17762306a36Sopenharmony_ci
17862306a36Sopenharmony_ci	if (size < DSP_DTMF_NPOINTS)
17962306a36Sopenharmony_ci		return dsp->dtmf.digits;
18062306a36Sopenharmony_ci
18162306a36Sopenharmony_ci	dsp->dtmf.size = 0;
18262306a36Sopenharmony_ci
18362306a36Sopenharmony_ci	/* now we have a full buffer of signed long samples - we do goertzel */
18462306a36Sopenharmony_ci	for (k = 0; k < NCOEFF; k++) {
18562306a36Sopenharmony_ci		sk = 0;
18662306a36Sopenharmony_ci		sk1 = 0;
18762306a36Sopenharmony_ci		sk2 = 0;
18862306a36Sopenharmony_ci		buf = dsp->dtmf.buffer;
18962306a36Sopenharmony_ci		cos2pik_ = cos2pik[k];
19062306a36Sopenharmony_ci		for (n = 0; n < DSP_DTMF_NPOINTS; n++) {
19162306a36Sopenharmony_ci			sk = ((cos2pik_ * sk1) >> 15) - sk2 + (*buf++);
19262306a36Sopenharmony_ci			sk2 = sk1;
19362306a36Sopenharmony_ci			sk1 = sk;
19462306a36Sopenharmony_ci		}
19562306a36Sopenharmony_ci		sk >>= 8;
19662306a36Sopenharmony_ci		sk2 >>= 8;
19762306a36Sopenharmony_ci		if (sk > 32767 || sk < -32767 || sk2 > 32767 || sk2 < -32767)
19862306a36Sopenharmony_ci			printk(KERN_WARNING "DTMF-Detection overflow\n");
19962306a36Sopenharmony_ci		/* compute |X(k)|**2 */
20062306a36Sopenharmony_ci		result[k] =
20162306a36Sopenharmony_ci			(sk * sk) -
20262306a36Sopenharmony_ci			(((cos2pik[k] * sk) >> 15) * sk2) +
20362306a36Sopenharmony_ci			(sk2 * sk2);
20462306a36Sopenharmony_ci	}
20562306a36Sopenharmony_ci
20662306a36Sopenharmony_ci	/* our (squared) coefficients have been calculated, we need to process
20762306a36Sopenharmony_ci	 * them.
20862306a36Sopenharmony_ci	 */
20962306a36Sopenharmony_cicoefficients:
21062306a36Sopenharmony_ci	tresh = 0;
21162306a36Sopenharmony_ci	for (i = 0; i < NCOEFF; i++) {
21262306a36Sopenharmony_ci		if (result[i] < 0)
21362306a36Sopenharmony_ci			result[i] = 0;
21462306a36Sopenharmony_ci		if (result[i] > dsp->dtmf.treshold) {
21562306a36Sopenharmony_ci			if (result[i] > tresh)
21662306a36Sopenharmony_ci				tresh = result[i];
21762306a36Sopenharmony_ci		}
21862306a36Sopenharmony_ci	}
21962306a36Sopenharmony_ci
22062306a36Sopenharmony_ci	if (tresh == 0) {
22162306a36Sopenharmony_ci		what = 0;
22262306a36Sopenharmony_ci		goto storedigit;
22362306a36Sopenharmony_ci	}
22462306a36Sopenharmony_ci
22562306a36Sopenharmony_ci	if (dsp_debug & DEBUG_DSP_DTMFCOEFF) {
22662306a36Sopenharmony_ci		s32 tresh_100 = tresh/100;
22762306a36Sopenharmony_ci
22862306a36Sopenharmony_ci		if (tresh_100 == 0) {
22962306a36Sopenharmony_ci			tresh_100 = 1;
23062306a36Sopenharmony_ci			printk(KERN_DEBUG
23162306a36Sopenharmony_ci				"tresh(%d) too small set tresh/100 to 1\n",
23262306a36Sopenharmony_ci				tresh);
23362306a36Sopenharmony_ci		}
23462306a36Sopenharmony_ci		printk(KERN_DEBUG "a %3d %3d %3d %3d %3d %3d %3d %3d"
23562306a36Sopenharmony_ci		       " tr:%3d r %3d %3d %3d %3d %3d %3d %3d %3d\n",
23662306a36Sopenharmony_ci		       result[0] / 10000, result[1] / 10000, result[2] / 10000,
23762306a36Sopenharmony_ci		       result[3] / 10000, result[4] / 10000, result[5] / 10000,
23862306a36Sopenharmony_ci		       result[6] / 10000, result[7] / 10000, tresh / 10000,
23962306a36Sopenharmony_ci		       result[0] / (tresh_100), result[1] / (tresh_100),
24062306a36Sopenharmony_ci		       result[2] / (tresh_100), result[3] / (tresh_100),
24162306a36Sopenharmony_ci		       result[4] / (tresh_100), result[5] / (tresh_100),
24262306a36Sopenharmony_ci		       result[6] / (tresh_100), result[7] / (tresh_100));
24362306a36Sopenharmony_ci	}
24462306a36Sopenharmony_ci
24562306a36Sopenharmony_ci	/* calc digit (lowgroup/highgroup) */
24662306a36Sopenharmony_ci	lowgroup = -1;
24762306a36Sopenharmony_ci	highgroup = -1;
24862306a36Sopenharmony_ci	treshl = tresh >> 3;  /* tones which are not on, must be below 9 dB */
24962306a36Sopenharmony_ci	tresh = tresh >> 2;  /* touchtones must match within 6 dB */
25062306a36Sopenharmony_ci	for (i = 0; i < NCOEFF; i++) {
25162306a36Sopenharmony_ci		if (result[i] < treshl)
25262306a36Sopenharmony_ci			continue;  /* ignore */
25362306a36Sopenharmony_ci		if (result[i] < tresh) {
25462306a36Sopenharmony_ci			lowgroup = -1;
25562306a36Sopenharmony_ci			highgroup = -1;
25662306a36Sopenharmony_ci			break;  /* noise in between */
25762306a36Sopenharmony_ci		}
25862306a36Sopenharmony_ci		/* good level found. This is allowed only one time per group */
25962306a36Sopenharmony_ci		if (i < NCOEFF / 2) {
26062306a36Sopenharmony_ci			/* lowgroup */
26162306a36Sopenharmony_ci			if (lowgroup >= 0) {
26262306a36Sopenharmony_ci				/* Bad. Another tone found. */
26362306a36Sopenharmony_ci				lowgroup = -1;
26462306a36Sopenharmony_ci				break;
26562306a36Sopenharmony_ci			} else
26662306a36Sopenharmony_ci				lowgroup = i;
26762306a36Sopenharmony_ci		} else {
26862306a36Sopenharmony_ci			/* higroup */
26962306a36Sopenharmony_ci			if (highgroup >= 0) {
27062306a36Sopenharmony_ci				/* Bad. Another tone found. */
27162306a36Sopenharmony_ci				highgroup = -1;
27262306a36Sopenharmony_ci				break;
27362306a36Sopenharmony_ci			} else
27462306a36Sopenharmony_ci				highgroup = i - (NCOEFF / 2);
27562306a36Sopenharmony_ci		}
27662306a36Sopenharmony_ci	}
27762306a36Sopenharmony_ci
27862306a36Sopenharmony_ci	/* get digit or null */
27962306a36Sopenharmony_ci	what = 0;
28062306a36Sopenharmony_ci	if (lowgroup >= 0 && highgroup >= 0)
28162306a36Sopenharmony_ci		what = dtmf_matrix[lowgroup][highgroup];
28262306a36Sopenharmony_ci
28362306a36Sopenharmony_cistoredigit:
28462306a36Sopenharmony_ci	if (what && (dsp_debug & DEBUG_DSP_DTMF))
28562306a36Sopenharmony_ci		printk(KERN_DEBUG "DTMF what: %c\n", what);
28662306a36Sopenharmony_ci
28762306a36Sopenharmony_ci	if (dsp->dtmf.lastwhat != what)
28862306a36Sopenharmony_ci		dsp->dtmf.count = 0;
28962306a36Sopenharmony_ci
29062306a36Sopenharmony_ci	/* the tone (or no tone) must remain 3 times without change */
29162306a36Sopenharmony_ci	if (dsp->dtmf.count == 2) {
29262306a36Sopenharmony_ci		if (dsp->dtmf.lastdigit != what) {
29362306a36Sopenharmony_ci			dsp->dtmf.lastdigit = what;
29462306a36Sopenharmony_ci			if (what) {
29562306a36Sopenharmony_ci				if (dsp_debug & DEBUG_DSP_DTMF)
29662306a36Sopenharmony_ci					printk(KERN_DEBUG "DTMF digit: %c\n",
29762306a36Sopenharmony_ci					       what);
29862306a36Sopenharmony_ci				if ((strlen(dsp->dtmf.digits) + 1)
29962306a36Sopenharmony_ci				    < sizeof(dsp->dtmf.digits)) {
30062306a36Sopenharmony_ci					dsp->dtmf.digits[strlen(
30162306a36Sopenharmony_ci							dsp->dtmf.digits) + 1] = '\0';
30262306a36Sopenharmony_ci					dsp->dtmf.digits[strlen(
30362306a36Sopenharmony_ci							dsp->dtmf.digits)] = what;
30462306a36Sopenharmony_ci				}
30562306a36Sopenharmony_ci			}
30662306a36Sopenharmony_ci		}
30762306a36Sopenharmony_ci	} else
30862306a36Sopenharmony_ci		dsp->dtmf.count++;
30962306a36Sopenharmony_ci
31062306a36Sopenharmony_ci	dsp->dtmf.lastwhat = what;
31162306a36Sopenharmony_ci
31262306a36Sopenharmony_ci	goto again;
31362306a36Sopenharmony_ci}
314