mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-10-26 06:26:41 +00:00 
			
		
		
		
	
		
			
	
	
		
			231 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			231 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
|   | 
 | ||
|  | 
 | ||
|  |    /******************************************************************
 | ||
|  | 
 | ||
|  |        iLBC Speech Coder ANSI-C Source Code | ||
|  | 
 | ||
|  |        createCB.c | ||
|  | 
 | ||
|  |        Copyright (C) The Internet Society (2004). | ||
|  |        All Rights Reserved. | ||
|  | 
 | ||
|  |    ******************************************************************/ | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  |    #include "iLBC_define.h"
 | ||
|  |    #include "constants.h"
 | ||
|  |    #include <string.h>
 | ||
|  |    #include <math.h>
 | ||
|  | 
 | ||
|  |    /*----------------------------------------------------------------*
 | ||
|  |     *  Construct an additional codebook vector by filtering the | ||
|  |     *  initial codebook buffer. This vector is then used to expand | ||
|  |     *  the codebook with an additional section. | ||
|  |     *---------------------------------------------------------------*/ | ||
|  | 
 | ||
|  |    void filteredCBvecs( | ||
|  |        float *cbvectors,   /* (o) Codebook vectors for the
 | ||
|  |                                   higher section */ | ||
|  |        float *mem,         /* (i) Buffer to create codebook
 | ||
|  |                                   vector from */ | ||
|  |        int lMem        /* (i) Length of buffer */ | ||
|  |    ){ | ||
|  |        int j, k; | ||
|  |        float *pp, *pp1; | ||
|  |        float tempbuff2[CB_MEML+CB_FILTERLEN]; | ||
|  |        float *pos; | ||
|  | 
 | ||
|  |        memset(tempbuff2, 0, (CB_HALFFILTERLEN-1)*sizeof(float)); | ||
|  |        memcpy(&tempbuff2[CB_HALFFILTERLEN-1], mem, lMem*sizeof(float)); | ||
|  |        memset(&tempbuff2[lMem+CB_HALFFILTERLEN-1], 0, | ||
|  |            (CB_HALFFILTERLEN+1)*sizeof(float)); | ||
|  | 
 | ||
|  |        /* Create codebook vector for higher section by filtering */ | ||
|  | 
 | ||
|  |        /* do filtering */ | ||
|  |        pos=cbvectors; | ||
|  |        memset(pos, 0, lMem*sizeof(float)); | ||
|  |        for (k=0; k<lMem; k++) { | ||
|  |            pp=&tempbuff2[k]; | ||
|  |            pp1=&cbfiltersTbl[CB_FILTERLEN-1]; | ||
|  |            for (j=0;j<CB_FILTERLEN;j++) { | ||
|  |                (*pos)+=(*pp++)*(*pp1--); | ||
|  |            } | ||
|  |            pos++; | ||
|  |        } | ||
|  |    } | ||
|  | 
 | ||
