/* * iLBC - a library for the iLBC codec * * iLBC_decode.c - The iLBC low bit rate speech codec. * * Adapted by Steve Underwood 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 #endif #include #include #include #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" /*----------------------------------------------------------------* * 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 */ memcpy(mem, mem + SUBL, (CB_MEML - SUBL)*sizeof(float)); memcpy(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 */ memcpy(mem, mem + SUBL, (CB_MEML - SUBL)*sizeof(float)); memcpy(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 check; 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); check = 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; }