blob: adbb9c5be60727a08fa0201850dfbc3415b3c8e3 [file] [log] [blame]
/*
* 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.
*/
#include "acm_codec_database.h"
#include "acm_common_defs.h"
#include "acm_isac.h"
#include "acm_neteq.h"
#include "trace.h"
#include "webrtc_neteq.h"
#include "webrtc_neteq_help_macros.h"
#ifdef WEBRTC_CODEC_ISAC
#include "acm_isac_macros.h"
#include "isac.h"
#endif
#ifdef WEBRTC_CODEC_ISACFX
#include "acm_isac_macros.h"
#include "isacfix.h"
#endif
namespace webrtc
{
// we need this otherwise we cannot use forward declaration
// in the header file
#if (defined(WEBRTC_CODEC_ISAC) || defined(WEBRTC_CODEC_ISACFX))
struct ACMISACInst
{
ACM_ISAC_STRUCT *inst;
};
#endif
#define ISAC_MIN_RATE 10000
#define ISAC_MAX_RATE 56000
// How the scaling is computed. iSAC computes a gain based on the
// bottleneck. It follows the following expression for that
//
// G(BN_kbps) = pow(10, (a + b * BN_kbps + c * BN_kbps * BN_kbps) / 20.0)
// / 3.4641;
//
// Where for 30 ms framelength we have,
//
// a = -23; b = 0.48; c = 0;
//
// As the default encoder is operating at 32kbps we have the scale as
//
// S(BN_kbps) = G(BN_kbps) / G(32);
#define ISAC_NUM_SUPPORTED_RATES 9
const WebRtc_UWord16 isacSuportedRates[ISAC_NUM_SUPPORTED_RATES] = {
32000, 30000, 26000, 23000, 21000,
19000, 17000, 15000, 12000};
const float isacScale[ISAC_NUM_SUPPORTED_RATES] = {
1.0f, 0.8954f, 0.7178f, 0.6081f, 0.5445f,
0.4875f, 0.4365f, 0.3908f, 0.3311f};
// Tables for bandwidth estimates
#define NR_ISAC_BANDWIDTHS 24
const WebRtc_Word32 isacRatesWB[NR_ISAC_BANDWIDTHS] =
{
10000, 11100, 12300, 13700, 15200, 16900,
18800, 20900, 23300, 25900, 28700, 31900,
10100, 11200, 12400, 13800, 15300, 17000,
18900, 21000, 23400, 26000, 28800, 32000};
const WebRtc_Word32 isacRatesSWB[NR_ISAC_BANDWIDTHS] =
{
10000, 11000, 12400, 13800, 15300, 17000,
18900, 21000, 23200, 25400, 27600, 29800,
32000, 34100, 36300, 38500, 40700, 42900,
45100, 47300, 49500, 51700, 53900, 56000,
};
#if (!defined(WEBRTC_CODEC_ISAC) && !defined(WEBRTC_CODEC_ISACFX))
ACMISAC::ACMISAC(WebRtc_Word16 /* codecID */)
: _codecInstPtr(NULL),
_isEncInitialized(false),
_isacCodingMode(CHANNEL_INDEPENDENT),
_enforceFrameSize(false),
_isacCurrentBN(32000),
_samplesIn10MsAudio(160) { // Initiates to 16 kHz mode.
// Initiate decoder parameters for the 32 kHz mode.
memset(&_decoderParams32kHz, 0, sizeof(WebRtcACMCodecParams));
_decoderParams32kHz.codecInstant.pltype = -1;
return;
}
ACMISAC::~ACMISAC()
{
return;
}
ACMGenericCodec*
ACMISAC::CreateInstance(void)
{
return NULL;
}
WebRtc_Word16
ACMISAC::InternalEncode(
WebRtc_UWord8* /* bitstream */,
WebRtc_Word16* /* bitStreamLenByte */)
{
return -1;
}
WebRtc_Word16
ACMISAC::DecodeSafe(
WebRtc_UWord8* /* bitStream */,
WebRtc_Word16 /* bitStreamLenByte */,
WebRtc_Word16* /* audio */,
WebRtc_Word16* /* audioSamples */,
WebRtc_Word8* /* speechType */)
{
return 0;
}
WebRtc_Word16
ACMISAC::InternalInitEncoder(
WebRtcACMCodecParams* /* codecParams */)
{
return -1;
}
WebRtc_Word16
ACMISAC::InternalInitDecoder(
WebRtcACMCodecParams* /* codecParams */)
{
return -1;
}
WebRtc_Word16
ACMISAC::InternalCreateDecoder()
{
return -1;
}
void
ACMISAC::DestructDecoderSafe()
{
return;
}
WebRtc_Word16
ACMISAC::InternalCreateEncoder()
{
return -1;
}
void
ACMISAC::DestructEncoderSafe()
{
return;
}
WebRtc_Word32
ACMISAC::CodecDef(
WebRtcNetEQ_CodecDef& /* codecDef */,
const CodecInst& /* codecInst */)
{
return -1;
}
void
ACMISAC::InternalDestructEncoderInst(
void* /* ptrInst */)
{
return;
}
WebRtc_Word16
ACMISAC::DeliverCachedIsacData(
WebRtc_UWord8* /* bitStream */,
WebRtc_Word16* /* bitStreamLenByte */,
WebRtc_UWord32* /* timestamp */,
WebRtcACMEncodingType* /* encodingType */,
const WebRtc_UWord16 /* isacRate */,
const WebRtc_UWord8 /* isacBWestimate */)
{
return -1;
}
WebRtc_Word16
ACMISAC::Transcode(
WebRtc_UWord8* /* bitStream */,
WebRtc_Word16* /* bitStreamLenByte */,
WebRtc_Word16 /* qBWE */,
WebRtc_Word32 /* scale */,
bool /* isRED */)
{
return -1;
}
WebRtc_Word16
ACMISAC::SetBitRateSafe(
WebRtc_Word32 /* bitRate */)
{
return -1;
}
WebRtc_Word32
ACMISAC::GetEstimatedBandwidthSafe()
{
return -1;
}
WebRtc_Word32
ACMISAC::SetEstimatedBandwidthSafe(
WebRtc_Word32 /* estimatedBandwidth */)
{
return -1;
}
WebRtc_Word32
ACMISAC::GetRedPayloadSafe(
WebRtc_UWord8* /* redPayload */,
WebRtc_Word16* /* payloadBytes */)
{
return -1;
}
WebRtc_Word16
ACMISAC::UnregisterFromNetEqSafe(
ACMNetEQ* /* netEq */,
WebRtc_Word16 /* payloadType */)
{
return -1;
}
WebRtc_Word16
ACMISAC::UpdateDecoderSampFreq(
WebRtc_Word16 /* codecId */)
{
return -1;
}
WebRtc_Word16
ACMISAC::UpdateEncoderSampFreq(
WebRtc_UWord16 /* encoderSampFreqHz */)
{
return -1;
}
WebRtc_Word16
ACMISAC::EncoderSampFreq(
WebRtc_UWord16& /* sampFreqHz */)
{
return -1;
}
WebRtc_Word32
ACMISAC::ConfigISACBandwidthEstimator(
const WebRtc_UWord8 /* initFrameSizeMsec */,
const WebRtc_UWord16 /* initRateBitPerSec */,
const bool /* enforceFrameSize */)
{
return -1;
}
WebRtc_Word32
ACMISAC::SetISACMaxPayloadSize(
const WebRtc_UWord16 /* maxPayloadLenBytes */)
{
return -1;
}
WebRtc_Word32
ACMISAC::SetISACMaxRate(
const WebRtc_UWord32 /* maxRateBitPerSec */)
{
return -1;
}
void
ACMISAC::UpdateFrameLen()
{
return;
}
void
ACMISAC::CurrentRate(
WebRtc_Word32& /*rateBitPerSec */)
{
return;
}
bool
ACMISAC::DecoderParamsSafe(
WebRtcACMCodecParams* /* decParams */,
const WebRtc_UWord8 /* payloadType */)
{
return false;
}
void
ACMISAC::SaveDecoderParamSafe(
const WebRtcACMCodecParams* /* codecParams */)
{
return;
}
WebRtc_Word16
ACMISAC::REDPayloadISAC(
const WebRtc_Word32 /* isacRate */,
const WebRtc_Word16 /* isacBwEstimate */,
WebRtc_UWord8* /* payload */,
WebRtc_Word16* /* payloadLenBytes */)
{
return -1;
}
#else //===================== Actual Implementation =======================
#ifdef WEBRTC_CODEC_ISACFX
enum IsacSamplingRate
{
kIsacWideband = 16,
kIsacSuperWideband = 32
};
static float
ACMISACFixTranscodingScale(
WebRtc_UWord16 rate)
{
// find the scale for transcoding, the scale is rounded
// downward
float scale = -1;
for(WebRtc_Word16 n=0; n < ISAC_NUM_SUPPORTED_RATES; n++)
{
if(rate >= isacSuportedRates[n])
{
scale = isacScale[n];
break;
}
}
return scale;
}
static void
ACMISACFixGetSendBitrate(
ACM_ISAC_STRUCT* inst,
WebRtc_Word32* bottleNeck)
{
*bottleNeck = WebRtcIsacfix_GetUplinkBw(inst);
}
static WebRtc_Word16
ACMISACFixGetNewBitstream(
ACM_ISAC_STRUCT* inst,
WebRtc_Word16 BWEIndex,
WebRtc_Word16 /* jitterIndex */,
WebRtc_Word32 rate,
WebRtc_Word16* bitStream,
bool isRED)
{
if (isRED)
{
// RED not supported with iSACFIX
return -1;
}
float scale = ACMISACFixTranscodingScale((WebRtc_UWord16)rate);
return WebRtcIsacfix_GetNewBitStream(inst, BWEIndex, scale, bitStream);
}
static WebRtc_Word16
ACMISACFixGetSendBWE(
ACM_ISAC_STRUCT* inst,
WebRtc_Word16* rateIndex,
WebRtc_Word16* /* dummy */)
{
WebRtc_Word16 localRateIndex;
WebRtc_Word16 status = WebRtcIsacfix_GetDownLinkBwIndex(inst, &localRateIndex);
if(status < 0)
{
return -1;
}
else
{
*rateIndex = localRateIndex;
return 0;
}
}
static WebRtc_Word16
ACMISACFixControlBWE(
ACM_ISAC_STRUCT* inst,
WebRtc_Word32 rateBPS,
WebRtc_Word16 frameSizeMs,
WebRtc_Word16 enforceFrameSize)
{
return WebRtcIsacfix_ControlBwe(inst, (WebRtc_Word16)rateBPS,
frameSizeMs, enforceFrameSize);
}
static WebRtc_Word16
ACMISACFixControl(
ACM_ISAC_STRUCT* inst,
WebRtc_Word32 rateBPS,
WebRtc_Word16 frameSizeMs)
{
return WebRtcIsacfix_Control(inst, (WebRtc_Word16)rateBPS,
frameSizeMs);
}
static IsacSamplingRate
ACMISACFixGetEncSampRate(
ACM_ISAC_STRUCT* /* inst */)
{
return kIsacWideband;
}
static IsacSamplingRate
ACMISACFixGetDecSampRate(
ACM_ISAC_STRUCT* /* inst */)
{
return kIsacWideband;
}
#endif
ACMISAC::ACMISAC(WebRtc_Word16 codecID)
: _isEncInitialized(false),
_isacCodingMode(CHANNEL_INDEPENDENT),
_enforceFrameSize(false),
_isacCurrentBN(32000),
_samplesIn10MsAudio(160) { // Initiates to 16 kHz mode.
_codecID = codecID;
// Create codec instance.
_codecInstPtr = new ACMISACInst;
if (_codecInstPtr == NULL) {
return;
}
_codecInstPtr->inst = NULL;
// Initiate decoder parameters for the 32 kHz mode.
memset(&_decoderParams32kHz, 0, sizeof(WebRtcACMCodecParams));
_decoderParams32kHz.codecInstant.pltype = -1;
// TODO(tlegrand): Check if the following is really needed, now that
// ACMGenericCodec has been updated to initialize this value.
// Initialize values that can be used uninitialized otherwise
_decoderParams.codecInstant.pltype = -1;
}
ACMISAC::~ACMISAC()
{
if (_codecInstPtr != NULL)
{
if(_codecInstPtr->inst != NULL)
{
ACM_ISAC_FREE(_codecInstPtr->inst);
_codecInstPtr->inst = NULL;
}
delete _codecInstPtr;
_codecInstPtr = NULL;
}
return;
}
ACMGenericCodec*
ACMISAC::CreateInstance(void)
{
return NULL;
}
WebRtc_Word16
ACMISAC::InternalEncode(
WebRtc_UWord8* bitstream,
WebRtc_Word16* bitStreamLenByte)
{
// ISAC takes 10ms audio everytime we call encoder, therefor,
// it should be treated like codecs with 'basic coding block'
// non-zero, and the following 'while-loop' should not be necessary.
// However, due to a mistake in the codec the frame-size might change
// at the first 10ms pushed in to iSAC if the bit-rate is low, this is
// sort of a bug in iSAC. to address this we treat iSAC as the
// following.
if (_codecInstPtr == NULL)
{
return -1;
}
*bitStreamLenByte = 0;
while((*bitStreamLenByte == 0) && (_inAudioIxRead < _frameLenSmpl))
{
if(_inAudioIxRead > _inAudioIxWrite)
{
// something is wrong.
WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceAudioCoding, _uniqueID,
"The actual fram-size of iSAC appears to be larger that expected. All audio \
pushed in but no bit-stream is generated.");
return -1;
}
*bitStreamLenByte = ACM_ISAC_ENCODE(_codecInstPtr->inst,
&_inAudio[_inAudioIxRead], (WebRtc_Word16*)bitstream);
// increment the read index this tell the caller that how far
// we have gone forward in reading the audio buffer
_inAudioIxRead += _samplesIn10MsAudio;
}
if(*bitStreamLenByte == 0)
{
WEBRTC_TRACE(webrtc::kTraceWarning, webrtc::kTraceAudioCoding, _uniqueID,
"ISAC Has encoded the whole frame but no bit-stream is generated.");
}
// a packet is generated iSAC, is set in adaptive mode may change
// the frame length and we like to update the bottleneck value as
// well, although updating bottleneck is not crucial
if((*bitStreamLenByte > 0) && (_isacCodingMode == ADAPTIVE))
{
//_frameLenSmpl = ACM_ISAC_GETNEWFRAMELEN(_codecInstPtr->inst);
ACM_ISAC_GETSENDBITRATE(_codecInstPtr->inst, &_isacCurrentBN);
}
UpdateFrameLen();
return *bitStreamLenByte;
}
WebRtc_Word16
ACMISAC::DecodeSafe(
WebRtc_UWord8* /* bitStream */,
WebRtc_Word16 /* bitStreamLenByte */,
WebRtc_Word16* /* audio */,
WebRtc_Word16* /* audioSamples */,
WebRtc_Word8* /* speechType */)
{
return 0;
}
WebRtc_Word16
ACMISAC::InternalInitEncoder(
WebRtcACMCodecParams* codecParams)
{
// if rate is set to -1 then iSAC has to be in adaptive mode
if(codecParams->codecInstant.rate == -1)
{
_isacCodingMode = ADAPTIVE;
}
// sanity check that rate is in acceptable range
else if((codecParams->codecInstant.rate >= ISAC_MIN_RATE) &&
(codecParams->codecInstant.rate <= ISAC_MAX_RATE))
{
_isacCodingMode = CHANNEL_INDEPENDENT;
_isacCurrentBN = codecParams->codecInstant.rate;
}
else
{
return -1;
}
// we need to set the encoder sampling frequency.
if(UpdateEncoderSampFreq((WebRtc_UWord16)codecParams->codecInstant.plfreq) < 0)
{
return -1;
}
if(ACM_ISAC_ENCODERINIT(_codecInstPtr->inst, _isacCodingMode) < 0)
{
return -1;
}
// apply the frame-size and rate if operating in
// channel-independent mode
if(_isacCodingMode == CHANNEL_INDEPENDENT)
{
if(ACM_ISAC_CONTROL(_codecInstPtr->inst,
codecParams->codecInstant.rate,
codecParams->codecInstant.pacsize /
(codecParams->codecInstant.plfreq / 1000)) < 0)
{
return -1;
}
}
else
{
// We need this for adaptive case and has to be called
// after initialization
ACM_ISAC_GETSENDBITRATE(
_codecInstPtr->inst, &_isacCurrentBN);
}
_frameLenSmpl = ACM_ISAC_GETNEWFRAMELEN(_codecInstPtr->inst);
return 0;
}
WebRtc_Word16
ACMISAC::InternalInitDecoder(
WebRtcACMCodecParams* codecParams)
{
if (_codecInstPtr == NULL)
{
return -1;
}
// set decoder sampling frequency.
if(codecParams->codecInstant.plfreq == 32000)
{
UpdateDecoderSampFreq(ACMCodecDB::kISACSWB);
}
else
{
UpdateDecoderSampFreq(ACMCodecDB::kISAC);
}
// in a one-way communication we may never register send-codec.
// However we like that the BWE to work properly so it has to
// be initialized. The BWE is initialized when iSAC encoder is initialized.
// Therefore, we need this.
if(!_encoderInitialized)
{
// Since we don't require a valid rate or a valid packet size when initializing
// the decoder, we set valid values before initializing encoder
codecParams->codecInstant.rate = kIsacWbDefaultRate;
codecParams->codecInstant.pacsize = kIsacPacSize960;
if(InternalInitEncoder(codecParams) < 0)
{
return -1;
}
_encoderInitialized = true;
}
return ACM_ISAC_DECODERINIT(_codecInstPtr->inst);
}
WebRtc_Word16
ACMISAC::InternalCreateDecoder()
{
if (_codecInstPtr == NULL)
{
return -1;
}
WebRtc_Word16 status = ACM_ISAC_CREATE (&(_codecInstPtr->inst));
// specific to codecs with one instance for encoding and decoding
_encoderInitialized = false;
if(status < 0)
{
_encoderExist = false;
}
else
{
_encoderExist = true;
}
return status;
}
void
ACMISAC::DestructDecoderSafe()
{
// codec with shared instance cannot delete.
_decoderInitialized = false;
return;
}
WebRtc_Word16
ACMISAC::InternalCreateEncoder()
{
if (_codecInstPtr == NULL)
{
return -1;
}
WebRtc_Word16 status = ACM_ISAC_CREATE(&(_codecInstPtr->inst));
// specific to codecs with one instance for encoding and decoding
_decoderInitialized = false;
if(status < 0)
{
_decoderExist = false;
}
else
{
_decoderExist = true;
}
return status;
}
void
ACMISAC::DestructEncoderSafe()
{
// codec with shared instance cannot delete.
_encoderInitialized = false;
return;
}
WebRtc_Word32
ACMISAC::CodecDef(
WebRtcNetEQ_CodecDef& codecDef,
const CodecInst& codecInst)
{
// Sanity checks
if (_codecInstPtr == NULL)
{
return -1;
}
if (!_decoderInitialized || !_decoderExist)
{
// Todo:
// log error
return -1;
}
// Fill up the structure by calling
// "SET_CODEC_PAR" & "SET_ISAC_FUNCTION."
// Then call NetEQ to add the codec to it's
// database.
if(codecInst.plfreq == 16000)
{
SET_CODEC_PAR((codecDef), kDecoderISAC, codecInst.pltype,
_codecInstPtr->inst, 16000);
#ifdef WEBRTC_CODEC_ISAC
SET_ISAC_FUNCTIONS((codecDef));
#else
SET_ISACfix_FUNCTIONS((codecDef));
#endif
}
else
{
#ifdef WEBRTC_CODEC_ISAC
SET_CODEC_PAR((codecDef), kDecoderISACswb, codecInst.pltype,
_codecInstPtr->inst, 32000);
SET_ISACSWB_FUNCTIONS((codecDef));
#else
return -1;
#endif
}
return 0;
}
void
ACMISAC::InternalDestructEncoderInst(
void* ptrInst)
{
if(ptrInst != NULL)
{
ACM_ISAC_FREE((ACM_ISAC_STRUCT *)ptrInst);
}
return;
}
WebRtc_Word16
ACMISAC::Transcode(
WebRtc_UWord8* bitStream,
WebRtc_Word16* bitStreamLenByte,
WebRtc_Word16 qBWE,
WebRtc_Word32 rate,
bool isRED)
{
WebRtc_Word16 jitterInfo = 0;
// transcode from a higher rate to lower rate
// sanity check
if (_codecInstPtr == NULL)
{
return -1;
}
*bitStreamLenByte = ACM_ISAC_GETNEWBITSTREAM(_codecInstPtr->inst,
qBWE, jitterInfo, rate, (WebRtc_Word16*)bitStream, (isRED)? 1:0);
if(*bitStreamLenByte < 0)
{
// error happened
*bitStreamLenByte = 0;
return -1;
}
else
{
return *bitStreamLenByte;
}
}
WebRtc_Word16
ACMISAC::SetBitRateSafe(
WebRtc_Word32 bitRate)
{
if (_codecInstPtr == NULL)
{
return -1;
}
WebRtc_UWord16 encoderSampFreq;
EncoderSampFreq(encoderSampFreq);
bool reinit = false;
// change the BN of iSAC
if(bitRate == -1)
{
// ADAPTIVE MODE
// Check if it was already in adaptive mode
if(_isacCodingMode != ADAPTIVE)
{
// was not in adaptive, then set the mode to adaptive
// and flag for re-initialization
_isacCodingMode = ADAPTIVE;
reinit = true;
}
}
// Sanity check if the rate valid
else if((bitRate >= ISAC_MIN_RATE) &&
(bitRate <= ISAC_MAX_RATE))
{
//check if it was in channel-independent mode before
if(_isacCodingMode != CHANNEL_INDEPENDENT)
{
// was not in channel independent, set the mode to
// channel-independent and flag for re-initialization
_isacCodingMode = CHANNEL_INDEPENDENT;
reinit = true;
}
// store the bottleneck
_isacCurrentBN = (WebRtc_UWord16)bitRate;
}
else
{
// invlaid rate
return -1;
}
WebRtc_Word16 status = 0;
if(reinit)
{
// initialize and check if it is successful
if(ACM_ISAC_ENCODERINIT(_codecInstPtr->inst, _isacCodingMode) < 0)
{
// failed initialization
return -1;
}
}
if(_isacCodingMode == CHANNEL_INDEPENDENT)
{
status = ACM_ISAC_CONTROL(_codecInstPtr->inst, _isacCurrentBN,
(encoderSampFreq == 32000)? 30:(_frameLenSmpl / 16));
if(status < 0)
{
status = -1;
}
}
// Update encoder parameters
_encoderParams.codecInstant.rate = bitRate;
UpdateFrameLen();
return status;
}
WebRtc_Word32
ACMISAC::GetEstimatedBandwidthSafe()
{
WebRtc_Word16 bandwidthIndex;
WebRtc_Word16 delayIndex;
IsacSamplingRate sampRate;
// Get bandwidth information
ACM_ISAC_GETSENDBWE(_codecInstPtr->inst, &bandwidthIndex, &delayIndex);
// Validy check of index
if ((bandwidthIndex < 0) || (bandwidthIndex >= NR_ISAC_BANDWIDTHS))
{
return -1;
}
// Check sample frequency
sampRate = ACM_ISAC_GETDECSAMPRATE(_codecInstPtr->inst);
if(sampRate == kIsacWideband)
{
return isacRatesWB[bandwidthIndex];
}
else
{
return isacRatesSWB[bandwidthIndex];
}
}
WebRtc_Word32
ACMISAC::SetEstimatedBandwidthSafe(
WebRtc_Word32 estimatedBandwidth)
{
IsacSamplingRate sampRate;
WebRtc_Word16 bandwidthIndex;
// Check sample frequency and choose appropriate table
sampRate = ACM_ISAC_GETENCSAMPRATE(_codecInstPtr->inst);
if(sampRate == kIsacWideband)
{
// Search through the WB rate table to find the index
bandwidthIndex = NR_ISAC_BANDWIDTHS/2 - 1;
for (int i=0; i<(NR_ISAC_BANDWIDTHS/2); i++)
{
if (estimatedBandwidth == isacRatesWB[i])
{
bandwidthIndex = i;
break;
} else if (estimatedBandwidth == isacRatesWB[i+NR_ISAC_BANDWIDTHS/2])
{
bandwidthIndex = i + NR_ISAC_BANDWIDTHS/2;
break;
} else if (estimatedBandwidth < isacRatesWB[i])
{
bandwidthIndex = i;
break;
}
}
}
else
{
// Search through the SWB rate table to find the index
bandwidthIndex = NR_ISAC_BANDWIDTHS - 1;
for (int i=0; i<NR_ISAC_BANDWIDTHS; i++)
{
if(estimatedBandwidth <= isacRatesSWB[i])
{
bandwidthIndex = i;
break;
}
}
}
// Set iSAC Bandwidth Estimate
ACM_ISAC_SETBWE(_codecInstPtr->inst, bandwidthIndex);
return 0;
}
WebRtc_Word32
ACMISAC::GetRedPayloadSafe(
#if (!defined(WEBRTC_CODEC_ISAC))
WebRtc_UWord8* /* redPayload */,
WebRtc_Word16* /* payloadBytes */)
{
return -1;
#else
WebRtc_UWord8* redPayload,
WebRtc_Word16* payloadBytes)
{
WebRtc_Word16 bytes = WebRtcIsac_GetRedPayload(_codecInstPtr->inst, (WebRtc_Word16*)redPayload);
if (bytes < 0)
{
return -1;
}
*payloadBytes = bytes;
return 0;
#endif
}
WebRtc_Word16
ACMISAC::UnregisterFromNetEqSafe(
ACMNetEQ* netEq,
WebRtc_Word16 payloadType)
{
if(payloadType == _decoderParams.codecInstant.pltype)
{
return netEq->RemoveCodec(kDecoderISAC);
}
else if(payloadType == _decoderParams32kHz.codecInstant.pltype)
{
return netEq->RemoveCodec(kDecoderISACswb);
}
else
{
WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceAudioCoding, _uniqueID,
"Cannot unregister codec %s given payload-type %d does not match \
the stored payload type %d or %d",
_decoderParams.codecInstant.plname,
payloadType,
_decoderParams.codecInstant.pltype,
_decoderParams32kHz.codecInstant.pltype);
return -1;
}
}
WebRtc_Word16
ACMISAC::UpdateDecoderSampFreq(
#ifdef WEBRTC_CODEC_ISAC
WebRtc_Word16 codecId)
{
if(ACMCodecDB::kISAC == codecId)
{
return WebRtcIsac_SetDecSampRate(_codecInstPtr->inst, kIsacWideband);
}
else if(ACMCodecDB::kISACSWB == codecId)
{
return WebRtcIsac_SetDecSampRate(_codecInstPtr->inst, kIsacSuperWideband);
}
else
{
return -1;
}
#else
WebRtc_Word16 /* codecId */)
{
return 0;
#endif
}
WebRtc_Word16
ACMISAC::UpdateEncoderSampFreq(
#ifdef WEBRTC_CODEC_ISAC
WebRtc_UWord16 encoderSampFreqHz)
{
WebRtc_UWord16 currentSampRateHz;
EncoderSampFreq(currentSampRateHz);
if(currentSampRateHz != encoderSampFreqHz)
{
if((encoderSampFreqHz != 16000) && (encoderSampFreqHz != 32000))
{
return -1;
}
else
{
_inAudioIxRead = 0;
_inAudioIxWrite = 0;
_inTimestampIxWrite = 0;
if(encoderSampFreqHz == 16000)
{
if(WebRtcIsac_SetEncSampRate(_codecInstPtr->inst, kIsacWideband) < 0)
{
return -1;
}
_samplesIn10MsAudio = 160;
}
else
{
if(WebRtcIsac_SetEncSampRate(_codecInstPtr->inst, kIsacSuperWideband) < 0)
{
return -1;
}
_samplesIn10MsAudio = 320;
}
_frameLenSmpl = ACM_ISAC_GETNEWFRAMELEN(_codecInstPtr->inst);
_encoderParams.codecInstant.pacsize = _frameLenSmpl;
_encoderParams.codecInstant.plfreq = encoderSampFreqHz;
return 0;
}
}
#else
WebRtc_UWord16 /* codecId */)
{
#endif
return 0;
}
WebRtc_Word16
ACMISAC::EncoderSampFreq(
WebRtc_UWord16& sampFreqHz)
{
IsacSamplingRate sampRate;
sampRate = ACM_ISAC_GETENCSAMPRATE(_codecInstPtr->inst);
if(sampRate == kIsacSuperWideband)
{
sampFreqHz = 32000;
}
else
{
sampFreqHz = 16000;
}
return 0;
}
WebRtc_Word32
ACMISAC::ConfigISACBandwidthEstimator(
const WebRtc_UWord8 initFrameSizeMsec,
const WebRtc_UWord16 initRateBitPerSec,
const bool enforceFrameSize)
{
WebRtc_Word16 status;
{
WebRtc_UWord16 sampFreqHz;
EncoderSampFreq(sampFreqHz);
// @TODO: at 32kHz we hardcode calling with 30ms and enforce
// the frame-size otherwise we might get error. Revise if
// control-bwe is changed.
if(sampFreqHz == 32000)
{
status = ACM_ISAC_CONTROL_BWE(_codecInstPtr->inst,
initRateBitPerSec, 30, 1);
}
else
{
status = ACM_ISAC_CONTROL_BWE(_codecInstPtr->inst,
initRateBitPerSec, initFrameSizeMsec, enforceFrameSize? 1:0);
}
}
if(status < 0)
{
WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceAudioCoding, _uniqueID,
"Coutn't config iSAC BWE.");
return -1;
}
UpdateFrameLen();
ACM_ISAC_GETSENDBITRATE(_codecInstPtr->inst, &_isacCurrentBN);
return 0;
}
WebRtc_Word32
ACMISAC::SetISACMaxPayloadSize(
const WebRtc_UWord16 maxPayloadLenBytes)
{
return ACM_ISAC_SETMAXPAYLOADSIZE(_codecInstPtr->inst, maxPayloadLenBytes);
}
WebRtc_Word32
ACMISAC::SetISACMaxRate(
const WebRtc_UWord32 maxRateBitPerSec)
{
return ACM_ISAC_SETMAXRATE(_codecInstPtr->inst, maxRateBitPerSec);
}
void
ACMISAC::UpdateFrameLen()
{
_frameLenSmpl = ACM_ISAC_GETNEWFRAMELEN(_codecInstPtr->inst);
_encoderParams.codecInstant.pacsize = _frameLenSmpl;
}
void
ACMISAC::CurrentRate(WebRtc_Word32& rateBitPerSec)
{
if(_isacCodingMode == ADAPTIVE)
{
ACM_ISAC_GETSENDBITRATE(_codecInstPtr->inst, &rateBitPerSec);
}
}
bool
ACMISAC::DecoderParamsSafe(
WebRtcACMCodecParams* decParams,
const WebRtc_UWord8 payloadType)
{
if(_decoderInitialized)
{
if(payloadType == _decoderParams.codecInstant.pltype)
{
memcpy(decParams, &_decoderParams, sizeof(WebRtcACMCodecParams));
return true;
}
if(payloadType == _decoderParams32kHz.codecInstant.pltype)
{
memcpy(decParams, &_decoderParams32kHz,
sizeof(WebRtcACMCodecParams));
return true;
}
}
return false;
}
void
ACMISAC::SaveDecoderParamSafe(
const WebRtcACMCodecParams* codecParams)
{
// set decoder sampling frequency.
if(codecParams->codecInstant.plfreq == 32000)
{
memcpy(&_decoderParams32kHz, codecParams, sizeof(WebRtcACMCodecParams));
}
else
{
memcpy(&_decoderParams, codecParams, sizeof(WebRtcACMCodecParams));
}
}
WebRtc_Word16
ACMISAC::REDPayloadISAC(
const WebRtc_Word32 isacRate,
const WebRtc_Word16 isacBwEstimate,
WebRtc_UWord8* payload,
WebRtc_Word16* payloadLenBytes)
{
WebRtc_Word16 status;
ReadLockScoped rl(_codecWrapperLock);
status = Transcode(payload, payloadLenBytes, isacBwEstimate, isacRate, true);
return status;
}
#endif
} // namespace webrtc