1535 lines
		
	
	
		
			45 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1535 lines
		
	
	
		
			45 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  *  Copyright (c) 2011 The WebRTC project authors. All Rights Reserved.
 | |
|  *
 | |
|  *  Use of this source code is governed by a BSD-style license
 | |
|  *  that can be found in the LICENSE file in the root of the source
 | |
|  *  tree. An additional intellectual property rights grant can be found
 | |
|  *  in the file PATENTS.  All contributing project authors may
 | |
|  *  be found in the AUTHORS file in the root of the source tree.
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * isacfix.c
 | |
|  *
 | |
|  * This C file contains the functions for the ISAC API
 | |
|  *
 | |
|  */
 | |
| 
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| 
 | |
| #include "isacfix.h"
 | |
| #include "bandwidth_estimator.h"
 | |
| #include "codec.h"
 | |
| #include "entropy_coding.h"
 | |
| #include "structs.h"
 | |
| 
 | |
| 
 | |
| /**************************************************************************
 | |
|  * WebRtcIsacfix_AssignSize(...)
 | |
|  *
 | |
|  * Functions used when malloc is not allowed
 | |
|  * Returns number of bytes needed to allocate for iSAC struct.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_AssignSize(int *sizeinbytes) {
 | |
|   *sizeinbytes=sizeof(ISACFIX_SubStruct)*2/sizeof(WebRtc_Word16);
 | |
|   return(0);
 | |
| }
 | |
| 
 | |
| /***************************************************************************
 | |
|  * WebRtcIsacfix_Assign(...)
 | |
|  *
 | |
|  * Functions used when malloc is not allowed
 | |
|  * Place struct at given address
 | |
|  *
 | |
|  * If successful, Return 0, else Return -1
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_Assign(ISACFIX_MainStruct **inst, void *ISACFIX_inst_Addr) {
 | |
|   if (ISACFIX_inst_Addr!=NULL) {
 | |
|     *inst = (ISACFIX_MainStruct*)ISACFIX_inst_Addr;
 | |
|     (*(ISACFIX_SubStruct**)inst)->errorcode = 0;
 | |
|     (*(ISACFIX_SubStruct**)inst)->initflag = 0;
 | |
|     (*(ISACFIX_SubStruct**)inst)->ISACenc_obj.SaveEnc_ptr = NULL;
 | |
|     return(0);
 | |
|   } else {
 | |
|     return(-1);
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| #ifndef ISACFIX_NO_DYNAMIC_MEM
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_Create(...)
 | |
|  *
 | |
|  * This function creates a ISAC instance, which will contain the state
 | |
|  * information for one coding/decoding channel.
 | |
|  *
 | |
|  * Input:
 | |
|  *      - *ISAC_main_inst   : a pointer to the coder instance.
 | |
|  *
 | |
|  * Return value             :  0 - Ok
 | |
|  *                            -1 - Error
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_Create(ISACFIX_MainStruct **ISAC_main_inst)
 | |
| {
 | |
|   ISACFIX_SubStruct *tempo;
 | |
|   tempo = malloc(1 * sizeof(ISACFIX_SubStruct));
 | |
|   *ISAC_main_inst = (ISACFIX_MainStruct *)tempo;
 | |
|   if (*ISAC_main_inst!=NULL) {
 | |
|     (*(ISACFIX_SubStruct**)ISAC_main_inst)->errorcode = 0;
 | |
|     (*(ISACFIX_SubStruct**)ISAC_main_inst)->initflag = 0;
 | |
|     (*(ISACFIX_SubStruct**)ISAC_main_inst)->ISACenc_obj.SaveEnc_ptr = NULL;
 | |
|     return(0);
 | |
|   } else {
 | |
|     return(-1);
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_CreateInternal(...)
 | |
|  *
 | |
|  * This function creates the memory that is used to store data in the encoder
 | |
|  *
 | |
|  * Input:
 | |
|  *      - *ISAC_main_inst   : a pointer to the coder instance.
 | |
|  *
 | |
|  * Return value             :  0 - Ok
 | |
|  *                            -1 - Error
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_CreateInternal(ISACFIX_MainStruct *ISAC_main_inst)
 | |
| {
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
| 
 | |
|   /* typecast pointer to real structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
|   /* Allocate memory for storing encoder data */
 | |
|   ISAC_inst->ISACenc_obj.SaveEnc_ptr = malloc(1 * sizeof(ISAC_SaveEncData_t));
 | |
| 
 | |
