pinebuds/services/app_ai/voice_sbc/voice_sbc.cpp

197 lines
6.4 KiB
C++

#ifndef VOICE_DATAPATH
/***************************************************************************
*
* Copyright 2015-2019 BES.
* All rights reserved. All unpublished rights reserved.
*
* No part of this work may be used or reproduced in any form or by any
* means, or stored in a database or retrieval system, without prior written
* permission of BES.
*
* Use of this work is governed by a license granted by BES.
* This work contains confidential and proprietary information of
* BES. which is protected by copyright, trade secret,
* trademark and other intellectual property rights.
*
****************************************************************************/
#include "voice_sbc.h"
#include "hal_trace.h"
static VOICE_SBC_CONFIG_T voice_sbc_config =
{
VOICE_SBC_CHANNEL_COUNT ,
VOICE_SBC_CHANNEL_MODE ,
VOICE_SBC_BIT_POOL ,
VOICE_SBC_SIZE_PER_SAMPLE ,
VOICE_SBC_SAMPLE_RATE ,
VOICE_SBC_NUM_BLOCKS ,
VOICE_SBC_NUM_SUB_BANDS ,
VOICE_SBC_MSBC_FLAG ,
VOICE_SBC_ALLOC_METHOD ,
};
static int voice_sbc_init_encoder(void);
static btif_sbc_encoder_t voice_sbc_encoder;
static btif_sbc_decoder_t voice_sbc_decoder;
static uint32_t voice_sbc_frame_len = 0;
static float voice_sbc_eq_band_gain[8] = {1, 1, 1, 1, 1, 1, 1, 1};
static int voice_sbc_init_encoder(void)
{
btif_sbc_init_encoder(&voice_sbc_encoder);
voice_sbc_encoder.streamInfo.numChannels = voice_sbc_config.channelCnt;
voice_sbc_encoder.streamInfo.channelMode = voice_sbc_config.channelMode;
voice_sbc_encoder.streamInfo.bitPool = voice_sbc_config.bitPool;
voice_sbc_encoder.streamInfo.sampleFreq = voice_sbc_config.sampleRate;
voice_sbc_encoder.streamInfo.allocMethod = voice_sbc_config.allocMethod;
voice_sbc_encoder.streamInfo.numBlocks = voice_sbc_config.numBlocks;
voice_sbc_encoder.streamInfo.numSubBands = voice_sbc_config.numSubBands;
voice_sbc_encoder.streamInfo.mSbcFlag = voice_sbc_config.mSbcFlag;
voice_sbc_frame_len = btif_sbc_frame_len(&(voice_sbc_encoder.streamInfo));
TRACE(1,"frame len is %d", voice_sbc_frame_len);
return 0;
}
static int voice_sbc_init_decoder(void)
{
btif_sbc_init_decoder(&voice_sbc_decoder);
voice_sbc_decoder.streamInfo.numChannels = voice_sbc_config.channelCnt;
voice_sbc_decoder.streamInfo.channelMode = voice_sbc_config.channelMode;
voice_sbc_decoder.streamInfo.bitPool = voice_sbc_config.bitPool;
voice_sbc_decoder.streamInfo.sampleFreq = voice_sbc_config.sampleRate;
voice_sbc_decoder.streamInfo.allocMethod = voice_sbc_config.allocMethod;
voice_sbc_decoder.streamInfo.numBlocks = voice_sbc_config.numBlocks;
voice_sbc_decoder.streamInfo.numSubBands = voice_sbc_config.numSubBands;
voice_sbc_decoder.streamInfo.mSbcFlag = voice_sbc_config.mSbcFlag;
return 0;
}
uint32_t voice_sbc_get_frame_len(void)
{
return voice_sbc_frame_len;
}
uint32_t voice_sbc_encode(uint8_t *input, uint32_t inputBytes, uint32_t* purchasedBytes, uint8_t *output, uint8_t isReset)
{
if (isReset)
{
voice_sbc_init_encoder();
}
btif_sbc_pcm_data_t PcmEncData;
uint16_t outputSbcBytes, bytes_encoded;
PcmEncData.numChannels = voice_sbc_encoder.streamInfo.numChannels;
PcmEncData.sampleFreq = voice_sbc_encoder.streamInfo.sampleFreq;
if (voice_sbc_encoder.streamInfo.mSbcFlag)
{
uint16_t outputOffset = 0;
uint16_t oneChunkOutput = 0;
uint16_t oneChunkConsumed;
bytes_encoded = 0;
for (uint32_t index = 0;index < inputBytes;index += (uint32_t)VOICE_SBC_PCM_DATA_SIZE_PER_FRAME)
{
PcmEncData.data = input+index;
PcmEncData.dataLen = VOICE_SBC_PCM_DATA_SIZE_PER_FRAME;
//output[outputOffset++] = 0xAD;
//output[outputOffset++] = 0x02;
btif_sbc_encode_frames(&voice_sbc_encoder, &PcmEncData, &oneChunkConsumed,
&output[outputOffset], &oneChunkOutput, 0xFFFF) ;
outputOffset += oneChunkOutput;
//output[outputOffset++] = 0x00;
bytes_encoded += oneChunkConsumed;
}
outputSbcBytes = outputOffset;
}
else
{
PcmEncData.data = input;
PcmEncData.dataLen = inputBytes;
btif_sbc_encode_frames(&voice_sbc_encoder, &PcmEncData, &bytes_encoded,
output, &outputSbcBytes, 0xFFFF) ;
}
TRACE(2,"Encode %d bytes PCM data into %d bytes SBC data.", inputBytes, outputSbcBytes);
TRACE(1,"Consumed PCM data %d bytes", bytes_encoded);
*purchasedBytes = bytes_encoded;
return outputSbcBytes;
}
uint32_t voice_sbc_decode(uint8_t* pcm_buffer, CQueue* encodedDataQueue, uint32_t expectedOutputSize, uint8_t isReset)
{
if (isReset)
{
voice_sbc_init_decoder();
}
int r = 0;
unsigned char *e1 = NULL, *e2 = NULL;
unsigned int len1 = 0, len2 = 0;
uint32_t sbcDataBytesToDecode;
unsigned int pcm_offset = 0;
uint16_t byte_decode;
int8_t ret;
btif_sbc_pcm_data_t voice_PcmDecData;
get_again:
voice_PcmDecData.data = pcm_buffer+pcm_offset;
voice_PcmDecData.dataLen = 0;
if (LengthOfCQueue(encodedDataQueue) > VOICE_SBC_ENCODED_DATA_SIZE_PER_FRAME)
{
sbcDataBytesToDecode = VOICE_SBC_ENCODED_DATA_SIZE_PER_FRAME;
}
else
{
sbcDataBytesToDecode = LengthOfCQueue(encodedDataQueue);
}
len1 = len2 = 0;
r = PeekCQueue(encodedDataQueue, sbcDataBytesToDecode, &e1, &len1, &e2, &len2);
if ((r == CQ_ERR) || (0 == len1)) {
goto exit;
}
ret = btif_sbc_decode_frames(&voice_sbc_decoder, (unsigned char *)e1, len1, &byte_decode,
&voice_PcmDecData, expectedOutputSize-pcm_offset, voice_sbc_eq_band_gain);
TRACE(4,"len1 %d byte_decode %d expectedOutputSize %d pcm_offset %d",
len1, byte_decode, expectedOutputSize, pcm_offset);
TRACE(1,"voice_PcmDecData.dataLen %d", voice_PcmDecData.dataLen);
DeCQueue(encodedDataQueue, 0, byte_decode);
pcm_offset += voice_PcmDecData.dataLen;
if (expectedOutputSize == pcm_offset)
{
goto exit;
}
if ((ret == BT_STS_SDP_CONT_STATE) || (ret == BT_STS_SUCCESS)) {
goto get_again;
}
exit:
TRACE(1,"Get pcm data size %d", pcm_offset);
return pcm_offset;
}
int voice_sbc_init(VOICE_SBC_CONFIG_T* pConfig)
{
voice_sbc_config = *pConfig;
return 0;
}
#endif