|  |    /*----------------------------------------------------------------*
 | ||
|  |     *  Search the augmented part of the codebook to find the best | ||
|  |     *  measure. | ||
|  |     *----------------------------------------------------------------*/ | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  |    void searchAugmentedCB( | ||
|  |        int low,        /* (i) Start index for the search */ | ||
|  |        int high,           /* (i) End index for the search */ | ||
|  |        int stage,          /* (i) Current stage */ | ||
|  |        int startIndex,     /* (i) Codebook index for the first
 | ||
|  |                                   aug vector */ | ||
|  |        float *target,      /* (i) Target vector for encoding */ | ||
|  |        float *buffer,      /* (i) Pointer to the end of the buffer for
 | ||
|  |                                   augmented codebook construction */ | ||
|  |        float *max_measure, /* (i/o) Currently maximum measure */ | ||
|  |        int *best_index,/* (o) Currently the best index */ | ||
|  |        float *gain,    /* (o) Currently the best gain */ | ||
|  |        float *energy,      /* (o) Energy of augmented codebook
 | ||
|  |                                   vectors */ | ||
|  |        float *invenergy/* (o) Inv energy of augmented codebook
 | ||
|  |                                   vectors */ | ||
|  |    ) { | ||
|  |        int icount, ilow, j, tmpIndex; | ||
|  |        float *pp, *ppo, *ppi, *ppe, crossDot, alfa; | ||
|  |        float weighted, measure, nrjRecursive; | ||
|  |        float ftmp; | ||
|  | 
 | ||
|  |        /* Compute the energy for the first (low-5)
 | ||
|  |           noninterpolated samples */ | ||
|  |        nrjRecursive = (float) 0.0; | ||
|  |        pp = buffer - low + 1; | ||
|  |        for (j=0; j<(low-5); j++) { | ||
|  |            nrjRecursive += ( (*pp)*(*pp) ); | ||
|  |            pp++; | ||
|  |        } | ||
|  |        ppe = buffer - low; | ||
|  | 
 | ||
|  | 
 | ||
|  |        for (icount=low; icount<=high; icount++) { | ||
|  | 
 | ||
|  |            /* Index of the codebook vector used for retrieving
 | ||
|  |               energy values */ | ||
|  |            tmpIndex = startIndex+icount-20; | ||
|  | 
 | ||
|  |            ilow = icount-4; | ||
|  | 
 | ||
|  |            /* Update the energy recursively to save complexity */ | ||
|  |            nrjRecursive = nrjRecursive + (*ppe)*(*ppe); | ||
|  |            ppe--; | ||
|  |            energy[tmpIndex] = nrjRecursive; | ||
|  | 
 | ||
|  |            /* Compute cross dot product for the first (low-5)
 | ||
|  |               samples */ | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  |            crossDot = (float) 0.0; | ||
|  |            pp = buffer-icount; | ||
|  |            for (j=0; j<ilow; j++) { | ||
|  |                crossDot += target[j]*(*pp++); | ||
|  |            } | ||
|  | 
 | ||
|  |            /* interpolation */ | ||
|  |            alfa = (float) 0.2; | ||
|  |            ppo = buffer-4; | ||
|  |            ppi = buffer-icount-4; | ||
|  |            for (j=ilow; j<icount; j++) { | ||
|  |                weighted = ((float)1.0-alfa)*(*ppo)+alfa*(*ppi); | ||
|  |                ppo++; | ||
|  |                ppi++; | ||
|  |                energy[tmpIndex] += weighted*weighted; | ||
|  |                crossDot += target[j]*weighted; | ||
|  |                alfa += (float)0.2; | ||
|  |            } | ||
|  | 
 | ||
|  |            /* Compute energy and cross dot product for the
 | ||
|  |               remaining samples */ | ||
|  |            pp = buffer - icount; | ||
|  |            for (j=icount; j<SUBL; j++) { | ||
|  |                energy[tmpIndex] += (*pp)*(*pp); | ||
|  |                crossDot += target[j]*(*pp++); | ||
|  |            } | ||
|  | 
 | ||
|  |            if (energy[tmpIndex]>0.0) { | ||
|  |                invenergy[tmpIndex]=(float)1.0/(energy[tmpIndex]+EPS); | ||
|  |            } else { | ||
|  |                invenergy[tmpIndex] = (float) 0.0; | ||
|  |            } | ||
|  | 
 | ||
|  |            if (stage==0) { | ||
|  |                measure = (float)-10000000.0; | ||
|  | 
 | ||
|  |                if (crossDot > 0.0) { | ||
|  |                    measure = crossDot*crossDot*invenergy[tmpIndex]; | ||
|  |                } | ||
|  |            } | ||
|  |            else { | ||
|  |                measure = crossDot*crossDot*invenergy[tmpIndex]; | ||
|  |            } | ||
|  | 
 | ||
|  |            /* check if measure is better */ | ||
|  |            ftmp = crossDot*invenergy[tmpIndex]; | ||
|  | 
 | ||
|  |            if ((measure>*max_measure) && (fabs(ftmp)<CB_MAXGAIN)) { | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  |                *best_index = tmpIndex; | ||
|  |                *max_measure = measure; | ||
|  |                *gain = ftmp; | ||
|  |            } | ||
|  |        } | ||
|  |    } | ||
|  | 
 | ||
|  | 
 | ||
|  |    /*----------------------------------------------------------------*
 | ||
|  |     *  Recreate a specific codebook vector from the augmented part. | ||
|  |     * | ||
|  |     *----------------------------------------------------------------*/ | ||
|  | 
 | ||
|  |    void createAugmentedVec( | ||
|  |        int index,      /* (i) Index for the augmented vector
 | ||
|  |                               to be created */ | ||
|  |        float *buffer,  /* (i) Pointer to the end of the buffer for
 | ||
|  |                               augmented codebook construction */ | ||
|  |        float *cbVec/* (o) The construced codebook vector */ | ||
|  |    ) { | ||
|  |        int ilow, j; | ||
|  |        float *pp, *ppo, *ppi, alfa, alfa1, weighted; | ||
|  | 
 | ||
|  |        ilow = index-5; | ||
|  | 
 | ||
|  |        /* copy the first noninterpolated part */ | ||
|  | 
 | ||
|  |        pp = buffer-index; | ||
|  |        memcpy(cbVec,pp,sizeof(float)*index); | ||
|  | 
 | ||
|  |        /* interpolation */ | ||
|  | 
 | ||
|  |        alfa1 = (float)0.2; | ||
|  |        alfa = 0.0; | ||
|  |        ppo = buffer-5; | ||
|  |        ppi = buffer-index-5; | ||
|  |        for (j=ilow; j<index; j++) { | ||
|  |            weighted = ((float)1.0-alfa)*(*ppo)+alfa*(*ppi); | ||
|  |            ppo++; | ||
|  |            ppi++; | ||
|  |            cbVec[j] = weighted; | ||
|  |            alfa += alfa1; | ||
|  |        } | ||
|  | 
 | ||
|  |        /* copy the second noninterpolated part */ | ||
|  | 
 | ||
|  |        pp = buffer - index; | ||
|  |        memcpy(cbVec+index,pp,sizeof(float)*(SUBL-index)); | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  |    } |