|   if (ISAC_inst->ISACenc_obj.SaveEnc_ptr!=NULL) {
 | |
|     return(0);
 | |
|   } else {
 | |
|     return(-1);
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| #endif
 | |
| 
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_Free(...)
 | |
|  *
 | |
|  * This function frees the ISAC instance created at the beginning.
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst    : a ISAC instance.
 | |
|  *
 | |
|  * Return value             :  0 - Ok
 | |
|  *                            -1 - Error
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_Free(ISACFIX_MainStruct *ISAC_main_inst)
 | |
| {
 | |
|   free(ISAC_main_inst);
 | |
|   return(0);
 | |
| }
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_FreeInternal(...)
 | |
|  *
 | |
|  * This function frees the internal memory for storing encoder data.
 | |
|  *
 | |
|  * Input:
 | |
|  *       - ISAC_main_inst    : a ISAC instance.
 | |
|  *
 | |
|  * Return value              :  0 - Ok
 | |
|  *                             -1 - Error
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_FreeInternal(ISACFIX_MainStruct *ISAC_main_inst)
 | |
| {
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
| 
 | |
|   /* typecast pointer to real structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
|   /* Release memory */
 | |
|   free(ISAC_inst->ISACenc_obj.SaveEnc_ptr);
 | |
| 
 | |
|   return(0);
 | |
| }
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_EncoderInit(...)
 | |
|  *
 | |
|  * This function initializes a ISAC instance prior to the encoder calls.
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst    : ISAC instance.
 | |
|  *      - CodingMode        : 0 -> Bit rate and frame length are automatically
 | |
|  *                                 adjusted to available bandwidth on
 | |
|  *                                 transmission channel.
 | |
|  *                            1 -> User sets a frame length and a target bit
 | |
|  *                                 rate which is taken as the maximum short-term
 | |
|  *                                 average bit rate.
 | |
|  *
 | |
|  * Return value             :  0 - Ok
 | |
|  *                            -1 - Error
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_EncoderInit(ISACFIX_MainStruct *ISAC_main_inst,
 | |
|                                         WebRtc_Word16  CodingMode)
 | |
| {
 | |
|   int k;
 | |
|   WebRtc_Word16 statusInit;
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
| 
 | |
|   statusInit = 0;
 | |
|   /* typecast pointer to rela structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
|   /* flag encoder init */
 | |
|   ISAC_inst->initflag |= 2;
 | |
| 
 | |
|   if (CodingMode == 0)
 | |
|     /* Adaptive mode */
 | |
|     ISAC_inst->ISACenc_obj.new_framelength  = INITIAL_FRAMESAMPLES;
 | |
|   else if (CodingMode == 1)
 | |
|     /* Instantaneous mode */
 | |
|     ISAC_inst->ISACenc_obj.new_framelength = 480;    /* default for I-mode */
 | |
|   else {
 | |
|     ISAC_inst->errorcode = ISAC_DISALLOWED_CODING_MODE;
 | |
|     statusInit = -1;
 | |
|   }
 | |
| 
 | |
|   ISAC_inst->CodingMode = CodingMode;
 | |
| 
 | |
|   WebRtcIsacfix_InitMaskingEnc(&ISAC_inst->ISACenc_obj.maskfiltstr_obj);
 | |
|   WebRtcIsacfix_InitPreFilterbank(&ISAC_inst->ISACenc_obj.prefiltbankstr_obj);
 | |
|   WebRtcIsacfix_InitPitchFilter(&ISAC_inst->ISACenc_obj.pitchfiltstr_obj);
 | |
|   WebRtcIsacfix_InitPitchAnalysis(&ISAC_inst->ISACenc_obj.pitchanalysisstr_obj);
 | |
| 
 | |
| 
 | |
|   WebRtcIsacfix_InitBandwidthEstimator(&ISAC_inst->bwestimator_obj);
 | |
|   WebRtcIsacfix_InitRateModel(&ISAC_inst->ISACenc_obj.rate_data_obj);
 | |
| 
 | |
| 
 | |
|   ISAC_inst->ISACenc_obj.buffer_index   = 0;
 | |
|   ISAC_inst->ISACenc_obj.frame_nb    = 0;
 | |
|   ISAC_inst->ISACenc_obj.BottleNeck      = 32000; /* default for I-mode */
 | |
|   ISAC_inst->ISACenc_obj.MaxDelay    = 10;    /* default for I-mode */
 | |
|   ISAC_inst->ISACenc_obj.current_framesamples = 0;
 | |
|   ISAC_inst->ISACenc_obj.s2nr     = 0;
 | |
|   ISAC_inst->ISACenc_obj.MaxBits    = 0;
 | |
|   ISAC_inst->ISACenc_obj.bitstr_seed   = 4447;
 | |
|   ISAC_inst->ISACenc_obj.payloadLimitBytes30  = STREAM_MAXW16_30MS << 1;
 | |
|   ISAC_inst->ISACenc_obj.payloadLimitBytes60  = STREAM_MAXW16_60MS << 1;
 | |
|   ISAC_inst->ISACenc_obj.maxPayloadBytes      = STREAM_MAXW16_60MS << 1;
 | |
|   ISAC_inst->ISACenc_obj.maxRateInBytes       = STREAM_MAXW16_30MS << 1;
 | |
|   ISAC_inst->ISACenc_obj.enforceFrameSize     = 0;
 | |
| 
 | |
|   /* Init the bistream data area to zero */
 | |
|   for (k=0; k<STREAM_MAXW16_60MS; k++){
 | |
|     ISAC_inst->ISACenc_obj.bitstr_obj.stream[k] = 0;
 | |
|   }
 | |
| 
 | |
| #ifdef NB_CALLS
 | |
|   WebRtcIsacfix_InitPostFilterbank(&ISAC_inst->ISACenc_obj.interpolatorstr_obj);
 | |
| #endif
 | |
| 
 | |
| 
 | |
|   return statusInit;
 | |
| }
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_Encode(...)
 | |
|  *
 | |
|  * This function encodes 10ms frame(s) and inserts it into a package.
 | |
|  * Input speech length has to be 160 samples (10ms). The encoder buffers those
 | |
|  * 10ms frames until it reaches the chosen Framesize (480 or 960 samples
 | |
|  * corresponding to 30 or 60 ms frames), and then proceeds to the encoding.
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst    : ISAC instance.
 | |
|  *      - speechIn          : input speech vector.
 | |
|  *
 | |
|  * Output:
 | |
|  *      - encoded           : the encoded data vector
 | |
|  *
 | |
|  * Return value:
 | |
|  *                          : >0 - Length (in bytes) of coded data
 | |
|  *                          :  0 - The buffer didn't reach the chosen framesize
 | |
|  *                            so it keeps buffering speech samples.
 | |
|  *                          : -1 - Error
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_Encode(ISACFIX_MainStruct *ISAC_main_inst,
 | |
|                                    const WebRtc_Word16    *speechIn,
 | |
|                                    WebRtc_Word16          *encoded)
 | |
| {
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
|   WebRtc_Word16 stream_len;
 | |
| #ifndef WEBRTC_BIG_ENDIAN
 | |
|   int k;
 | |
| #endif
 | |
| 
 | |
|   /* typecast pointer to rela structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
| 
 | |
|   /* check if encoder initiated */
 | |
|   if ((ISAC_inst->initflag & 2) != 2) {
 | |
|     ISAC_inst->errorcode = ISAC_ENCODER_NOT_INITIATED;
 | |
|     return (-1);
 | |
|   }
 | |
| 
 | |
|   stream_len = WebRtcIsacfix_EncodeImpl((WebRtc_Word16*)speechIn,
 | |
|                                     &ISAC_inst->ISACenc_obj,
 | |
|                                     &ISAC_inst->bwestimator_obj,
 | |
|                                     ISAC_inst->CodingMode);
 | |
|   if (stream_len<0) {
 | |
|     ISAC_inst->errorcode = - stream_len;
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
| 
 | |
|   /* convert from bytes to WebRtc_Word16 */
 | |
| #ifndef WEBRTC_BIG_ENDIAN
 | |
|   for (k=0;k<(stream_len+1)>>1;k++) {
 | |
|     encoded[k] = (WebRtc_Word16)( ( (WebRtc_UWord16)(ISAC_inst->ISACenc_obj.bitstr_obj).stream[k] >> 8 )
 | |
|                                   | (((ISAC_inst->ISACenc_obj.bitstr_obj).stream[k] & 0x00FF) << 8));
 | |
|   }
 | |
| 
 | |
| #else
 | |
|   WEBRTC_SPL_MEMCPY_W16(encoded, (ISAC_inst->ISACenc_obj.bitstr_obj).stream, (stream_len + 1)>>1);
 | |
| #endif
 | |
| 
 | |
| 
 | |
| 
 | |
|   return stream_len;
 | |
| 
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_EncodeNb(...)
 | |
|  *
 | |
|  * This function encodes 10ms narrow band (8 kHz sampling) frame(s) and inserts
 | |
|  * it into a package. Input speech length has to be 80 samples (10ms). The encoder
 | |
|  * interpolates into wide-band (16 kHz sampling) buffers those
 | |
|  * 10ms frames until it reaches the chosen Framesize (480 or 960 wide-band samples
 | |
|  * corresponding to 30 or 60 ms frames), and then proceeds to the encoding.
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst    : ISAC instance.
 | |
|  *      - speechIn          : input speech vector.
 | |
|  *
 | |
|  * Output:
 | |
|  *      - encoded           : the encoded data vector
 | |
|  *
 | |
|  * Return value:
 | |
|  *                          : >0 - Length (in bytes) of coded data
 | |
|  *                          :  0 - The buffer didn't reach the chosen framesize
 | |
|  *                            so it keeps buffering speech samples.
 | |
|  *                          : -1 - Error
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_EncodeNb(ISACFIX_MainStruct *ISAC_main_inst,
 | |
|                                       const WebRtc_Word16    *speechIn,
 | |
|                                       WebRtc_Word16          *encoded)
 | |
| {
 | |
| #ifdef NB_CALLS
 | |
| 
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
|   WebRtc_Word16 stream_len;
 | |
|   WebRtc_Word16 speechInWB[FRAMESAMPLES_10ms];
 | |
|   WebRtc_Word16 Vector_Word16_1[FRAMESAMPLES_10ms/2];
 | |
|   WebRtc_Word16 Vector_Word16_2[FRAMESAMPLES_10ms/2];
 | |
| 
 | |
|   int k;
 | |
| 
 | |
| 
 | |
|   /* typecast pointer to rela structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
| 
 | |
|   /* check if encoder initiated */
 | |
|   if ((ISAC_inst->initflag & 2) != 2) {
 | |
|     ISAC_inst->errorcode = ISAC_ENCODER_NOT_INITIATED;
 | |
|     return (-1);
 | |
|   }
 | |
| 
 | |
| 
 | |
|   /* Oversample to WB */
 | |
| 
 | |
|   /* Form polyphase signals, and compensate for DC offset */
 | |
|   for (k=0;k<FRAMESAMPLES_10ms/2;k++) {
 | |
|     Vector_Word16_1[k] = speechIn[k] + 1;
 | |
|     Vector_Word16_2[k] = speechIn[k];
 | |
|   }
 | |
|   WebRtcIsacfix_FilterAndCombine2(Vector_Word16_1, Vector_Word16_2, speechInWB, &ISAC_inst->ISACenc_obj.interpolatorstr_obj, FRAMESAMPLES_10ms);
 | |
| 
 | |
| 
 | |
|   /* Encode WB signal */
 | |
|   stream_len = WebRtcIsacfix_EncodeImpl((WebRtc_Word16*)speechInWB,
 | |
|                                     &ISAC_inst->ISACenc_obj,
 | |
|                                     &ISAC_inst->bwestimator_obj,
 | |
|                                     ISAC_inst->CodingMode);
 | |
|   if (stream_len<0) {
 | |
|     ISAC_inst->errorcode = - stream_len;
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
| 
 | |
|   /* convert from bytes to WebRtc_Word16 */
 | |
| #ifndef WEBRTC_BIG_ENDIAN
 | |
|   for (k=0;k<(stream_len+1)>>1;k++) {
 | |
|     encoded[k] = (WebRtc_Word16)(((WebRtc_UWord16)(ISAC_inst->ISACenc_obj.bitstr_obj).stream[k] >> 8)
 | |
|                                  | (((ISAC_inst->ISACenc_obj.bitstr_obj).stream[k] & 0x00FF) << 8));
 | |
|   }
 | |
| 
 | |
| #else
 | |
|   WEBRTC_SPL_MEMCPY_W16(encoded, (ISAC_inst->ISACenc_obj.bitstr_obj).stream, (stream_len + 1)>>1);
 | |
| #endif
 | |
| 
 | |
| 
 | |
| 
 | |
|   return stream_len;
 | |
| 
 | |
| #else /* NB_CALLS not defined */
 | |
|   return -1;
 | |
| #endif  /* NB_CALLS */
 | |
| }
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_GetNewBitStream(...)
 | |
|  *
 | |
|  * This function returns encoded data, with the recieved bwe-index in the
 | |
|  * stream. It should always return a complete packet, i.e. only called once
 | |
|  * even for 60 msec frames
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst    : ISAC instance.
 | |
|  *      - bweIndex          : index of bandwidth estimate to put in new bitstream
 | |
|  *
 | |
|  * Output:
 | |
|  *      - encoded           : the encoded data vector
 | |
|  *
 | |
|  * Return value:
 | |
|  *                          : >0 - Length (in bytes) of coded data
 | |
|  *                          : -1 - Error
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_GetNewBitStream(ISACFIX_MainStruct *ISAC_main_inst,
 | |
|                                             WebRtc_Word16      bweIndex,
 | |
|                                             float              scale,
 | |
|                                             WebRtc_Word16        *encoded)
 | |
| {
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
|   WebRtc_Word16 stream_len;
 | |
| #ifndef WEBRTC_BIG_ENDIAN
 | |
|   int k;
 | |
| #endif
 | |
| 
 | |
|   /* typecast pointer to rela structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
| 
 | |
|   /* check if encoder initiated */
 | |
|   if ((ISAC_inst->initflag & 2) != 2) {
 | |
|     ISAC_inst->errorcode = ISAC_ENCODER_NOT_INITIATED;
 | |
|     return (-1);
 | |
|   }
 | |
| 
 | |
|   stream_len = WebRtcIsacfix_EncodeStoredData(&ISAC_inst->ISACenc_obj,
 | |
|                                               bweIndex,
 | |
|                                               scale,
 | |
|                                               ISAC_inst->CodingMode);
 | |
|   if (stream_len<0) {
 | |
|     ISAC_inst->errorcode = - stream_len;
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
| #ifndef WEBRTC_BIG_ENDIAN
 | |
|   for (k=0;k<(stream_len+1)>>1;k++) {
 | |
|     encoded[k] = (WebRtc_Word16)( ( (WebRtc_UWord16)(ISAC_inst->ISACenc_obj.bitstr_obj).stream[k] >> 8 )
 | |
|                                   | (((ISAC_inst->ISACenc_obj.bitstr_obj).stream[k] & 0x00FF) << 8));
 | |
|   }
 | |
| 
 | |
| #else
 | |
|   WEBRTC_SPL_MEMCPY_W16(encoded, (ISAC_inst->ISACenc_obj.bitstr_obj).stream, (stream_len + 1)>>1);
 | |
| #endif
 | |
| 
 | |
|   return stream_len;
 | |
| 
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_DecoderInit(...)
 | |
|  *
 | |
|  * This function initializes a ISAC instance prior to the decoder calls.
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst    : ISAC instance.
 | |
|  *
 | |
|  * Return value
 | |
|  *                          :  0 - Ok
 | |
|  *                            -1 - Error
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_DecoderInit(ISACFIX_MainStruct *ISAC_main_inst)
 | |
| {
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
| 
 | |
|   /* typecast pointer to real structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
|   /* flag decoder init */
 | |
|   ISAC_inst->initflag |= 1;
 | |
| 
 | |
| 
 | |
|   WebRtcIsacfix_InitMaskingDec(&ISAC_inst->ISACdec_obj.maskfiltstr_obj);
 | |
|   WebRtcIsacfix_InitPostFilterbank(&ISAC_inst->ISACdec_obj.postfiltbankstr_obj);
 | |
|   WebRtcIsacfix_InitPitchFilter(&ISAC_inst->ISACdec_obj.pitchfiltstr_obj);
 | |
| 
 | |
|   /* TS */
 | |
|   WebRtcIsacfix_InitPlc( &ISAC_inst->ISACdec_obj.plcstr_obj );
 | |
| 
 | |
| 
 | |
| #ifdef NB_CALLS
 | |
|   WebRtcIsacfix_InitPreFilterbank(&ISAC_inst->ISACdec_obj.decimatorstr_obj);
 | |
| #endif
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_UpdateBwEstimate1(...)
 | |
|  *
 | |
|  * This function updates the estimate of the bandwidth.
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst    : ISAC instance.
 | |
|  *      - encoded           : encoded ISAC frame(s).
 | |
|  *      - packet_size       : size of the packet.
 | |
|  *      - rtp_seq_number    : the RTP number of the packet.
 | |
|  *      - arr_ts            : the arrival time of the packet (from NetEq)
 | |
|  *                            in samples.
 | |
|  *
 | |
|  * Return value             :  0 - Ok
 | |
|  *                            -1 - Error
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_UpdateBwEstimate1(ISACFIX_MainStruct *ISAC_main_inst,
 | |
|                                      const WebRtc_UWord16   *encoded,
 | |
|                                      WebRtc_Word32          packet_size,
 | |
|                                      WebRtc_UWord16         rtp_seq_number,
 | |
|                                      WebRtc_UWord32         arr_ts)
 | |
| {
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
|   Bitstr_dec streamdata;
 | |
|   WebRtc_UWord16 partOfStream[5];
 | |
| #ifndef WEBRTC_BIG_ENDIAN
 | |
|   int k;
 | |
| #endif
 | |
|   WebRtc_Word16 err;
 | |
| 
 | |
|   /* Set stream pointer to point at partOfStream */
 | |
|   streamdata.stream = (WebRtc_UWord16 *)partOfStream;
 | |
| 
 | |
|   /* typecast pointer to real structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
|   /* Sanity check of packet length */
 | |
|   if (packet_size <= 0) {
 | |
|     /* return error code if the packet length is null or less */
 | |
|     ISAC_inst->errorcode = ISAC_EMPTY_PACKET;
 | |
|     return -1;
 | |
|   } else if (packet_size > (STREAM_MAXW16<<1)) {
 | |
|     /* return error code if length of stream is too long */
 | |
|     ISAC_inst->errorcode = ISAC_LENGTH_MISMATCH;
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   /* check if decoder initiated */
 | |
|   if ((ISAC_inst->initflag & 1) != 1) {
 | |
|     ISAC_inst->errorcode = ISAC_DECODER_NOT_INITIATED;
 | |
|     return (-1);
 | |
|   }
 | |
| 
 | |
|   streamdata.W_upper = 0xFFFFFFFF;
 | |
|   streamdata.streamval = 0;
 | |
|   streamdata.stream_index = 0;
 | |
|   streamdata.full = 1;
 | |
| 
 | |
| #ifndef WEBRTC_BIG_ENDIAN
 | |
|   for (k=0; k<5; k++) {
 | |
|     streamdata.stream[k] = (WebRtc_UWord16) (((WebRtc_UWord16)encoded[k] >> 8)|((encoded[k] & 0xFF)<<8));
 | |
|   }
 | |
| #else
 | |
|   memcpy(streamdata.stream, encoded, 5);
 | |
| #endif
 | |
| 
 | |
|   if (packet_size == 0)
 | |
|   {
 | |
|     /* return error code if the packet length is null */
 | |
|     ISAC_inst->errorcode = ISAC_EMPTY_PACKET;
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   err = WebRtcIsacfix_EstimateBandwidth(&ISAC_inst->bwestimator_obj,
 | |
|                                         &streamdata,
 | |
|                                         packet_size,
 | |
|                                         rtp_seq_number,
 | |
|                                         0,
 | |
|                                         arr_ts);
 | |
| 
 | |
| 
 | |
|   if (err < 0)
 | |
|   {
 | |
|     /* return error code if something went wrong */
 | |
|     ISAC_inst->errorcode = -err;
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_UpdateBwEstimate(...)
 | |
|  *
 | |
|  * This function updates the estimate of the bandwidth.
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst    : ISAC instance.
 | |
|  *      - encoded           : encoded ISAC frame(s).
 | |
|  *      - packet_size       : size of the packet.
 | |
|  *      - rtp_seq_number    : the RTP number of the packet.
 | |
|  *      - send_ts           : Send Time Stamp from RTP header
 | |
|  *      - arr_ts            : the arrival time of the packet (from NetEq)
 | |
|  *                            in samples.
 | |
|  *
 | |
|  * Return value             :  0 - Ok
 | |
|  *                            -1 - Error
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_UpdateBwEstimate(ISACFIX_MainStruct *ISAC_main_inst,
 | |
|                                        const WebRtc_UWord16   *encoded,
 | |
|                                        WebRtc_Word32          packet_size,
 | |
|                                        WebRtc_UWord16         rtp_seq_number,
 | |
|                                        WebRtc_UWord32         send_ts,
 | |
|                                        WebRtc_UWord32         arr_ts)
 | |
| {
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
|   Bitstr_dec streamdata;
 | |
|   WebRtc_UWord16 partOfStream[5];
 | |
| #ifndef WEBRTC_BIG_ENDIAN
 | |
|   int k;
 | |
| #endif
 | |
|   WebRtc_Word16 err;
 | |
| 
 | |
|   /* Set stream pointer to point at partOfStream */
 | |
|   streamdata.stream = (WebRtc_UWord16 *)partOfStream;
 | |
| 
 | |
|   /* typecast pointer to real structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
|   /* Sanity check of packet length */
 | |
|   if (packet_size <= 0) {
 | |
|     /* return error code if the packet length is null  or less */
 | |
|     ISAC_inst->errorcode = ISAC_EMPTY_PACKET;
 | |
|     return -1;
 | |
|   } else if (packet_size > (STREAM_MAXW16<<1)) {
 | |
|     /* return error code if length of stream is too long */
 | |
|     ISAC_inst->errorcode = ISAC_LENGTH_MISMATCH;
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   /* check if decoder initiated */
 | |
|   if ((ISAC_inst->initflag & 1) != 1) {
 | |
|     ISAC_inst->errorcode = ISAC_DECODER_NOT_INITIATED;
 | |
|     return (-1);
 | |
|   }
 | |
| 
 | |
|   streamdata.W_upper = 0xFFFFFFFF;
 | |
|   streamdata.streamval = 0;
 | |
|   streamdata.stream_index = 0;
 | |
|   streamdata.full = 1;
 | |
| 
 | |
| #ifndef WEBRTC_BIG_ENDIAN
 | |
|   for (k=0; k<5; k++) {
 | |
|     streamdata.stream[k] = (WebRtc_UWord16) ((encoded[k] >> 8)|((encoded[k] & 0xFF)<<8));
 | |
|   }
 | |
| #else
 | |
|   memcpy(streamdata.stream, encoded, 5);
 | |
| #endif
 | |
| 
 | |
|   if (packet_size == 0)
 | |
|   {
 | |
|     /* return error code if the packet length is null */
 | |
|     ISAC_inst->errorcode = ISAC_EMPTY_PACKET;
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   err = WebRtcIsacfix_EstimateBandwidth(&ISAC_inst->bwestimator_obj,
 | |
|                                         &streamdata,
 | |
|                                         packet_size,
 | |
|                                         rtp_seq_number,
 | |
|                                         send_ts,
 | |
|                                         arr_ts);
 | |
| 
 | |
|   if (err < 0)
 | |
|   {
 | |
|     /* return error code if something went wrong */
 | |
|     ISAC_inst->errorcode = -err;
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_Decode(...)
 | |
|  *
 | |
|  * This function decodes a ISAC frame. Output speech length
 | |
|  * will be a multiple of 480 samples: 480 or 960 samples,
 | |
|  * depending on the framesize (30 or 60 ms).
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst    : ISAC instance.
 | |
|  *      - encoded           : encoded ISAC frame(s)
 | |
|  *      - len               : bytes in encoded vector
 | |
|  *
 | |
|  * Output:
 | |
|  *      - decoded           : The decoded vector
 | |
|  *
 | |
|  * Return value             : >0 - number of samples in decoded vector
 | |
|  *                            -1 - Error
 | |
|  */
 | |
| 
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_Decode(ISACFIX_MainStruct *ISAC_main_inst,
 | |
|                                      const WebRtc_UWord16   *encoded,
 | |
|                                      WebRtc_Word16          len,
 | |
|                                      WebRtc_Word16          *decoded,
 | |
|                                      WebRtc_Word16     *speechType)
 | |
| {
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
|   /* number of samples (480 or 960), output from decoder */
 | |
|   /* that were actually used in the encoder/decoder (determined on the fly) */
 | |
|   WebRtc_Word16     number_of_samples;
 | |
| #ifndef WEBRTC_BIG_ENDIAN
 | |
|   int k;
 | |
| #endif
 | |
|   WebRtc_Word16 declen = 0;
 | |
| 
 | |
|   /* typecast pointer to real structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
|   /* check if decoder initiated */
 | |
|   if ((ISAC_inst->initflag & 1) != 1) {
 | |
|     ISAC_inst->errorcode = ISAC_DECODER_NOT_INITIATED;
 | |
|     return (-1);
 | |
|   }
 | |
| 
 | |
|   /* Sanity check of packet length */
 | |
|   if (len <= 0) {
 | |
|     /* return error code if the packet length is null  or less */
 | |
|     ISAC_inst->errorcode = ISAC_EMPTY_PACKET;
 | |
|     return -1;
 | |
|   } else if (len > (STREAM_MAXW16<<1)) {
 | |
|     /* return error code if length of stream is too long */
 | |
|     ISAC_inst->errorcode = ISAC_LENGTH_MISMATCH;
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   (ISAC_inst->ISACdec_obj.bitstr_obj).stream = (WebRtc_UWord16 *)encoded;
 | |
| 
 | |
|   /* convert bitstream from WebRtc_Word16 to bytes */
 | |
| #ifndef WEBRTC_BIG_ENDIAN
 | |
|   for (k=0; k<(len>>1); k++) {
 | |
|     (ISAC_inst->ISACdec_obj.bitstr_obj).stream[k] = (WebRtc_UWord16) ((encoded[k] >> 8)|((encoded[k] & 0xFF)<<8));
 | |
|   }
 | |
|   if (len & 0x0001)
 | |
|     (ISAC_inst->ISACdec_obj.bitstr_obj).stream[k] = (WebRtc_UWord16) ((encoded[k] & 0xFF)<<8);
 | |
| #endif
 | |
| 
 | |
|   /* added for NetEq purposes (VAD/DTX related) */
 | |
|   *speechType=1;
 | |
| 
 | |
|   declen = WebRtcIsacfix_DecodeImpl(decoded,&ISAC_inst->ISACdec_obj, &number_of_samples);
 | |
| 
 | |
|   if (declen < 0) {
 | |
|     /* Some error inside the decoder */
 | |
|     ISAC_inst->errorcode = -declen;
 | |
|     memset(decoded, 0, sizeof(WebRtc_Word16) * MAX_FRAMESAMPLES);
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   /* error check */
 | |
| 
 | |
|   if (declen & 0x0001) {
 | |
|     if (len != declen && len != declen + (((ISAC_inst->ISACdec_obj.bitstr_obj).stream[declen>>1]) & 0x00FF) ) {
 | |
|       ISAC_inst->errorcode = ISAC_LENGTH_MISMATCH;
 | |
|       memset(decoded, 0, sizeof(WebRtc_Word16) * number_of_samples);
 | |
|       return -1;
 | |
|     }
 | |
|   } else {
 | |
|     if (len != declen && len != declen + (((ISAC_inst->ISACdec_obj.bitstr_obj).stream[declen>>1]) >> 8) ) {
 | |
|       ISAC_inst->errorcode = ISAC_LENGTH_MISMATCH;
 | |
|       memset(decoded, 0, sizeof(WebRtc_Word16) * number_of_samples);
 | |
|       return -1;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return number_of_samples;
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_DecodeNb(...)
 | |
|  *
 | |
|  * This function decodes a ISAC frame in narrow-band (8 kHz sampling).
 | |
|  * Output speech length will be a multiple of 240 samples: 240 or 480 samples,
 | |
|  * depending on the framesize (30 or 60 ms).
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst    : ISAC instance.
 | |
|  *      - encoded           : encoded ISAC frame(s)
 | |
|  *      - len               : bytes in encoded vector
 | |
|  *
 | |
|  * Output:
 | |
|  *      - decoded           : The decoded vector
 | |
|  *
 | |
|  * Return value             : >0 - number of samples in decoded vector
 | |
|  *                            -1 - Error
 | |
|  */
 | |
| 
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_DecodeNb(ISACFIX_MainStruct *ISAC_main_inst,
 | |
|                                         const WebRtc_UWord16   *encoded,
 | |
|                                         WebRtc_Word16          len,
 | |
|                                         WebRtc_Word16          *decoded,
 | |
|                                         WebRtc_Word16    *speechType)
 | |
| {
 | |
| #ifdef NB_CALLS
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
|   /* twice the number of samples (480 or 960), output from decoder */
 | |
|   /* that were actually used in the encoder/decoder (determined on the fly) */
 | |
|   WebRtc_Word16     number_of_samples;
 | |
| #ifndef WEBRTC_BIG_ENDIAN
 | |
|   int k;
 | |
| #endif
 | |
|   WebRtc_Word16 declen = 0;
 | |
|   WebRtc_Word16 dummy[FRAMESAMPLES/2];
 | |
| 
 | |
| 
 | |
|   /* typecast pointer to real structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
|   /* check if decoder initiated */
 | |
|   if ((ISAC_inst->initflag & 1) != 1) {
 | |
|     ISAC_inst->errorcode = ISAC_DECODER_NOT_INITIATED;
 | |
|     return (-1);
 | |
|   }
 | |
| 
 | |
|   if (len == 0)
 | |
|   {  /* return error code if the packet length is null */
 | |
| 
 | |
|     ISAC_inst->errorcode = ISAC_EMPTY_PACKET;
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   (ISAC_inst->ISACdec_obj.bitstr_obj).stream = (WebRtc_UWord16 *)encoded;
 | |
| 
 | |
|   /* convert bitstream from WebRtc_Word16 to bytes */
 | |
| #ifndef WEBRTC_BIG_ENDIAN
 | |
|   for (k=0; k<(len>>1); k++) {
 | |
|     (ISAC_inst->ISACdec_obj.bitstr_obj).stream[k] = (WebRtc_UWord16) ((encoded[k] >> 8)|((encoded[k] & 0xFF)<<8));
 | |
|   }
 | |
|   if (len & 0x0001)
 | |
|     (ISAC_inst->ISACdec_obj.bitstr_obj).stream[k] = (WebRtc_UWord16) ((encoded[k] & 0xFF)<<8);
 | |
| #endif
 | |
| 
 | |
|   /* added for NetEq purposes (VAD/DTX related) */
 | |
|   *speechType=1;
 | |
| 
 | |
|   declen = WebRtcIsacfix_DecodeImpl(decoded,&ISAC_inst->ISACdec_obj, &number_of_samples);
 | |
| 
 | |
|   if (declen < 0) {
 | |
|     /* Some error inside the decoder */
 | |
|     ISAC_inst->errorcode = -declen;
 | |
|     memset(decoded, 0, sizeof(WebRtc_Word16) * FRAMESAMPLES);
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   /* error check */
 | |
| 
 | |
|   if (declen & 0x0001) {
 | |
|     if (len != declen && len != declen + (((ISAC_inst->ISACdec_obj.bitstr_obj).stream[declen>>1]) & 0x00FF) ) {
 | |
|       ISAC_inst->errorcode = ISAC_LENGTH_MISMATCH;
 | |
|       memset(decoded, 0, sizeof(WebRtc_Word16) * number_of_samples);
 | |
|       return -1;
 | |
|     }
 | |
|   } else {
 | |
|     if (len != declen && len != declen + (((ISAC_inst->ISACdec_obj.bitstr_obj).stream[declen>>1]) >> 8) ) {
 | |
|       ISAC_inst->errorcode = ISAC_LENGTH_MISMATCH;
 | |
|       memset(decoded, 0, sizeof(WebRtc_Word16) * number_of_samples);
 | |
|       return -1;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   WebRtcIsacfix_SplitAndFilter2(decoded, decoded, dummy, &ISAC_inst->ISACdec_obj.decimatorstr_obj);
 | |
| 
 | |
|   if (number_of_samples>FRAMESAMPLES) {
 | |
|     WebRtcIsacfix_SplitAndFilter2(decoded + FRAMESAMPLES, decoded + FRAMESAMPLES/2,
 | |
|                                   dummy, &ISAC_inst->ISACdec_obj.decimatorstr_obj);
 | |
|   }
 | |
| 
 | |
|   return number_of_samples/2;
 | |
| 
 | |
| #else /* NB_CALLS not defined */
 | |
|   return -1;
 | |
| #endif /* NB_CALLS */
 | |
| }
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_DecodePlcNb(...)
 | |
|  *
 | |
|  * This function conducts PLC for ISAC frame(s) in narrow-band (8kHz sampling).
 | |
|  * Output speech length  will be "240*noOfLostFrames" samples
 | |
|  * that is equevalent of "30*noOfLostFrames" millisecond.
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst    : ISAC instance.
 | |
|  *      - noOfLostFrames    : Number of PLC frames (240 sample=30ms) to produce
 | |
|  *
 | |
|  * Output:
 | |
|  *      - decoded           : The decoded vector
 | |
|  *
 | |
|  * Return value             : >0 - number of samples in decoded PLC vector
 | |
|  *                            -1 - Error
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_DecodePlcNb(ISACFIX_MainStruct *ISAC_main_inst,
 | |
|                                          WebRtc_Word16          *decoded,
 | |
|                                          WebRtc_Word16 noOfLostFrames )
 | |
| {
 | |
| 
 | |
| #ifdef NB_CALLS
 | |
| 
 | |
|   WebRtc_Word16 no_of_samples, declen, k, ok;
 | |
|   WebRtc_Word16 outframeNB[FRAMESAMPLES];
 | |
|   WebRtc_Word16 outframeWB[FRAMESAMPLES];
 | |
|   WebRtc_Word16 dummy[FRAMESAMPLES/2];
 | |
| 
 | |
| 
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
|   /* typecast pointer to real structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
|   /* Limit number of frames to two = 60 msec. Otherwise we exceed data vectors */
 | |
|   if (noOfLostFrames > 2){
 | |
|     noOfLostFrames = 2;
 | |
|   }
 | |
| 
 | |
|   k = 0;
 | |
|   declen = 0;
 | |
|   while( noOfLostFrames > 0 )
 | |
|   {
 | |
|     ok = WebRtcIsacfix_DecodePlcImpl( outframeWB, &ISAC_inst->ISACdec_obj, &no_of_samples );
 | |
|     if(ok)
 | |
|       return -1;
 | |
| 
 | |
|     WebRtcIsacfix_SplitAndFilter2(outframeWB, &(outframeNB[k*240]), dummy, &ISAC_inst->ISACdec_obj.decimatorstr_obj);
 | |
| 
 | |
|     declen += no_of_samples;
 | |
|     noOfLostFrames--;
 | |
|     k++;
 | |
|   }
 | |
| 
 | |
|   declen>>=1;
 | |
| 
 | |
|   for (k=0;k<declen;k++) {
 | |
|     decoded[k] = outframeNB[k];
 | |
|   }
 | |
| 
 | |
|   return declen;
 | |
| 
 | |
| 
 | |
| #else /* NB_CALLS not defined */
 | |
|   return -1;
 | |
| #endif /* NB_CALLS */
 | |
| 
 | |
| 
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_DecodePlc(...)
 | |
|  *
 | |
|  * This function conducts PLC for ISAC frame(s) in wide-band (16kHz sampling).
 | |
|  * Output speech length  will be "480*noOfLostFrames" samples
 | |
|  * that is equevalent of "30*noOfLostFrames" millisecond.
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst    : ISAC instance.
 | |
|  *      - noOfLostFrames    : Number of PLC frames (480sample = 30ms)
 | |
|  *                                to produce
 | |
|  *
 | |
|  * Output:
 | |
|  *      - decoded           : The decoded vector
 | |
|  *
 | |
|  * Return value             : >0 - number of samples in decoded PLC vector
 | |
|  *                            -1 - Error
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_DecodePlc(ISACFIX_MainStruct *ISAC_main_inst,
 | |
|                                       WebRtc_Word16          *decoded,
 | |
|                                       WebRtc_Word16 noOfLostFrames)
 | |
| {
 | |
| 
 | |
|   WebRtc_Word16 no_of_samples, declen, k, ok;
 | |
|   WebRtc_Word16 outframe16[MAX_FRAMESAMPLES];
 | |
| 
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
|   /* typecast pointer to real structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
|   /* Limit number of frames to two = 60 msec. Otherwise we exceed data vectors */
 | |
|   if (noOfLostFrames > 2) {
 | |
|     noOfLostFrames = 2;
 | |
|   }
 | |
|   k = 0;
 | |
|   declen = 0;
 | |
|   while( noOfLostFrames > 0 )
 | |
|   {
 | |
|     ok = WebRtcIsacfix_DecodePlcImpl( &(outframe16[k*480]), &ISAC_inst->ISACdec_obj, &no_of_samples );
 | |
|     if(ok)
 | |
|       return -1;
 | |
|     declen += no_of_samples;
 | |
|     noOfLostFrames--;
 | |
|     k++;
 | |
|   }
 | |
| 
 | |
|   for (k=0;k<declen;k++) {
 | |
|     decoded[k] = outframe16[k];
 | |
|   }
 | |
| 
 | |
|   return declen;
 | |
| }
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_Control(...)
 | |
|  *
 | |
|  * This function sets the limit on the short-term average bit rate and the
 | |
|  * frame length. Should be used only in Instantaneous mode.
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst    : ISAC instance.
 | |
|  *      - rate              : limit on the short-term average bit rate,
 | |
|  *                            in bits/second (between 10000 and 32000)
 | |
|  *      - framesize         : number of milliseconds per frame (30 or 60)
 | |
|  *
 | |
|  * Return value             : 0  - ok
 | |
|  *                            -1 - Error
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_Control(ISACFIX_MainStruct *ISAC_main_inst,
 | |
|                                     WebRtc_Word16          rate,
 | |
|                                     WebRtc_Word16          framesize)
 | |
| {
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
|   /* typecast pointer to real structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
|   if (ISAC_inst->CodingMode == 0)
 | |
|   {
 | |
|     /* in adaptive mode */
 | |
|     ISAC_inst->errorcode = ISAC_MODE_MISMATCH;
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
| 
 | |
|   if (rate >= 10000 && rate <= 32000)
 | |
|     ISAC_inst->ISACenc_obj.BottleNeck = rate;
 | |
|   else {
 | |
|     ISAC_inst->errorcode = ISAC_DISALLOWED_BOTTLENECK;
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
| 
 | |
| 
 | |
|   if (framesize  == 30 || framesize == 60)
 | |
|     ISAC_inst->ISACenc_obj.new_framelength = (FS/1000) * framesize;
 | |
|   else {
 | |
|     ISAC_inst->errorcode = ISAC_DISALLOWED_FRAME_LENGTH;
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_ControlBwe(...)
 | |
|  *
 | |
|  * This function sets the initial values of bottleneck and frame-size if
 | |
|  * iSAC is used in channel-adaptive mode. Through this API, users can
 | |
|  * enforce a frame-size for all values of bottleneck. Then iSAC will not
 | |
|  * automatically change the frame-size.
 | |
|  *
 | |
|  *
 | |
|  * Input:
 | |
|  *  - ISAC_main_inst : ISAC instance.
 | |
|  *      - rateBPS           : initial value of bottleneck in bits/second
 | |
|  *                            10000 <= rateBPS <= 32000 is accepted
 | |
|  *                            For default bottleneck set rateBPS = 0
 | |
|  *      - frameSizeMs       : number of milliseconds per frame (30 or 60)
 | |
|  *      - enforceFrameSize  : 1 to enforce the given frame-size through out
 | |
|  *                            the adaptation process, 0 to let iSAC change
 | |
|  *                            the frame-size if required.
 | |
|  *
 | |
|  * Return value    : 0  - ok
 | |
|  *         -1 - Error
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_ControlBwe(ISACFIX_MainStruct *ISAC_main_inst,
 | |
|                                         WebRtc_Word16 rateBPS,
 | |
|                                         WebRtc_Word16 frameSizeMs,
 | |
|                                         WebRtc_Word16 enforceFrameSize)
 | |
| {
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
|   /* Typecast pointer to real structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
|   /* check if encoder initiated */
 | |
|   if ((ISAC_inst->initflag & 2) != 2) {
 | |
|     ISAC_inst->errorcode = ISAC_ENCODER_NOT_INITIATED;
 | |
|     return (-1);
 | |
|   }
 | |
| 
 | |
|   /* Check that we are in channel-adaptive mode, otherwise, return -1 */
 | |
|   if (ISAC_inst->CodingMode != 0) {
 | |
|     ISAC_inst->errorcode = ISAC_MODE_MISMATCH;
 | |
|     return (-1);
 | |
|   }
 | |
| 
 | |
|   /* Set struct variable if enforceFrameSize is set. ISAC will then keep the */
 | |
|   /* chosen frame size.                                                      */
 | |
|   ISAC_inst->ISACenc_obj.enforceFrameSize = (enforceFrameSize != 0)? 1:0;
 | |
| 
 | |
|   /* Set initial rate, if value between 10000 and 32000,                */
 | |
|   /* if rateBPS is 0, keep the default initial bottleneck value (15000) */
 | |
|   if ((rateBPS >= 10000) && (rateBPS <= 32000)) {
 | |
|     ISAC_inst->bwestimator_obj.sendBwAvg = (((WebRtc_UWord32)rateBPS) << 7);
 | |
|   } else if (rateBPS != 0) {
 | |
|     ISAC_inst->errorcode = ISAC_DISALLOWED_BOTTLENECK;
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   /* Set initial framesize. If enforceFrameSize is set the frame size will not change */
 | |
|   if ((frameSizeMs  == 30) || (frameSizeMs == 60)) {
 | |
|     ISAC_inst->ISACenc_obj.new_framelength = (FS/1000) * frameSizeMs;
 | |
|   } else {
 | |
|     ISAC_inst->errorcode = ISAC_DISALLOWED_FRAME_LENGTH;
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_GetDownLinkBwIndex(...)
 | |
|  *
 | |
|  * This function returns index representing the Bandwidth estimate from
 | |
|  * other side to this side.
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst: iSAC struct
 | |
|  *
 | |
|  * Output:
 | |
|  *      - rateIndex     : Bandwidth estimate to transmit to other side.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_GetDownLinkBwIndex(ISACFIX_MainStruct* ISAC_main_inst,
 | |
|                                        WebRtc_Word16*     rateIndex)
 | |
| {
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
| 
 | |
|   /* typecast pointer to real structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
|   /* Call function to get Bandwidth Estimate */
 | |
|   *rateIndex = WebRtcIsacfix_GetDownlinkBwIndexImpl(&ISAC_inst->bwestimator_obj);
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_UpdateUplinkBw(...)
 | |
|  *
 | |
|  * This function takes an index representing the Bandwidth estimate from
 | |
|  * this side to other side and updates BWE.
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst: iSAC struct
 | |
|  *      - rateIndex     : Bandwidth estimate from other side.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_UpdateUplinkBw(ISACFIX_MainStruct* ISAC_main_inst,
 | |
|                                    WebRtc_Word16     rateIndex)
 | |
| {
 | |
|   WebRtc_Word16 err = 0;
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
| 
 | |
|   /* typecast pointer to real structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
|   /* Call function to update BWE with received Bandwidth Estimate */
 | |
|   err = WebRtcIsacfix_UpdateUplinkBwRec(&ISAC_inst->bwestimator_obj, rateIndex);
 | |
|   if (err < 0) {
 | |
|     ISAC_inst->errorcode = -err;
 | |
|     return (-1);
 | |
|   }
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_ReadFrameLen(...)
 | |
|  *
 | |
|  * This function returns the length of the frame represented in the packet.
 | |
|  *
 | |
|  * Input:
 | |
|  *      - encoded       : Encoded bitstream
 | |
|  *
 | |
|  * Output:
 | |
|  *      - frameLength   : Length of frame in packet (in samples)
 | |
|  *
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_ReadFrameLen(const WebRtc_Word16* encoded,
 | |
|                                         WebRtc_Word16* frameLength)
 | |
| {
 | |
|   Bitstr_dec streamdata;
 | |
|   WebRtc_UWord16 partOfStream[5];
 | |
| #ifndef WEBRTC_BIG_ENDIAN
 | |
|   int k;
 | |
| #endif
 | |
|   WebRtc_Word16 err;
 | |
| 
 | |
|   /* Set stream pointer to point at partOfStream */
 | |
|   streamdata.stream = (WebRtc_UWord16 *)partOfStream;
 | |
| 
 | |
|   streamdata.W_upper = 0xFFFFFFFF;
 | |
|   streamdata.streamval = 0;
 | |
|   streamdata.stream_index = 0;
 | |
|   streamdata.full = 1;
 | |
| 
 | |
| #ifndef WEBRTC_BIG_ENDIAN
 | |
|   for (k=0; k<5; k++) {
 | |
|     streamdata.stream[k] = (WebRtc_UWord16) (((WebRtc_UWord16)encoded[k] >> 8)|((encoded[k] & 0xFF)<<8));
 | |
|   }
 | |
| #else
 | |
|   memcpy(streamdata.stream, encoded, 5);
 | |
| #endif
 | |
| 
 | |
|   /* decode frame length */
 | |
|   err = WebRtcIsacfix_DecodeFrameLen(&streamdata, frameLength);
 | |
|   if (err<0)  // error check
 | |
|     return err;
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_ReadBwIndex(...)
 | |
|  *
 | |
|  * This function returns the index of the Bandwidth estimate from the bitstream.
 | |
|  *
 | |
|  * Input:
 | |
|  *      - encoded       : Encoded bitstream
 | |
|  *
 | |
|  * Output:
 | |
|  *      - frameLength   : Length of frame in packet (in samples)
 | |
|  *      - rateIndex     : Bandwidth estimate in bitstream
 | |
|  *
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_ReadBwIndex(const WebRtc_Word16* encoded,
 | |
|                                    WebRtc_Word16* rateIndex)
 | |
| {
 | |
|   Bitstr_dec streamdata;
 | |
|   WebRtc_UWord16 partOfStream[5];
 | |
| #ifndef WEBRTC_BIG_ENDIAN
 | |
|   int k;
 | |
| #endif
 | |
|   WebRtc_Word16 err;
 | |
| 
 | |
|   /* Set stream pointer to point at partOfStream */
 | |
|   streamdata.stream = (WebRtc_UWord16 *)partOfStream;
 | |
| 
 | |
|   streamdata.W_upper = 0xFFFFFFFF;
 | |
|   streamdata.streamval = 0;
 | |
|   streamdata.stream_index = 0;
 | |
|   streamdata.full = 1;
 | |
| 
 | |
| #ifndef WEBRTC_BIG_ENDIAN
 | |
|   for (k=0; k<5; k++) {
 | |
|     streamdata.stream[k] = (WebRtc_UWord16) (((WebRtc_UWord16)encoded[k] >> 8)|((encoded[k] & 0xFF)<<8));
 | |
|   }
 | |
| #else
 | |
|   memcpy(streamdata.stream, encoded, 5);
 | |
| #endif
 | |
| 
 | |
|   /* decode frame length, needed to get to the rateIndex in the bitstream */
 | |
|   err = WebRtcIsacfix_DecodeFrameLen(&streamdata, rateIndex);
 | |
|   if (err<0)  // error check
 | |
|     return err;
 | |
| 
 | |
|   /* decode BW estimation */
 | |
|   err = WebRtcIsacfix_DecodeSendBandwidth(&streamdata, rateIndex);
 | |
|   if (err<0)  // error check
 | |
|     return err;
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_GetErrorCode(...)
 | |
|  *
 | |
|  * This function can be used to check the error code of an iSAC instance. When
 | |
|  * a function returns -1 a error code will be set for that instance. The
 | |
|  * function below extract the code of the last error that occured in the
 | |
|  * specified instance.
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst    : ISAC instance
 | |
|  *
 | |
|  * Return value             : Error code
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_GetErrorCode(ISACFIX_MainStruct *ISAC_main_inst)
 | |
| {
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
|   /* typecast pointer to real structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
|   return ISAC_inst->errorcode;
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_GetUplinkBw(...)
 | |
|  *
 | |
|  * This function returns the inst quantized iSAC send bitrate
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst    : iSAC instance
 | |
|  *
 | |
|  * Return value             : bitrate
 | |
|  */
 | |
| 
 | |
| WebRtc_Word32 WebRtcIsacfix_GetUplinkBw(ISACFIX_MainStruct *ISAC_main_inst)
 | |
| {
 | |
|   ISACFIX_SubStruct *ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
|   BwEstimatorstr * bw = (BwEstimatorstr*)&(ISAC_inst->bwestimator_obj);
 | |
| 
 | |
|   return (WebRtc_Word32) WebRtcIsacfix_GetUplinkBandwidth(bw);
 | |
| }
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_GetNewFrameLen(...)
 | |
|  *
 | |
|  * This function return the next frame length (in samples) of iSAC.
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst    : iSAC instance
 | |
|  *
 | |
|  * Return value             :  frame lenght in samples
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_GetNewFrameLen(ISACFIX_MainStruct *ISAC_main_inst)
 | |
| {
 | |
|   ISACFIX_SubStruct *ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
|   return ISAC_inst->ISACenc_obj.new_framelength;
 | |
| }
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_SetMaxPayloadSize(...)
 | |
|  *
 | |
|  * This function sets a limit for the maximum payload size of iSAC. The same
 | |
|  * value is used both for 30 and 60 msec packets.
 | |
|  * The absolute max will be valid until next time the function is called.
 | |
|  * NOTE! This function may override the function WebRtcIsacfix_SetMaxRate()
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst    : iSAC instance
 | |
|  *      - maxPayloadBytes   : maximum size of the payload in bytes
 | |
|  *                            valid values are between 100 and 400 bytes
 | |
|  *
 | |
|  *
 | |
|  * Return value             : 0 if sucessful
 | |
|  *                           -1 if error happens
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_SetMaxPayloadSize(ISACFIX_MainStruct *ISAC_main_inst,
 | |
|                                               WebRtc_Word16 maxPayloadBytes)
 | |
| {
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
| 
 | |
|   /* typecast pointer to real structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
|   if((maxPayloadBytes < 100) || (maxPayloadBytes > 400))
 | |
|   {
 | |
|     /* maxPayloadBytes is out of valid range */
 | |
|     return -1;
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     /* Set new absolute max, which will not change unless this function
 | |
|        is called again with a new value */
 | |
|     ISAC_inst->ISACenc_obj.maxPayloadBytes = maxPayloadBytes;
 | |
| 
 | |
|     /* Set new maximum values for 30 and 60 msec packets */
 | |
|     if (maxPayloadBytes < ISAC_inst->ISACenc_obj.maxRateInBytes) {
 | |
|       ISAC_inst->ISACenc_obj.payloadLimitBytes30 = maxPayloadBytes;
 | |
|     } else {
 | |
|       ISAC_inst->ISACenc_obj.payloadLimitBytes30 = ISAC_inst->ISACenc_obj.maxRateInBytes;
 | |
|     }
 | |
| 
 | |
|     if ( maxPayloadBytes < (ISAC_inst->ISACenc_obj.maxRateInBytes << 1)) {
 | |
|       ISAC_inst->ISACenc_obj.payloadLimitBytes60 = maxPayloadBytes;
 | |
|     } else {
 | |
|       ISAC_inst->ISACenc_obj.payloadLimitBytes60 = (ISAC_inst->ISACenc_obj.maxRateInBytes << 1);
 | |
|     }
 | |
|   }
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_SetMaxRate(...)
 | |
|  *
 | |
|  * This function sets the maximum rate which the codec may not exceed for a
 | |
|  * singel packet. The maximum rate is set in bits per second.
 | |
|  * The codec has an absolute maximum rate of 53400 bits per second (200 bytes
 | |
|  * per 30 msec).
 | |
|  * It is possible to set a maximum rate between 32000 and 53400 bits per second.
 | |
|  *
 | |
|  * The rate limit is valid until next time the function is called.
 | |
|  *
 | |
|  * NOTE! Packet size will never go above the value set if calling
 | |
|  * WebRtcIsacfix_SetMaxPayloadSize() (default max packet size is 400 bytes).
 | |
|  *
 | |
|  * Input:
 | |
|  *      - ISAC_main_inst    : iSAC instance
 | |
|  *      - maxRateInBytes    : maximum rate in bits per second,
 | |
|  *                            valid values are 32000 to 53400 bits
 | |
|  *
 | |
|  * Return value             : 0 if sucessful
 | |
|  *                           -1 if error happens
 | |
|  */
 | |
| 
 | |
| WebRtc_Word16 WebRtcIsacfix_SetMaxRate(ISACFIX_MainStruct *ISAC_main_inst,
 | |
|                                        WebRtc_Word32 maxRate)
 | |
| {
 | |
|   ISACFIX_SubStruct *ISAC_inst;
 | |
|   WebRtc_Word16 maxRateInBytes;
 | |
| 
 | |
|   /* typecast pointer to real structure */
 | |
|   ISAC_inst = (ISACFIX_SubStruct *)ISAC_main_inst;
 | |
| 
 | |
|   if((maxRate < 32000) || (maxRate > 53400))
 | |
|   {
 | |
|     /* maxRate is out of valid range */
 | |
|     return -1;
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     /* Calculate maximum number of bytes per 30 msec packets for the given
 | |
|        maximum rate. Multiply with 30/1000 to get number of bits per 30 msec,
 | |
|        divide by 8 to get number of bytes per 30 msec:
 | |
|        maxRateInBytes = floor((maxRate * 30/1000) / 8); */
 | |
|     maxRateInBytes = (WebRtc_Word16)( WebRtcSpl_DivW32W16ResW16(WEBRTC_SPL_MUL(maxRate, 3), 800) );
 | |
| 
 | |
|     /* Store the value for usage in the WebRtcIsacfix_SetMaxPayloadSize-function */
 | |
|     ISAC_inst->ISACenc_obj.maxRateInBytes = maxRateInBytes;
 | |
| 
 | |
|     /* For 30 msec packets: if the new limit is below the maximum
 | |
|        payload size, set a new limit */
 | |
|     if (maxRateInBytes < ISAC_inst->ISACenc_obj.maxPayloadBytes) {
 | |
|       ISAC_inst->ISACenc_obj.payloadLimitBytes30 = maxRateInBytes;
 | |
|     } else {
 | |
|       ISAC_inst->ISACenc_obj.payloadLimitBytes30 = ISAC_inst->ISACenc_obj.maxPayloadBytes;
 | |
|     }
 | |
| 
 | |
|     /* For 60 msec packets: if the new limit (times 2) is below the
 | |
|        maximum payload size, set a new limit */
 | |
|     if ( (maxRateInBytes << 1) < ISAC_inst->ISACenc_obj.maxPayloadBytes) {
 | |
|       ISAC_inst->ISACenc_obj.payloadLimitBytes60 = (maxRateInBytes << 1);
 | |
|     } else {
 | |
|       ISAC_inst->ISACenc_obj.payloadLimitBytes60 = ISAC_inst->ISACenc_obj.maxPayloadBytes;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * WebRtcIsacfix_version(...)
 | |
|  *
 | |
|  * This function returns the version number.
 | |
|  *
 | |
|  * Output:
 | |
|  *      - version  : Pointer to character string
 | |
|  *
 | |
|  */
 | |
| 
 | |
| void WebRtcIsacfix_version(char *version)
 | |
| {
 | |
|   strcpy(version, "3.6.0");
 | |
| }
 | 
