📄 dtmfs.c
字号:
ac1 >>= 16; ac0 += ac1 * ac1;
*psOut++ = dtmf_en2logS(ac0);;
}
}
/*-------------------------------------------------------------------------*/
static void freq_lo
/*-------------------------------------------------------------------------*/
(
DTMF_tDb *pDb,
DTMF_tSc *pSc
)
{
S16 *psDev = pSc->asLoDevEn;
const S16 *psDevFlt = &__aasLoDevFlt[pSc->sLoFreqNo][0];
S32 ac0;
ac0 = psDev[3]-psDev[0];
ac0 += psDev[2]-psDev[1];
ac0 *= psDevFlt[0];
ac0 >>= 16;
/* limit to be meaningfull, 5% is the max that could be expected */
if (ac0 > 170*5) ac0 = 170*5;
if (ac0 < -170*5) ac0 = -170*5;
pSc->sLoFreqDev = (S16)(ac0);
ac0 = (ac0)*(ac0);
ac0 = ac0 * psDevFlt[1];
ac0 >>= 16;
pSc->sLoMaxFreqEn = pSc->asLoMainEn[pSc->sLoFreqNo] + (S16)(ac0);
}
/*-------------------------------------------------------------------------*/
static void dft_hi_main
/*-------------------------------------------------------------------------*/
(
DTMF_tDb *pDb,
DTMF_tSc *pSc
)
{
S32 ac0;
S32 ac1;
const S16 *psCDP = dtmf_aHi;
S16 *psI = pSc->asHiData;
S16 *psQ = psI + 1;
S16 *psOut = pSc->asHiMainEn;
S16 sFilterNo;
S16 k;
for (sFilterNo = 0; sFilterNo < 4; sFilterNo++)
{
ac0 = 0;
ac1 = 0;
/* psCDP points to Cos coeffs */
for (k = 0; k < DTMF_HI_DFT_SZ; k++)
{
ac0 += *psCDP * (S32) *psI;
ac1 += *psCDP * (S32) *psQ;
psI += 2;
psQ += 2;
psCDP++;
}
/* psCDP points to Sin coeffs */
/* return data pointers back */
psI -= DTMF_HI_DFT_SZ*2;
psQ -= DTMF_HI_DFT_SZ*2;
for (k = 0; k < DTMF_HI_DFT_SZ; k++)
{
ac1 += *psCDP * (S32) *psI;
ac0 -= *psCDP * (S32) *psQ;
psI += 2;
psQ += 2;
psCDP++;
}
/* return data pointers back */
psI -= DTMF_HI_DFT_SZ*2;
psQ -= DTMF_HI_DFT_SZ*2;
/* psCDP points to Cos coeffs of the next filter */
/* ac0 = I * c - Q * s;
ac1 = I * s + Q * c;
*/
ac0 >>= 16; ac0 = ac0 * ac0;
ac1 >>= 16; ac0 += ac1 * ac1;
*psOut++ = dtmf_en2logS(ac0);;
}
}
/*-------------------------------------------------------------------------*/
static void sort_hi
/*-------------------------------------------------------------------------*/
(
DTMF_tDb *pDb,
DTMF_tSc *pSc
)
{
S16 *psIn = pSc->asHiMainEn;
S16 sFreqNo = 0;
S16 sMaxFreqEn = psIn[0];
S16 k;
for (k = 1; k < 4; k++)
{
if (psIn[k] > sMaxFreqEn)
{
sMaxFreqEn = psIn[k];
sFreqNo = k;
}
}
sMaxFreqEn = -32767;
for (k = 0; k < 4; k++)
{
if (k != sFreqNo)
{
if (psIn[k] > sMaxFreqEn)
{
sMaxFreqEn = psIn[k];
}
}
}
pSc->sHiNextEn = sMaxFreqEn;
pSc->sHiFreqNo = sFreqNo;
}
/*-------------------------------------------------------------------------*/
static void dft_hi_dev
/*-------------------------------------------------------------------------*/
(
DTMF_tDb *pDb,
DTMF_tSc *pSc
)
{
S32 ac0;
S32 ac1;
const S16 *psCDP = __apsHiDev[pSc->sHiFreqNo];
S16 *psI = pSc->asHiData;
S16 *psQ = psI + 1;
S16 *psOut = pSc->asHiDevEn;
S16 sFilterNo;
S16 k;
for (sFilterNo = 0; sFilterNo < 4; sFilterNo++)
{
ac0 = 0;
ac1 = 0;
/* psCDP points to Cos coeffs */
for (k = 0; k < DTMF_HI_DFT_SZ; k++)
{
ac0 += *psCDP * (S32) *psI;
ac1 += *psCDP * (S32) *psQ;
psI += 2;
psQ += 2;
psCDP++;
}
/* psCDP points to Sin coeffs */
/* return data pointers back */
psI -= DTMF_HI_DFT_SZ*2;
psQ -= DTMF_HI_DFT_SZ*2;
for (k = 0; k < DTMF_HI_DFT_SZ; k++)
{
ac1 += *psCDP * (S32) *psI;
ac0 -= *psCDP * (S32) *psQ;
psI += 2;
psQ += 2;
psCDP++;
}
/* return data pointers back */
psI -= DTMF_HI_DFT_SZ*2;
psQ -= DTMF_HI_DFT_SZ*2;
/* psCDP points to Cos coeffs of the next filter */
/* ac0 = I * c - Q * s;
ac1 = I * s + Q * c;
*/
ac0 >>= 16; ac0 = ac0 * ac0;
ac1 >>= 16; ac0 += ac1 * ac1;
*psOut++ = dtmf_en2logS(ac0);;
}
}
/*-------------------------------------------------------------------------*/
static void freq_hi
/*-------------------------------------------------------------------------*/
(
DTMF_tDb *pDb,
DTMF_tSc *pSc
)
{
S32 ac0;
S16 *psDev = pSc->asHiDevEn;
const S16 *psDevFlt = &__aasHiDevFlt[pSc->sHiFreqNo][0];
ac0 = psDev[3]-psDev[0];
ac0 += psDev[2]-psDev[1];
ac0 *= psDevFlt[0];
ac0 >>= 16;
/* limit to be meaningfull, 5% is the max that could be expected */
if (ac0 > 170*5) ac0 = 170*5;
if (ac0 < -170*5) ac0 = -170*5;
pSc->sHiFreqDev = (S16)(ac0);
ac0 = (ac0)*(ac0);
ac0 = ac0 * psDevFlt[1];
ac0 >>= 16;
pSc->sHiMaxFreqEn = pSc->asHiMainEn[pSc->sHiFreqNo] + (S16)(ac0);
}
/*-------------------------------------------------------------------------*/
static void frame_en
/*-------------------------------------------------------------------------*/
(
DTMF_tDb *pDb,
DTMF_tSc *pSc
)
{
#if (IDTMF_FR_SZ == 40)
S32 ac0 = 0;
S16 *ps0 = &pSc->as2kData[0];
S16 Sample;
for (Sample = 0; Sample < IDTMF_FR_SZ; Sample++)
{
ac0 += (*ps0 * (long)*ps0);
ps0 += 1;
}
pSc->slFrameEn = (ac0);
#else
S32 ac0 = 0;
S32 ac1 = 0;
S16 *ps0 = &pSc->as2kData[0];
S16 *ps1 = &pSc->as2kData[IDTMF_FR_SZ/2];
S16 Sample;
for (Sample = 0; Sample < IDTMF_FR_SZ/2; Sample++)
{
ac0 += (*ps0 * (long)*ps0);
ac1 += (*ps1 * (long)*ps1);
ps0 += 1;
ps1 += 1;
}
ac0 >>= 3;
ac1 >>= 3;
ac0 += ac1;
pSc->sFrameEn = dtmf_en2logS(ac0);
#endif
}
/*-------------------------------------------------------------------------*/
static void test
/*-------------------------------------------------------------------------*/
(
DTMF_tDb *pDb,
DTMF_tSc *pSc
)
{
const IDTMF_tCfg *pCfg = pDb->pCfg;
S16 sTestFailed = 0;
/* test that frame energy is high enough. */
if (pSc->sSumEn < pCfg->sMinEnThr)
sTestFailed |= DTMF_MASK_MIN_EN;
#if (IDTMF_FR_SZ == 40)
#if ! defined(_dsp)
DBG_sFrameEn = pDb->v.sZFrameEn;
#endif
/* test that {dial + lo + hi} energies constitute
a major part of average of 2 frames energy. */
if ((pDb->v.sZFrameEn - pSc->sSumDialEn) > pCfg->sSumEnThr)
sTestFailed |= DTMF_MASK_SUM_EN;
#else
if ((pSc->sFrameEn - pSc->sSumDialEn) > pCfg->sSumEnThr)
sTestFailed |= DTMF_MASK_SUM_EN;
#endif
/* test leading edge */
{
#if (IDTMF_FR_SZ == 40)
S16 ac0 = pDb->v.asSumEn[0];
S16 ac1 = pDb->v.asSumEn[1];
S16 ac2 = pDb->v.asSumEn[4];
S16 ac3 = pDb->v.asSumEn[5];
if (ac0 > ac1) ac0 = ac1; /* ac0 = min of 2 most recent frames */
if (ac2 < ac3) ac2 = ac3; /* ac2 = max of 2 oldest frames */
ac0 -= ac2; /* energy diff */
#else
S16 ac0 = pDb->v.asSumEn[0];
S16 ac1 = pDb->v.asSumEn[2];
ac0 -= ac1;
#endif
if (ac0 < pCfg->sNoiseThr)
sTestFailed |= DTMF_MASK_RS_EDGE;
}
/* test falling edge */
if ((pDb->v.Stat.sEn - pSc->sSumEn) < pCfg->sNoiseThr)
sTestFailed |= DTMF_MASK_FL_EDGE;
/* test that lo(idx) and hi(idx) energies are reasonably stable */
if (abs(pSc->sLoMaxFreqEn - pDb->v.Stat.sLoEn) > pCfg->sVarThr)
sTestFailed |= DTMF_MASK_STABLE;
if (abs(pSc->sHiMaxFreqEn - pDb->v.Stat.sHiEn) > pCfg->sVarThr)
sTestFailed |= DTMF_MASK_STABLE;
/* test that <2k energy is significantly bigger than >2k energy */
if (pSc->s2kEnDlt < pCfg->s2kUpThr)
sTestFailed |= DTMF_MASK_2K_BOUND;
/* test that twists are ok */
/* if ((pSc->Hi.sMaxFreqEn - pSc->Lo.sMaxFreqEn) > pCfg->sFwdTwistThr) */
if ((pSc->sHiBpEn-pSc->sLoBpEn) > (pCfg->sFwdTwistThr+(170+85)))
sTestFailed |= DTMF_MASK_FWD_TWIST;
if ((pSc->sHiBpEn-pSc->sLoBpEn) < -(pCfg->sRevTwistThr-85) )
sTestFailed |= DTMF_MASK_REV_TWIST;
if (pDb->v.sState < DTMF_ST_TONE)
{
/* DFT delay lines are not full. at least one frame less.
we shall account for wider sinc mainlobe and lower energy.
Noise does not have much influence on the parameters
in this case, so the threshold are hard coded.
tuning criteria was chosen the number of early-on hits
on talk-off imunity test.
varying those thrs shall affect early detection delay
and talk-off immunity as well.
*/
if ((pSc->sLoBpEn - pSc->sLoMaxFreqEn) > (6*IDTMF_1DB))
sTestFailed |= DTMF_MASK_LO_PART;
if ((pSc->sLoMaxFreqEn - pSc->sLoNextEn) < (5*IDTMF_1DB))
sTestFailed |= DTMF_MASK_LO_CLEAN;
#if (IDTMF_FR_SZ == 40)
if ((pSc->sHiBpEn - pSc->sHiMaxFreqEn) > (4*IDTMF_1DB))
sTestFailed |= DTMF_MASK_HI_PART;
if ((pSc->sHiMaxFreqEn - pSc->sHiNextEn) < (7*IDTMF_1DB))
sTestFailed |= DTMF_MASK_HI_CLEAN;
#endif
}
else
{
if ((pSc->sLoBpEn - pSc->sLoMaxFreqEn) > pCfg->sPartThr)
sTestFailed |= DTMF_MASK_LO_PART;
if ((pSc->sLoMaxFreqEn - pSc->sLoNextEn) < pCfg->sCleanThr)
sTestFailed |= DTMF_MASK_LO_CLEAN;
#if (IDTMF_FR_SZ == 40)
if ((pSc->sHiBpEn - pSc->sHiMaxFreqEn) > pCfg->sPartThr)
sTestFailed |= DTMF_MASK_HI_PART;
if ((pSc->sHiMaxFreqEn - pSc->sHiNextEn) < pCfg->sCleanThr)
sTestFailed |= DTMF_MASK_HI_CLEAN;
#endif
}
#if (IDTMF_FR_SZ == 80)
if ((pSc->sHiBpEn - pSc->sHiMaxFreqEn) > pCfg->sPartThr)
sTestFailed |= DTMF_MASK_HI_PART;
if ((pSc->sHiMaxFreqEn - pSc->sHiNextEn) < pCfg->sCleanThr)
sTestFailed |= DTMF_MASK_HI_CLEAN;
#endif
if (abs(pSc->sLoFreqDev) > pCfg->sMaxFreqDevThr)
sTestFailed |= DTMF_MASK_LO_INRANGE;
if (abs(pSc->sHiFreqDev) > pCfg->sMaxFreqDevThr)
sTestFailed |= DTMF_MASK_HI_INRANGE;
pSc->sDigit = pSc->sLoFreqNo | (pSc->sHiFreqNo<<2);
if (pDb->v.sLastDigit != pSc->sDigit)
sTestFailed |= DTMF_MASK_SAME_FRQ;
pSc->sTestFailed = sTestFailed;
}
/*-------------------------------------------------------------------------*/
static void dtmf_move_scratch_to_dbC
/*-------------------------------------------------------------------------*/
(
DTMF_tDb *pDb,
DTMF_tSc *pSc
)
{
S16 k;
#if (IDTMF_FR_SZ == 40)
{
S32 slTmp = pSc->slFrameEn + pDb->v.slFrameEn;
S16 sTmp = dtmf_en2logS(slTmp>>3);
pDb->v.sZFrameEn = sTmp;
}
/* required for next frame */
pDb->v.slFrameEn = pSc->slFrameEn;
#else
#endif
for (k = 0; k < DTMF_2KFLT_SZ; k++)
{
pDb->as2kDataSav[k] = pSc->as2kData[k+IDTMF_FR_SZ];
}
for (k = 0; k < DTMF_BPFLT_SZ; k++)
{
pDb->asBpDataSav[k] = pSc->asBpData[k+DTMF_BPFR_SZ];
}
for (k = 0; k < DTMF_LOSD_SZ; k++)
{
pDb->asLoDataSav[k] = pSc->asLoData[k+DTMF_DFR_SZ*2];
}
#if (IDTMF_FR_SZ != 80)
for (k = 0; k < DTMF_HISD_SZ; k++)
{
pDb->asHiDataSav[k] = pSc->asHiData[k+DTMF_DFR_SZ*2];
}
#endif
}
/*-------------------------------------------------------------------------*/
void dtmf_filter_testC
/*-------------------------------------------------------------------------*/
(
DTMF_tDb *pDb,
DTMF_tSc *pSc,
S16 *psIn
)
{
move_data(pDb, pSc, psIn);
filter_2k(pDb, pSc);
filter_lo(pDb, pSc);
filter_hi(pDb, pSc);
pSc->sSumEn = dtmf_en2logS(pSc->slLoBpEn + pSc->slHiBpEn);
pDb->v.asSumEn[0] = pSc->sSumEn;
filter_dial(pDb, pSc);
dft_lo_main(pDb, pSc);
sort_lo(pDb, pSc);
dft_lo_dev(pDb, pSc);
freq_lo(pDb, pSc);
dft_hi_main(pDb, pSc);
sort_hi(pDb, pSc);
dft_hi_dev(pDb, pSc);
freq_hi(pDb, pSc);
frame_en(pDb, pSc);
test(pDb, pSc);
dtmf_move_scratch_to_dbC (pDb, pSc);
}
#if ! defined (_dsp)
/*-------------------------------------------------------------------------*/
S16 dtmf_en2log
/*-------------------------------------------------------------------------*/
(
S32 ac0 /* but it is positive */
)
{
S16 t0 = 0;
ac0 += 1; // min val -13952
// mant aco :: nexp ac0, t0
for (;;)
{
if (ac0 &0x40000000L)
break;
ac0 <<= 1;
t0--;
}
ac0 >>= 21;
ac0 += 0x780;
ac0 += t0*512;
return (S16)ac0;
}
#endif
/*-------------------------------------------------------------------------*/
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -