freeswitch/libs/ilbc/src/iLBC_decode.c

651 lines
22 KiB
C
Raw Normal View History

/*
* iLBC - a library for the iLBC codec
*
* iLBC_decode.c - The iLBC low bit rate speech codec.
*
* Adapted by Steve Underwood <steveu@coppice.org> from the reference
* iLBC code supplied in RFC3951.
*
* Original code Copyright (C) The Internet Society (2004).
* All changes to produce this version Copyright (C) 2008 by Steve Underwood
* All Rights Reserved.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
*
* $Id: iLBC_decode.c,v 1.2 2008/03/06 12:27:38 steveu Exp $
*/
/*! \file */
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <inttypes.h>
#include <math.h>
#include <stdlib.h>
#include "ilbc.h"
#include "StateConstructW.h"
#include "LPCdecode.h"
#include "iCBConstruct.h"
#include "doCPLC.h"
#include "helpfun.h"
#include "constants.h"
#include "packing.h"
#include "string.h"
#include "enhancer.h"
#include "hpOutput.h"
#include "syntFilter.h"
2013-11-30 04:16:49 +00:00
#if (defined(WIN32) || defined(_WIN32)) && (_MSC_VER < 1800)
#if (defined(WIN32) || defined(_WIN32)) && !defined(_WIN64)
__inline long int rint(double dbl)
{
_asm
{
fld dbl
frndint
2011-02-04 05:46:19 +00:00
fstp dbl
}
2011-02-04 05:46:19 +00:00
return (long int) dbl;
}
#elif defined (_WIN64)
#include <intrin.h>
__inline__ long int rint(double x)
{
#ifdef _M_X64
return (long int)_mm_cvtsd_si64x( _mm_loadu_pd ((const double*)&x) );
#else
#warning "Not Supported: Replacing with a simple C cast."
return (long int) (x);
#endif
}
#endif
2013-11-30 04:16:49 +00:00
#endif
/*----------------------------------------------------------------*
* Initiation of decoder instance.
*---------------------------------------------------------------*/
ilbc_decode_state_t *ilbc_decode_init(ilbc_decode_state_t *iLBCdec_inst, /* (i/o) Decoder instance */
int mode, /* (i) frame size mode */
int use_enhancer) /* (i) 1 to use enhancer
0 to run without enhancer */
{
int i;
iLBCdec_inst->mode = mode;
if (mode == 30)
{
iLBCdec_inst->blockl = ILBC_BLOCK_LEN_30MS;
iLBCdec_inst->nsub = NSUB_30MS;
iLBCdec_inst->nasub = NASUB_30MS;
iLBCdec_inst->lpc_n = LPC_N_30MS;
iLBCdec_inst->no_of_bytes = ILBC_NO_OF_BYTES_30MS;
iLBCdec_inst->state_short_len = STATE_SHORT_LEN_30MS;
/* ULP init */
iLBCdec_inst->ULP_inst = &ULP_30msTbl;
}
else if (mode == 20)
{
iLBCdec_inst->blockl = ILBC_BLOCK_LEN_20MS;
iLBCdec_inst->nsub = NSUB_20MS;
iLBCdec_inst->nasub = NASUB_20MS;
iLBCdec_inst->lpc_n = LPC_N_20MS;
iLBCdec_inst->no_of_bytes = ILBC_NO_OF_BYTES_20MS;
iLBCdec_inst->state_short_len = STATE_SHORT_LEN_20MS;
/* ULP init */
iLBCdec_inst->ULP_inst = &ULP_20msTbl;
}
else
{
return NULL;
}
memset(iLBCdec_inst->syntMem, 0, ILBC_LPC_FILTERORDER*sizeof(float));
memcpy((*iLBCdec_inst).lsfdeqold, lsfmeanTbl, ILBC_LPC_FILTERORDER*sizeof(float));
memset(iLBCdec_inst->old_syntdenum,
0,
((ILBC_LPC_FILTERORDER + 1)*ILBC_NUM_SUB_MAX)*sizeof(float));
for (i = 0; i < ILBC_NUM_SUB_MAX; i++)
iLBCdec_inst->old_syntdenum[i*(ILBC_LPC_FILTERORDER + 1)] = 1.0f;
iLBCdec_inst->last_lag = 20;
iLBCdec_inst->prevLag = 120;
iLBCdec_inst->per = 0.0;
iLBCdec_inst->consPLICount = 0;
iLBCdec_inst->prevPLI = 0;
iLBCdec_inst->prevLpc[0] = 1.0f;
memset(iLBCdec_inst->prevLpc + 1, 0, ILBC_LPC_FILTERORDER*sizeof(float));
memset(iLBCdec_inst->prevResidual, 0, ILBC_BLOCK_LEN_MAX*sizeof(float));
iLBCdec_inst->seed = 777;
memset(iLBCdec_inst->hpomem, 0, 4*sizeof(float));
iLBCdec_inst->use_enhancer = use_enhancer;
memset(iLBCdec_inst->enh_buf, 0, ENH_BUFL*sizeof(float));
for (i = 0; i < ENH_NBLOCKS_TOT; i++)
iLBCdec_inst->enh_period[i] = 40.0f;
iLBCdec_inst->prev_enh_pl = 0;
return iLBCdec_inst;
}
/*----------------------------------------------------------------*
* frame residual decoder function (subrutine to iLBC_decode)
*---------------------------------------------------------------*/
static void Decode(ilbc_decode_state_t *iLBCdec_inst, /* (i/o) the decoder state structure */
float *decresidual, /* (o) decoded residual frame */
int start, /* (i) location of start state */
int idxForMax, /* (i) codebook index for the maximum value */
int *idxVec, /* (i) codebook indexes for the samples in the
start state */
float *syntdenum, /* (i) the decoded synthesis filter coefficients */
int *cb_index, /* (i) the indexes for the adaptive codebook */
int *gain_index, /* (i) the indexes for the corresponding gains */
int *extra_cb_index, /* (i) the indexes for the adaptive codebook part
of start state */
int *extra_gain_index, /* (i) the indexes for the corresponding gains */
int state_first) /* (i) 1 if non adaptive part of start state comes
first. 0 if that part comes last */
{
float reverseDecresidual[ILBC_BLOCK_LEN_MAX];
float mem[CB_MEML];
int k;
int meml_gotten;
int Nfor;
int Nback;
int i;
int diff;
int start_pos;
int subcount;
int subframe;
diff = STATE_LEN - iLBCdec_inst->state_short_len;
if (state_first == 1)
start_pos = (start - 1)*SUBL;
else
start_pos = (start - 1)*SUBL + diff;
/* decode scalar part of start state */
StateConstructW(idxForMax,
idxVec,
&syntdenum[(start - 1)*(ILBC_LPC_FILTERORDER + 1)],
&decresidual[start_pos],
iLBCdec_inst->state_short_len);
if (state_first)
{
/* put adaptive part in the end */
/* setup memory */
memset(mem, 0, (CB_MEML - iLBCdec_inst->state_short_len)*sizeof(float));
memcpy(&mem[CB_MEML - iLBCdec_inst->state_short_len],
decresidual + start_pos,
iLBCdec_inst->state_short_len*sizeof(float));
/* construct decoded vector */
iCBConstruct(&decresidual[start_pos+iLBCdec_inst->state_short_len],
extra_cb_index, extra_gain_index, mem+CB_MEML-stMemLTbl,
stMemLTbl,
diff,
CB_NSTAGES);
}
else
{
/* Put adaptive part in the beginning */
/* Create reversed vectors for prediction */
for (k = 0; k < diff; k++)
{
reverseDecresidual[k] =
decresidual[(start + 1)*SUBL - 1 - (k + iLBCdec_inst->state_short_len)];
}
/* Setup memory */
meml_gotten = iLBCdec_inst->state_short_len;
for (k = 0; k < meml_gotten; k++)
mem[CB_MEML - 1 - k] = decresidual[start_pos + k];
memset(mem, 0, (CB_MEML - k)*sizeof(float));
/* Construct decoded vector */
iCBConstruct(reverseDecresidual, extra_cb_index,
extra_gain_index,
mem + CB_MEML - stMemLTbl,
stMemLTbl,
diff,
CB_NSTAGES);
/* Get decoded residual from reversed vector */
for (k = 0; k < diff; k++)
decresidual[start_pos - 1 - k] = reverseDecresidual[k];
}
/* Counter for predicted sub-frames */
subcount = 0;
/* Forward prediction of sub-frames */
Nfor = iLBCdec_inst->nsub-start - 1;
if (Nfor > 0)
{
/* Setup memory */
memset(mem, 0, (CB_MEML - STATE_LEN)*sizeof(float));
memcpy(&mem[CB_MEML - STATE_LEN], decresidual + (start - 1)*SUBL, STATE_LEN*sizeof(float));
/* Loop over sub-frames to encode */
for (subframe = 0; subframe < Nfor; subframe++)
{
/* Construct decoded vector */
iCBConstruct(&decresidual[(start + 1 + subframe)*SUBL],
cb_index+subcount*CB_NSTAGES,
gain_index+subcount*CB_NSTAGES,
mem + CB_MEML - memLfTbl[subcount],
memLfTbl[subcount],
SUBL,
CB_NSTAGES);
/* Update memory */
memmove(mem, &mem[SUBL], (CB_MEML - SUBL)*sizeof(float));
memmove(&mem[CB_MEML - SUBL],
&decresidual[(start + 1 + subframe)*SUBL],
SUBL*sizeof(float));
subcount++;
}
}
/* Backward prediction of sub-frames */
Nback = start - 1;
if (Nback > 0)
{
/* Setup memory */
meml_gotten = SUBL*(iLBCdec_inst->nsub + 1 - start);
if (meml_gotten > CB_MEML)
meml_gotten = CB_MEML;
for (k = 0; k < meml_gotten; k++)
mem[CB_MEML - 1 - k] = decresidual[(start - 1)*SUBL + k];
memset(mem, 0, (CB_MEML - k)*sizeof(float));
/* Loop over subframes to decode */
for (subframe = 0; subframe < Nback; subframe++)
{
/* Construct decoded vector */
iCBConstruct(&reverseDecresidual[subframe*SUBL],
cb_index + subcount*CB_NSTAGES,
gain_index + subcount*CB_NSTAGES,
mem + CB_MEML - memLfTbl[subcount],
memLfTbl[subcount],
SUBL,
CB_NSTAGES);
/* Update memory */
memmove(mem, &mem[SUBL], (CB_MEML - SUBL)*sizeof(float));
memmove(&mem[CB_MEML - SUBL],
&reverseDecresidual[subframe*SUBL],
SUBL*sizeof(float));
subcount++;
}
/* Get decoded residual from reversed vector */
for (i = 0; i < SUBL*Nback; i++)
decresidual[SUBL*Nback - i - 1] = reverseDecresidual[i];
}
}
/*----------------------------------------------------------------*
* main decoder function
*---------------------------------------------------------------*/
static void ilbc_decode_frame(ilbc_decode_state_t *iLBCdec_inst, /* (i/o) the decoder state structure */
float decblock[], /* (o) decoded signal block */
const uint8_t bytes[], /* (i) encoded signal bits */
int mode) /* (i) 0: bad packet, PLC, 1: normal */
{
float data[ILBC_BLOCK_LEN_MAX];
float lsfdeq[ILBC_LPC_FILTERORDER*LPC_N_MAX];
float PLCresidual[ILBC_BLOCK_LEN_MAX];
float PLClpc[ILBC_LPC_FILTERORDER + 1];
float zeros[ILBC_BLOCK_LEN_MAX];
float one[ILBC_LPC_FILTERORDER + 1];
int k;
int i;
int start;
int idxForMax;
int pos;
int lastpart;
int ulp;
int lag;
int ilag;
float cc;
float maxcc;
int idxVec[STATE_LEN];
int gain_index[NASUB_MAX*CB_NSTAGES];
int extra_gain_index[CB_NSTAGES];
int cb_index[CB_NSTAGES*NASUB_MAX];
int extra_cb_index[CB_NSTAGES];
int lsf_i[LSF_NSPLIT*LPC_N_MAX];
int state_first;
int last_bit;
const uint8_t *pbytes;
float weightdenum[(ILBC_LPC_FILTERORDER + 1)*ILBC_NUM_SUB_MAX];
int order_plus_one;
float syntdenum[ILBC_NUM_SUB_MAX*(ILBC_LPC_FILTERORDER + 1)];
float decresidual[ILBC_BLOCK_LEN_MAX];
if (mode > 0)
{
/* The data is good. decode it. */
pbytes = bytes;
pos = 0;
/* Set everything to zero before decoding */
for (k = 0; k < LSF_NSPLIT*LPC_N_MAX; k++)
lsf_i[k] = 0;
start = 0;
state_first = 0;
idxForMax = 0;
for (k = 0; k < iLBCdec_inst->state_short_len; k++)
idxVec[k] = 0;
for (k = 0; k < CB_NSTAGES; k++)
{
extra_cb_index[k] = 0;
extra_gain_index[k] = 0;
}
for (i = 0; i < iLBCdec_inst->nasub; i++)
{
for (k = 0; k < CB_NSTAGES; k++)
{
cb_index[i*CB_NSTAGES + k] = 0;
gain_index[i*CB_NSTAGES + k] = 0;
}
}
/* Loop over ULP classes */
for (ulp = 0; ulp < 3; ulp++)
{
/* LSF */
for (k = 0; k < LSF_NSPLIT*iLBCdec_inst->lpc_n; k++)
{
unpack(&pbytes, &lastpart, iLBCdec_inst->ULP_inst->lsf_bits[k][ulp], &pos);
packcombine(&lsf_i[k], lastpart, iLBCdec_inst->ULP_inst->lsf_bits[k][ulp]);
}
/* Start block info */
unpack(&pbytes, &lastpart, iLBCdec_inst->ULP_inst->start_bits[ulp], &pos);
packcombine(&start, lastpart, iLBCdec_inst->ULP_inst->start_bits[ulp]);
unpack(&pbytes, &lastpart, iLBCdec_inst->ULP_inst->startfirst_bits[ulp], &pos);
packcombine(&state_first, lastpart, iLBCdec_inst->ULP_inst->startfirst_bits[ulp]);
unpack(&pbytes, &lastpart, iLBCdec_inst->ULP_inst->scale_bits[ulp], &pos);
packcombine(&idxForMax, lastpart, iLBCdec_inst->ULP_inst->scale_bits[ulp]);
for (k = 0; k < iLBCdec_inst->state_short_len; k++)
{
unpack(&pbytes, &lastpart, iLBCdec_inst->ULP_inst->state_bits[ulp], &pos);
packcombine(idxVec + k, lastpart, iLBCdec_inst->ULP_inst->state_bits[ulp]);
}
/* 23/22 (20ms/30ms) sample block */
for (k = 0; k < CB_NSTAGES; k++)
{
unpack(&pbytes, &lastpart, iLBCdec_inst->ULP_inst->extra_cb_index[k][ulp], &pos);
packcombine(extra_cb_index + k, lastpart, iLBCdec_inst->ULP_inst->extra_cb_index[k][ulp]);
}
for (k = 0; k < CB_NSTAGES; k++)
{
unpack(&pbytes, &lastpart, iLBCdec_inst->ULP_inst->extra_cb_gain[k][ulp], &pos);
packcombine(extra_gain_index + k, lastpart, iLBCdec_inst->ULP_inst->extra_cb_gain[k][ulp]);
}
/* The two/four (20ms/30ms) 40 sample sub-blocks */
for (i = 0; i < iLBCdec_inst->nasub; i++)
{
for (k = 0; k < CB_NSTAGES; k++)
{
unpack(&pbytes, &lastpart, iLBCdec_inst->ULP_inst->cb_index[i][k][ulp], &pos);
packcombine(cb_index + i*CB_NSTAGES + k, lastpart, iLBCdec_inst->ULP_inst->cb_index[i][k][ulp]);
}
}
for (i = 0; i < iLBCdec_inst->nasub; i++)
{
for (k = 0; k < CB_NSTAGES; k++)
{
unpack(&pbytes, &lastpart, iLBCdec_inst->ULP_inst->cb_gain[i][k][ulp], &pos);
packcombine(gain_index + i*CB_NSTAGES + k, lastpart, iLBCdec_inst->ULP_inst->cb_gain[i][k][ulp]);
}
}
}
/* Extract last bit. If it is 1 this indicates an empty/lost frame */
unpack(&pbytes, &last_bit, 1, &pos);
/* Check for bit errors or empty/lost frames */
if (start < 1)
mode = 0;
if (iLBCdec_inst->mode == 20 && start > 3)
mode = 0;
if (iLBCdec_inst->mode == 30 && start > 5)
mode = 0;
if (last_bit == 1)
mode = 0;
if (mode == 1)
{
/* No bit errors was detected, continue decoding */
/* Adjust index */
index_conv_dec(cb_index);
/* Decode the LSF */
SimplelsfDEQ(lsfdeq, lsf_i, iLBCdec_inst->lpc_n);
LSF_check(lsfdeq, ILBC_LPC_FILTERORDER, iLBCdec_inst->lpc_n);
DecoderInterpolateLSF(syntdenum, weightdenum, lsfdeq, ILBC_LPC_FILTERORDER, iLBCdec_inst);
Decode(iLBCdec_inst,
decresidual,
start,
idxForMax,
idxVec,
syntdenum,
cb_index,
gain_index,
extra_cb_index,
extra_gain_index,
state_first);
/* Preparing the plc for a future loss! */
doThePLC(PLCresidual,
PLClpc,
0,
decresidual,
syntdenum + (ILBC_LPC_FILTERORDER + 1)*(iLBCdec_inst->nsub - 1),
(*iLBCdec_inst).last_lag,
iLBCdec_inst);
memcpy(decresidual, PLCresidual, iLBCdec_inst->blockl*sizeof(float));
}
}
if (mode == 0)
{
/* The data is bad (either a PLC call
* was made or a severe bit error was detected)
*/
/* Apply packet loss concealmeant */
memset(zeros, 0, ILBC_BLOCK_LEN_MAX*sizeof(float));
one[0] = 1;
memset(one + 1, 0, ILBC_LPC_FILTERORDER*sizeof(float));
start = 0;
doThePLC(PLCresidual, PLClpc, 1, zeros, one, (*iLBCdec_inst).last_lag, iLBCdec_inst);
memcpy(decresidual, PLCresidual, iLBCdec_inst->blockl*sizeof(float));
order_plus_one = ILBC_LPC_FILTERORDER + 1;
for (i = 0; i < iLBCdec_inst->nsub; i++)
memcpy(&syntdenum[i*order_plus_one], PLClpc, order_plus_one*sizeof(float));
}
if (iLBCdec_inst->use_enhancer == 1)
{
/* Post filtering */
iLBCdec_inst->last_lag = enhancerInterface(data, decresidual, iLBCdec_inst);
/* Synthesis filtering */
if (iLBCdec_inst->mode == 20)
{
/* Enhancer has 40 samples delay */
i = 0;
syntFilter(data + i*SUBL,
iLBCdec_inst->old_syntdenum + (i + iLBCdec_inst->nsub - 1)*(ILBC_LPC_FILTERORDER + 1),
SUBL,
iLBCdec_inst->syntMem);
for (i = 1; i < iLBCdec_inst->nsub; i++)
{
syntFilter(data + i*SUBL,
syntdenum + (i - 1)*(ILBC_LPC_FILTERORDER + 1),
SUBL,
iLBCdec_inst->syntMem);
}
}
else if (iLBCdec_inst->mode == 30)
{
/* Enhancer has 80 samples delay */
for (i = 0; i < 2; i++)
{
syntFilter(data + i*SUBL,
iLBCdec_inst->old_syntdenum + (i + iLBCdec_inst->nsub - 2)*(ILBC_LPC_FILTERORDER+1),
SUBL,
iLBCdec_inst->syntMem);
}
for (i = 2; i < iLBCdec_inst->nsub; i++)
{
syntFilter(data + i*SUBL,
syntdenum + (i - 2)*(ILBC_LPC_FILTERORDER + 1),
SUBL,
iLBCdec_inst->syntMem);
}
}
}
else
{
/* Find last lag */
lag = 20;
maxcc = xCorrCoef(&decresidual[ILBC_BLOCK_LEN_MAX - ENH_BLOCKL],
&decresidual[ILBC_BLOCK_LEN_MAX - ENH_BLOCKL - lag],
ENH_BLOCKL);
for (ilag = 21; ilag < 120; ilag++)
{
cc = xCorrCoef(&decresidual[ILBC_BLOCK_LEN_MAX - ENH_BLOCKL],
&decresidual[ILBC_BLOCK_LEN_MAX - ENH_BLOCKL - ilag],
ENH_BLOCKL);
if (cc > maxcc)
{
maxcc = cc;
lag = ilag;
}
}
iLBCdec_inst->last_lag = lag;
/* Copy data and run synthesis filter */
memcpy(data, decresidual, iLBCdec_inst->blockl*sizeof(float));
for (i = 0; i < iLBCdec_inst->nsub; i++)
{
syntFilter(data + i*SUBL,
syntdenum + i*(ILBC_LPC_FILTERORDER + 1),
SUBL,
iLBCdec_inst->syntMem);
}
}
/* High pass filtering on output if desired, otherwise copy to out */
hpOutput(data, iLBCdec_inst->blockl, decblock, iLBCdec_inst->hpomem);
/* memcpy(decblock, data, iLBCdec_inst->blockl*sizeof(float));*/
memcpy(iLBCdec_inst->old_syntdenum,
syntdenum,
iLBCdec_inst->nsub*(ILBC_LPC_FILTERORDER + 1)*sizeof(float));
iLBCdec_inst->prev_enh_pl = 0;
if (mode == 0)
{
/* PLC was used */
iLBCdec_inst->prev_enh_pl = 1;
}
}
int ilbc_decode(ilbc_decode_state_t *s, /* (i/o) the decoder state structure */
int16_t amp[], /* (o) decoded signal block */
const uint8_t bytes[], /* (i) encoded signal bits */
int len)
{
int i;
int j;
int k;
float decblock[ILBC_BLOCK_LEN_MAX];
float dtmp;
for (i = 0, j = 0; j < len; i += s->blockl, j += s->no_of_bytes)
{
ilbc_decode_frame(s, decblock, bytes + j, 1);
/* Convert to int16_t */
for (k = 0; k < s->blockl; k++)
{
dtmp = decblock[k];
if (dtmp < MIN_SAMPLE)
dtmp = MIN_SAMPLE;
else if (dtmp > MAX_SAMPLE)
dtmp = MAX_SAMPLE;
amp[i + k] = (int16_t) rint(dtmp);
}
}
return i;
}
int ilbc_fillin(ilbc_decode_state_t *s, /* (i/o) the decoder state structure */
int16_t amp[], /* (o) decoded signal block */
int len)
{
int i;
int j;
int k;
float decblock[ILBC_BLOCK_LEN_MAX];
float dtmp;
for (i = 0, j = 0; j < len; i += s->blockl, j += s->no_of_bytes)
{
ilbc_decode_frame(s, decblock, NULL, 0);
/* Convert to int16_t */
for (k = 0; k < s->blockl; k++)
{
dtmp = decblock[k];
if (dtmp < MIN_SAMPLE)
dtmp = MIN_SAMPLE;
else if (dtmp > MAX_SAMPLE)
dtmp = MAX_SAMPLE;
amp[i + k] = (int16_t) rint(dtmp);
}
}
return i;
}