1999-03-07 17:09:03 +00:00
|
|
|
/*
|
|
|
|
* ----------------------------------------------------------------------------
|
|
|
|
* "THE BEER-WARE LICENSE" (Revision 42):
|
|
|
|
* <phk@FreeBSD.org> wrote this file. As long as you retain this notice you
|
|
|
|
* can do whatever you want with this stuff. If we meet some day, and you think
|
|
|
|
* this stuff is worth it, you can buy me a beer in return. Poul-Henning Kamp
|
|
|
|
* ----------------------------------------------------------------------------
|
|
|
|
*
|
1999-12-14 21:14:28 +00:00
|
|
|
* $Id: dtmfdecode.c,v 1.6 1999/12/13 21:25:24 hm Exp $
|
|
|
|
*
|
1999-08-28 01:35:59 +00:00
|
|
|
* $FreeBSD$
|
1999-03-07 17:09:03 +00:00
|
|
|
*
|
1999-05-20 10:14:57 +00:00
|
|
|
* Extract DTMF signalling from ISDN4BSD A-law coded audio data
|
1999-03-07 17:09:03 +00:00
|
|
|
*
|
|
|
|
* A-Law to linear conversion from the sox package.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <math.h>
|
|
|
|
|
1999-05-20 10:14:57 +00:00
|
|
|
/* Integer math scaling factor */
|
|
|
|
#define FSC (1<<12)
|
|
|
|
|
|
|
|
/* Alaw parameters */
|
2003-01-01 18:49:04 +00:00
|
|
|
#define SIGN_BIT (0x80) /* Sign bit for an A-law byte. */
|
1999-03-07 17:09:03 +00:00
|
|
|
#define QUANT_MASK (0xf) /* Quantization field mask. */
|
|
|
|
#define SEG_SHIFT (4) /* Left shift for segment number. */
|
|
|
|
#define SEG_MASK (0x70) /* Segment field mask. */
|
|
|
|
|
1999-05-20 10:14:57 +00:00
|
|
|
static int
|
1999-03-07 17:09:03 +00:00
|
|
|
alaw2linear(a_val)
|
|
|
|
unsigned char a_val;
|
|
|
|
{
|
|
|
|
int t;
|
|
|
|
int seg;
|
|
|
|
|
|
|
|
a_val ^= 0x55;
|
|
|
|
|
|
|
|
t = (a_val & QUANT_MASK) << 4;
|
|
|
|
seg = ((unsigned)a_val & SEG_MASK) >> SEG_SHIFT;
|
|
|
|
switch (seg) {
|
|
|
|
case 0:
|
|
|
|
t += 8;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
t += 0x108;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
t += 0x108;
|
|
|
|
t <<= seg - 1;
|
|
|
|
}
|
|
|
|
return ((a_val & SIGN_BIT) ? t : -t);
|
|
|
|
}
|
|
|
|
|
1999-05-20 10:14:57 +00:00
|
|
|
#ifdef USE_COS
|
|
|
|
/* The frequencies we're trying to detect */
|
|
|
|
static int dtmf[8] = {697, 770, 852, 941, 1209, 1336, 1477, 1633};
|
|
|
|
#else
|
|
|
|
/* precalculated: p1[kk] = (-cos(2 * 3.141592 * dtmf[kk] / 8000.0) * FSC) */
|
|
|
|
static int p1[8] = {-3497, -3369, -3212, -3027, -2384, -2040, -1635, -1164};
|
|
|
|
#endif
|
1999-03-07 17:09:03 +00:00
|
|
|
|
1999-05-20 10:14:57 +00:00
|
|
|
/* This is the Q of the filter (pole radius) */
|
1999-03-07 17:09:03 +00:00
|
|
|
#define POLRAD .99
|
|
|
|
|
1999-05-20 10:14:57 +00:00
|
|
|
#define P2 ((int)(POLRAD*POLRAD*FSC))
|
1999-03-07 17:09:03 +00:00
|
|
|
|
|
|
|
int
|
|
|
|
main(int argc, char **argv)
|
|
|
|
{
|
1999-05-20 10:14:57 +00:00
|
|
|
int i, kk, t, nn, s, so, ia;
|
|
|
|
int x, c, d, f, h[8], k[8], n, y[8];
|
|
|
|
#ifdef USE_COS
|
|
|
|
int p1[8];
|
|
|
|
#endif
|
|
|
|
int alaw[256];
|
|
|
|
char key[256];
|
1999-03-07 17:09:03 +00:00
|
|
|
|
|
|
|
for (kk = 0; kk < 8; kk++) {
|
1999-05-20 10:14:57 +00:00
|
|
|
y[kk] = h[kk] = k[kk] = 0;
|
|
|
|
#ifdef USE_COS
|
|
|
|
p1[kk] = (-cos(2 * 3.141592 * dtmf[kk] / 8000.0) * FSC);
|
|
|
|
#endif
|
1999-03-07 17:09:03 +00:00
|
|
|
}
|
|
|
|
|
1999-05-20 10:14:57 +00:00
|
|
|
for (i = 0; i < 256; i++) {
|
|
|
|
key[i] = '?';
|
|
|
|
alaw[i] = alaw2linear(i) / (32768/FSC);
|
1999-03-07 17:09:03 +00:00
|
|
|
}
|
|
|
|
|
1999-05-20 10:14:57 +00:00
|
|
|
/* We encode the tones in 8 bits, translate those to symbol */
|
1999-03-07 17:09:03 +00:00
|
|
|
key[0x00] = '\0';
|
1999-05-20 10:14:57 +00:00
|
|
|
|
|
|
|
key[0x11] = '1'; key[0x12] = '4'; key[0x14] = '7'; key[0x18] = '*';
|
|
|
|
key[0x21] = '2'; key[0x22] = '5'; key[0x24] = '8'; key[0x28] = '0';
|
|
|
|
key[0x41] = '3'; key[0x42] = '6'; key[0x44] = '9'; key[0x48] = '#';
|
|
|
|
key[0x81] = 'A'; key[0x82] = 'B'; key[0x84] = 'C'; key[0x88] = 'D';
|
|
|
|
|
1999-03-07 17:09:03 +00:00
|
|
|
nn = 0;
|
1999-05-20 10:14:57 +00:00
|
|
|
ia = 0;
|
|
|
|
so = 0;
|
|
|
|
t = 0;
|
|
|
|
while ((i = getchar()) != EOF)
|
|
|
|
{
|
|
|
|
t++;
|
|
|
|
|
|
|
|
/* Convert to our format */
|
|
|
|
x = alaw[i];
|
|
|
|
|
|
|
|
/* Input amplitude */
|
|
|
|
if (x > 0)
|
|
|
|
ia += (x - ia) / 128;
|
|
|
|
else
|
|
|
|
ia += (-x - ia) / 128;
|
1999-03-07 17:09:03 +00:00
|
|
|
|
1999-05-20 10:14:57 +00:00
|
|
|
/* For each tone */
|
1999-03-07 17:09:03 +00:00
|
|
|
s = 0;
|
|
|
|
for(kk = 0; kk < 8; kk++) {
|
1999-05-20 10:14:57 +00:00
|
|
|
|
|
|
|
/* Turn the crank */
|
|
|
|
c = (P2 * (x - k[kk])) / FSC;
|
|
|
|
d = x + c;
|
|
|
|
f = (p1[kk] * (d - h[kk])) / FSC;
|
|
|
|
n = x - k[kk] - c;
|
|
|
|
k[kk] = h[kk] + f;
|
|
|
|
h[kk] = f + d;
|
|
|
|
|
|
|
|
/* Detect and Average */
|
|
|
|
if (n > 0)
|
|
|
|
y[kk] += (n - y[kk]) / 64;
|
|
|
|
else
|
|
|
|
y[kk] += (-n - y[kk]) / 64;
|
|
|
|
|
|
|
|
/* Threshold */
|
|
|
|
if (y[kk] > FSC/10 && y[kk] > ia)
|
1999-03-07 17:09:03 +00:00
|
|
|
s |= 1 << kk;
|
|
|
|
}
|
1999-05-20 10:14:57 +00:00
|
|
|
|
|
|
|
/* Hysteresis and noise supressor */
|
|
|
|
if (s != so) {
|
|
|
|
/* printf("x %d %x -> %x\n",t,so, s); */
|
1999-03-07 17:09:03 +00:00
|
|
|
nn = 0;
|
1999-05-20 10:14:57 +00:00
|
|
|
so = s;
|
|
|
|
} else if (nn++ == 520 && key[s]) {
|
|
|
|
putchar(key[s]);
|
|
|
|
/* printf(" %d %x\n",t,s); */
|
1999-03-07 17:09:03 +00:00
|
|
|
}
|
|
|
|
}
|
1999-05-20 10:14:57 +00:00
|
|
|
putchar('\n');
|
1999-03-07 17:09:03 +00:00
|
|
|
return (0);
|
|
|
|
}
|