pinebuds/apps/audioplayers/a2dp_decoder/a2dp_decoder_lhdc.cpp

1250 lines
42 KiB
C++
Raw Normal View History

2022-08-15 04:20:27 -05:00
/***************************************************************************
*
* 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.
*
****************************************************************************/
// Standard C Included Files
#include "cmsis.h"
#include "plat_types.h"
#include <string.h>
#include "heap_api.h"
#include "hal_location.h"
#include "hal_timer.h"
#include "a2dp_decoder_internal.h"
typedef struct
{
uint16_t sequenceNumber;
uint32_t timestamp;
uint16_t curSubSequenceNumber;
uint16_t totalSubSequenceNumber;
uint8_t *buffer;
uint32_t buffer_len;
} a2dp_audio_lhdc_decoder_frame_t;
#define LHDC_MTU_LIMITER (100)
#if defined(A2DP_LHDC_V3)
#define A2DP_LHDC_OUTPUT_FRAME_SAMPLES (256)
#else
#define A2DP_LHDC_OUTPUT_FRAME_SAMPLES (512)
#endif
#define A2DP_LHDC_DEFAULT_LATENCY (1)
#if defined(A2DP_LHDC_V3)
#define PACKET_BUFFER_LENGTH (2 * 1024)
#else
#define PACKET_BUFFER_LENGTH (4 * 1024)
#endif
#define LHDC_READBUF_SIZE (512)
#define A2DP_LHDC_HDR_F_MSK 0x80
#define A2DP_LHDC_HDR_S_MSK 0x40
#define A2DP_LHDC_HDR_L_MSK 0x20
#define A2DP_LHDC_HDR_FLAG_MSK (A2DP_LHDC_HDR_F_MSK | A2DP_LHDC_HDR_S_MSK | A2DP_LHDC_HDR_L_MSK)
#define A2DP_LHDC_HDR_LATENCY_LOW 0x00
#define A2DP_LHDC_HDR_LATENCY_MID 0x01
#define A2DP_LHDC_HDR_LATENCY_HIGH 0x02
#define A2DP_LHDC_HDR_LATENCY_MASK (A2DP_LHDC_HDR_LATENCY_MID | A2DP_LHDC_HDR_LATENCY_HIGH)
#if defined(A2DP_LHDC_V3)
#define A2DP_LHDC_HDR_FRAME_NO_MASK 0xfc
#else
#define A2DP_LHDC_HDR_FRAME_NO_MASK 0x1c
#endif
typedef enum
{
ASM_PKT_WAT_STR,
ASM_PKT_WAT_LST,
} ASM_PKT_STATUS;
typedef enum
{
VERSION_2 = 200,
VERSION_3 = 300
} lhdc_ver_t;
/**
* get lhdc frame header
*/
/**
LHDC frame
*/
typedef struct _lhdc_frame_Info
{
uint32_t frame_len; // 该 frame 的长处,若是分离压缩,则表示单一声道的 frame 长度。
uint32_t isSplit; // 是否为分离方式压缩
uint32_t isLeft; // 左声道 == true 右声道 == false
} lhdc_frame_Info_t;
#ifdef A2DP_CP_ACCEL
struct A2DP_CP_LHDC_IN_FRM_INFO_T
{
uint16_t sequenceNumber;
uint32_t timestamp;
uint16_t curSubSequenceNumber;
uint16_t totalSubSequenceNumber;
};
struct A2DP_CP_LHDC_OUT_FRM_INFO_T
{
struct A2DP_CP_LHDC_IN_FRM_INFO_T in_info;
uint16_t frame_samples;
uint16_t decoded_frames;
uint16_t frame_idx;
uint16_t pcm_len;
};
#endif
extern "C"
{
typedef struct bes_bt_local_info_t
{
uint8_t bt_addr[BTIF_BD_ADDR_SIZE];
const char *bt_name;
uint8_t bt_len;
uint8_t ble_addr[BTIF_BD_ADDR_SIZE];
const char *ble_name;
uint8_t ble_len;
} bes_bt_local_info;
typedef int (*LHDC_GET_BT_INFO)(bes_bt_local_info *bt_info);
void lhdcInit(uint32_t bitPerSample, uint32_t sampleRate, uint32_t scaleTo16Bits, lhdc_ver_t version);
//void lhdcInit(uint32_t bitPerSample, uint32_t sampleRate, uint32_t scaleTo16Bits);
#if defined(A2DP_LHDC_V3)
uint32_t lhdcPutData(uint8_t *pIn, uint32_t len);
#else
uint32_t lhdcPutData(uint8_t *pIn, int32_t len);
#endif
uint32_t lhdcDecodeProcess(uint8_t *pOutBuf);
void lhdcDestroy();
bool lhdcSetLicenseKeyTable(uint8_t *licTable, LHDC_GET_BT_INFO pFunc);
const char *getVersionCode();
bool larcIsEnabled();
}
static A2DP_AUDIO_CONTEXT_T *a2dp_audio_context_p = NULL;
static A2DP_AUDIO_DECODER_LASTFRAME_INFO_T a2dp_audio_lhdc_lastframe_info;
static A2DP_AUDIO_OUTPUT_CONFIG_T a2dp_audio_lhdc_output_config;
static uint16_t lhdc_mtu_limiter = LHDC_MTU_LIMITER;
static uint8_t serial_no;
static bool is_synced;
static ASM_PKT_STATUS asm_pkt_st;
static uint8_t packet_buffer[PACKET_BUFFER_LENGTH];
static uint32_t packet_buf_len = 0;
static uint32_t lhdc_total_frame_nb = 0;
#if defined(A2DP_LHDC_LARC)
static uint32_t lhdc_drop_frame = 0;
static uint32_t lhdc_last_time = 0;
static struct A2DP_CP_LHDC_IN_FRM_INFO_T last_frame_info;
//static uint16_t demo_frame_cnt = 0;
#endif
extern int bes_bt_local_info_get(bes_bt_local_info *local_info);
//Local API declare
void sav_lhdc_log_bytes_len(uint32_t bytes_len);
#define STATISTICS_UPDATE_INTERVAL 1000 // in ms
#define INTERVAL_TIMEBASE 1000 //in ms
typedef struct
{
uint32_t sum_bytes;
uint32_t last_times; //in ms
float last_avg_val;
int32_t update_interval; //in ms
} LHDC_TRACFIC_STATISTICS;
LHDC_TRACFIC_STATISTICS statistic = {0, 0, 0, STATISTICS_UPDATE_INTERVAL};
void sav_lhdc_log_bytes_len(uint32_t bytes_len)
{
uint32_t time_current = GET_CURRENT_MS();
float time_diff = time_current - statistic.last_times;
statistic.sum_bytes += bytes_len;
if (time_diff >= statistic.update_interval)
{
statistic.last_avg_val = ((float)(statistic.sum_bytes * 8) / 1000) / (time_diff / INTERVAL_TIMEBASE);
TRACE_A2DP_DECODER_I("Avarage rate about %d kbps", (int)statistic.last_avg_val);
statistic.sum_bytes = 0;
statistic.last_times = time_current;
}
}
#if defined(A2DP_LHDC_LARC) && defined(A2DP_LHDC_V3)
static void sav_lhdc_save_last_info(struct A2DP_CP_LHDC_IN_FRM_INFO_T *f_info)
{
memcpy(&last_frame_info, f_info, sizeof(struct A2DP_CP_LHDC_IN_FRM_INFO_T));
}
static void sav_lhdc_get_next_info(struct A2DP_CP_LHDC_IN_FRM_INFO_T *f_info)
{
f_info->timestamp = last_frame_info.timestamp;
f_info->sequenceNumber = last_frame_info.sequenceNumber;
f_info->totalSubSequenceNumber = last_frame_info.totalSubSequenceNumber;
f_info->curSubSequenceNumber = last_frame_info.curSubSequenceNumber + 1;
if (f_info->curSubSequenceNumber > last_frame_info.totalSubSequenceNumber)
{
f_info->curSubSequenceNumber = 1;
f_info->timestamp = last_frame_info.timestamp + (A2DP_LHDC_OUTPUT_FRAME_SAMPLES * last_frame_info.totalSubSequenceNumber);
f_info->sequenceNumber = last_frame_info.sequenceNumber + 1;
}
}
#endif
static void *a2dp_audio_lhdc_frame_malloc(uint32_t packet_len)
{
a2dp_audio_lhdc_decoder_frame_t *decoder_frame_p = NULL;
uint8_t *buffer = NULL;
buffer = (uint8_t *)a2dp_audio_heap_malloc(packet_len);
decoder_frame_p = (a2dp_audio_lhdc_decoder_frame_t *)a2dp_audio_heap_malloc(sizeof(a2dp_audio_lhdc_decoder_frame_t));
decoder_frame_p->buffer = buffer;
decoder_frame_p->buffer_len = packet_len;
return (void *)decoder_frame_p;
}
static void reset_lhdc_assmeble_packet(void)
{
is_synced = false;
asm_pkt_st = ASM_PKT_WAT_STR;
packet_buf_len = 0;
}
static void initial_lhdc_assemble_packet(bool splitFlg)
{
memset(packet_buffer, 0, PACKET_BUFFER_LENGTH);
reset_lhdc_assmeble_packet();
serial_no = 0xff;
}
/**
* A2DP packet
*/
#if defined(A2DP_LHDC_V3)
int assemble_lhdc_packet(uint8_t *input, uint32_t input_len, uint8_t **pLout, uint32_t *pLlen)
{
uint8_t hdr = 0, seqno = 0xff;
int ret = -1;
uint32_t status = 0;
hdr = (*input);
input++;
seqno = (*input);
input++;
input_len -= 2;
//Check latency and update value when changed.
status = hdr & A2DP_LHDC_HDR_LATENCY_MASK;
//Get number of frame in packet.
status = (hdr & A2DP_LHDC_HDR_FRAME_NO_MASK) >> 2;
if (status <= 0)
{
TRACE_A2DP_DECODER_I("No any frame in packet.");
return 0;
}
lhdc_total_frame_nb = status;
if (seqno != serial_no)
{
TRACE_A2DP_DECODER_I("Packet lost! now(%d), expect(%d)", seqno, serial_no);
}
serial_no = seqno + 1;
sav_lhdc_log_bytes_len(input_len);
if (pLlen && pLout)
{
*pLlen = input_len;
*pLout = input;
}
ret = lhdc_total_frame_nb;
return ret;
}
#else
int assemble_lhdc_packet(uint8_t *input, uint32_t input_len, uint8_t **pLout, uint32_t *pLlen)
{
uint8_t hdr = 0, seqno = 0xff;
int ret = -1;
//uint32_t status = 0;
hdr = (*input);
input++;
seqno = (*input);
input++;
input_len -= 2;
if (is_synced)
{
if (seqno != serial_no)
{
reset_lhdc_assmeble_packet();
if ((hdr & A2DP_LHDC_HDR_FLAG_MSK) == 0 ||
(hdr & A2DP_LHDC_HDR_S_MSK) != 0)
{
goto lhdc_start;
}
else
TRACE_A2DP_DECODER_I("drop packet No. %u", seqno);
return 0;
}
serial_no = seqno + 1;
}
lhdc_start:
switch (asm_pkt_st)
{
case ASM_PKT_WAT_STR:
{
if ((hdr & A2DP_LHDC_HDR_FLAG_MSK) == 0)
{
memcpy(&packet_buffer[0], input, input_len);
if (pLlen && pLout)
{
*pLlen = input_len;
*pLout = packet_buffer;
}
//TRACE_A2DP_DECODER_I("Single payload size = %d", *pLlen);
asm_pkt_st = ASM_PKT_WAT_STR;
packet_buf_len = 0; //= packet_buf_left_len = packet_buf_right_len = 0;
lhdc_total_frame_nb = (hdr & A2DP_LHDC_HDR_FRAME_NO_MASK) >> 2;
;
//TRACE_A2DP_DECODER_I("Single packet. total %d frames", lhdc_total_frame_nb);
ret = lhdc_total_frame_nb;
}
else if (hdr & A2DP_LHDC_HDR_S_MSK)
{
ret = 0;
if (packet_buf_len + input_len >= PACKET_BUFFER_LENGTH)
{
packet_buf_len = 0;
asm_pkt_st = ASM_PKT_WAT_STR;
TRACE_A2DP_DECODER_I("ASM_PKT_WAT_STR:Frame buffer overflow!(%d)", packet_buf_len);
break;
}
memcpy(&packet_buffer, input, input_len);
packet_buf_len = input_len;
asm_pkt_st = ASM_PKT_WAT_LST;
lhdc_total_frame_nb = (hdr & A2DP_LHDC_HDR_FRAME_NO_MASK) >> 2;
//TRACE_A2DP_DECODER_I("start of multi packet.");
}
else
ret = -1;
if (ret >= 0)
{
if (!is_synced)
{
is_synced = true;
serial_no = seqno + 1;
}
}
break;
}
case ASM_PKT_WAT_LST:
{
if (packet_buf_len + input_len >= PACKET_BUFFER_LENGTH)
{
packet_buf_len = 0;
asm_pkt_st = ASM_PKT_WAT_STR;
TRACE_A2DP_DECODER_I("ASM_PKT_WAT_LST:Frame buffer overflow(%d)", packet_buf_len);
break;
}
memcpy(&packet_buffer[packet_buf_len], input, input_len);
//TRACE_A2DP_DECODER_I("multi:payload size = %d", input_len);
packet_buf_len += input_len;
ret = 0;
if (hdr & A2DP_LHDC_HDR_L_MSK)
{
if (pLlen && pLout)
{
*pLlen = packet_buf_len;
*pLout = packet_buffer;
}
//TRACE_A2DP_DECODER_I("end of multi packet. total %d frames.", lhdc_total_frame_nb);
packet_buf_len = 0; //packet_buf_left_len = packet_buf_right_len = 0;
ret = lhdc_total_frame_nb;
asm_pkt_st = ASM_PKT_WAT_STR;
}
break;
}
default:
ret = 0;
break;
}
return ret;
}
#endif
static int parse_lhdc_info(uint8_t *in, lhdc_frame_Info_t *h)
{
#define LHDC_HDR_LEN 4
uint32_t hdr = 0;
int ret = -1;
memcpy(&hdr, in, LHDC_HDR_LEN);
h->frame_len = (int)((hdr >> 8) & 0x1fff);
h->isSplit = ((hdr & 0x00600000) == 0x00600000);
h->isLeft = ((hdr & 0xf) == 0);
if ((hdr & 0xff000000) != 0x4c000000)
{
}
else
{
ret = 0;
}
return ret;
}
#ifdef A2DP_CP_ACCEL
extern "C" uint32_t get_in_cp_frame_cnt(void);
extern "C" unsigned int set_cp_reset_flag(uint8_t evt);
extern uint32_t app_bt_stream_get_dma_buffer_samples(void);
int a2dp_cp_lhdc_cp_decode(void);
TEXT_LHDC_LOC
static int a2dp_cp_lhdc_after_cache_underflow(void)
{
int ret = 0;
#ifdef A2DP_CP_ACCEL
a2dp_cp_deinit();
lhdcDestroy();
ret = a2dp_cp_init(a2dp_cp_lhdc_cp_decode, CP_PROC_DELAY_2_FRAMES);
ASSERT(ret == 0, "%s: a2dp_cp_init() failed: ret=%d", __func__, ret);
#endif
return ret;
}
//uint32_t demoTimer = 0;
static int a2dp_cp_lhdc_mcu_decode(uint8_t *buffer, uint32_t buffer_bytes)
{
a2dp_audio_lhdc_decoder_frame_t *lhdc_decoder_frame_p = NULL;
list_node_t *node = NULL;
list_t *list = a2dp_audio_context_p->audio_datapath.input_raw_packet_list;
int ret, dec_ret;
struct A2DP_CP_LHDC_IN_FRM_INFO_T in_info;
struct A2DP_CP_LHDC_OUT_FRM_INFO_T *p_out_info;
uint8_t *out;
uint32_t out_len;
uint32_t out_frame_len;
uint32_t cp_buffer_frames_max = 0;
#if defined(A2DP_LHDC_LARC)
//bool forceRecovery = false;
bool larcEnabled = larcIsEnabled();
uint32_t frame_number = 0;
if (larcEnabled)
{
/* The LARC is enabled. */
uint32_t time_diff = 0, time_now = GET_CURRENT_MS();
if (lhdc_last_time == 0)
{
lhdc_last_time = time_now;
}
// if (demoTimer == 0) {
// demoTimer = time_now;
// }
time_diff = time_now - lhdc_last_time;
lhdc_last_time = time_now;
frame_number = (uint32_t)((float)time_diff / ((A2DP_LHDC_OUTPUT_FRAME_SAMPLES * 1000) / a2dp_audio_lhdc_output_config.sample_rate));
TRACE_A2DP_DECODER_I("%s:current total list len %d, need frames %d", __func__, a2dp_audio_list_length(list) + get_in_cp_frame_cnt(), frame_number);
// if (time_now - demoTimer >= (5 * 1000)) {
// forceRecovery = true;
// demoTimer = time_now;
// demo_frame_cnt = (a2dp_audio_lhdc_output_config.sample_rate * 120) / (A2DP_LHDC_OUTPUT_FRAME_SAMPLES * 1000);
// }
}
#endif
cp_buffer_frames_max = app_bt_stream_get_dma_buffer_samples() / 2;
if (cp_buffer_frames_max % (a2dp_audio_lhdc_lastframe_info.frame_samples))
{
cp_buffer_frames_max = cp_buffer_frames_max / (a2dp_audio_lhdc_lastframe_info.frame_samples) + 1;
}
else
{
cp_buffer_frames_max = cp_buffer_frames_max / (a2dp_audio_lhdc_lastframe_info.frame_samples);
}
out_frame_len = sizeof(*p_out_info) + buffer_bytes;
ret = a2dp_cp_decoder_init(out_frame_len, cp_buffer_frames_max * 8);
if (ret)
{
TRACE_A2DP_DECODER_I("%s: a2dp_cp_decoder_init() failed: ret=%d", __func__, ret);
set_cp_reset_flag(true);
return A2DP_DECODER_DECODE_ERROR;
}
#if defined(A2DP_LHDC_LARC)
uint32_t put_cnt = 0;
uint32_t drop_cnt = 0;
while (larcEnabled &&
((a2dp_audio_list_length(list) < frame_number && get_in_cp_frame_cnt() == 0)))
{
sav_lhdc_get_next_info(&in_info);
ret = a2dp_cp_put_in_frame(&in_info, sizeof(in_info), NULL, 0x55);
if (ret)
{
//TRACE_A2DP_DECODER_I("%s piff !!!!!!ret: %d ",__func__, ret);
break;
}
sav_lhdc_save_last_info(&in_info);
lhdc_drop_frame++;
frame_number--;
//demo_frame_cnt--;
put_cnt++;
}
//forceRecovery = false;
TRACE_A2DP_DECODER_I("Recover %d frames", put_cnt);
put_cnt = 0;
drop_cnt = 0;
#endif
while ((node = a2dp_audio_list_begin(list)) != NULL)
{
lhdc_decoder_frame_p = (a2dp_audio_lhdc_decoder_frame_t *)a2dp_audio_list_node(node);
in_info.sequenceNumber = lhdc_decoder_frame_p->sequenceNumber;
in_info.timestamp = lhdc_decoder_frame_p->timestamp;
in_info.curSubSequenceNumber = lhdc_decoder_frame_p->curSubSequenceNumber;
in_info.totalSubSequenceNumber = lhdc_decoder_frame_p->totalSubSequenceNumber;
#if defined(A2DP_LHDC_LARC)
if (!lhdc_drop_frame)
{
ret = a2dp_cp_put_in_frame(&in_info, sizeof(in_info), lhdc_decoder_frame_p->buffer, lhdc_decoder_frame_p->buffer_len);
if (ret)
{
//TRACE_A2DP_DECODER_I("%s piff !!!!!!ret: %d ",__func__, ret);
break;
}
sav_lhdc_save_last_info(&in_info);
put_cnt++;
}
else
{
lhdc_drop_frame--;
drop_cnt++;
}
#else
ret = a2dp_cp_put_in_frame(&in_info, sizeof(in_info), lhdc_decoder_frame_p->buffer, lhdc_decoder_frame_p->buffer_len);
if (ret)
{
//TRACE_A2DP_DECODER_I("%s piff !!!!!!ret: %d ",__func__, ret);
break;
}
#endif
a2dp_audio_list_remove(list, lhdc_decoder_frame_p);
node = a2dp_audio_list_begin(list);
}
#if defined(A2DP_LHDC_LARC)
TRACE_A2DP_DECODER_I("Put %d frames, drop %d frames", put_cnt, drop_cnt);
#endif
ret = a2dp_cp_get_full_out_frame((void **)&out, &out_len);
if (ret)
{
TRACE_A2DP_DECODER_I("%s %d cp find cache underflow", __func__, ret);
TRACE_A2DP_DECODER_I("aud_list_len:%d. cp_get_in_frame:%d", a2dp_audio_list_length(list), get_in_cp_frame_cnt());
a2dp_cp_lhdc_after_cache_underflow();
return A2DP_DECODER_CACHE_UNDERFLOW_ERROR;
}
if (out_len == 0)
{
memset(buffer, 0, buffer_bytes);
a2dp_cp_consume_full_out_frame();
TRACE_A2DP_DECODER_I("%s olz!!!%d ", __func__, __LINE__);
return A2DP_DECODER_NO_ERROR;
}
if (out_len != out_frame_len)
{
TRACE_A2DP_DECODER_I("%s: Bad out len %u (should be %u)", __func__, out_len, out_frame_len);
set_cp_reset_flag(true);
return A2DP_DECODER_DECODE_ERROR;
}
p_out_info = (struct A2DP_CP_LHDC_OUT_FRM_INFO_T *)out;
if (p_out_info->pcm_len)
{
a2dp_audio_lhdc_lastframe_info.sequenceNumber = p_out_info->in_info.sequenceNumber;
a2dp_audio_lhdc_lastframe_info.timestamp = p_out_info->in_info.timestamp;
a2dp_audio_lhdc_lastframe_info.curSubSequenceNumber = p_out_info->in_info.curSubSequenceNumber;
a2dp_audio_lhdc_lastframe_info.totalSubSequenceNumber = p_out_info->in_info.totalSubSequenceNumber;
a2dp_audio_lhdc_lastframe_info.frame_samples = p_out_info->frame_samples;
a2dp_audio_lhdc_lastframe_info.decoded_frames += p_out_info->decoded_frames;
a2dp_audio_lhdc_lastframe_info.undecode_frames =
a2dp_audio_list_length(list) + a2dp_cp_get_in_frame_cnt_by_index(p_out_info->frame_idx) - 1;
a2dp_audio_decoder_internal_lastframe_info_set(&a2dp_audio_lhdc_lastframe_info);
TRACE_A2DP_DECODER_I("lhdc_decoder seq:%d cursub:%d ttlsub:%d decoded:%d/%d",
a2dp_audio_lhdc_lastframe_info.sequenceNumber,
a2dp_audio_lhdc_lastframe_info.curSubSequenceNumber,
a2dp_audio_lhdc_lastframe_info.totalSubSequenceNumber,
a2dp_audio_lhdc_lastframe_info.decoded_frames,
a2dp_audio_lhdc_lastframe_info.undecode_frames);
}
if (p_out_info->pcm_len == buffer_bytes)
{
memcpy(buffer, p_out_info + 1, p_out_info->pcm_len);
dec_ret = A2DP_DECODER_NO_ERROR;
}
else
{
TRACE_A2DP_DECODER_I("%s %d cp decoder error !!!!!!", __func__, __LINE__);
set_cp_reset_flag(true);
return A2DP_DECODER_DECODE_ERROR;
}
ret = a2dp_cp_consume_full_out_frame();
if (ret)
{
TRACE_A2DP_DECODER_I("%s: a2dp_cp_consume_full_out_frame() failed: ret=%d", __func__, ret);
set_cp_reset_flag(true);
return A2DP_DECODER_DECODE_ERROR;
}
return dec_ret;
}
#ifdef __CP_EXCEPTION_TEST__
static bool _cp_assert = false;
int cp_assert(void)
{
_cp_assert = true;
return 0;
}
#endif
#define LHDC_DECODED_FRAME_SIZE 1024
TEXT_LHDC_LOC
int a2dp_cp_lhdc_cp_decode(void)
{
int ret;
enum CP_EMPTY_OUT_FRM_T out_frm_st;
uint8_t *out;
uint32_t out_len;
uint8_t *dec_start;
uint32_t dec_len;
struct A2DP_CP_LHDC_IN_FRM_INFO_T *p_in_info;
struct A2DP_CP_LHDC_OUT_FRM_INFO_T *p_out_info;
uint8_t *in_buf;
uint32_t in_len;
int32_t dec_sum;
uint32_t lhdc_out_len = 0;
#ifdef __CP_EXCEPTION_TEST__
if (_cp_assert)
{
_cp_assert = false;
*(int *)0 = 1;
//ASSERT(0, "ASSERT %s %d", __func__, __LINE__);
}
#endif
out_frm_st = a2dp_cp_get_emtpy_out_frame((void **)&out, &out_len);
if (out_frm_st != CP_EMPTY_OUT_FRM_OK && out_frm_st != CP_EMPTY_OUT_FRM_WORKING)
{
return out_frm_st;
}
ASSERT(out_len > sizeof(*p_out_info), "%s: Bad out_len %u (should > %u)", __func__, out_len, sizeof(*p_out_info));
p_out_info = (struct A2DP_CP_LHDC_OUT_FRM_INFO_T *)out;
if (out_frm_st == CP_EMPTY_OUT_FRM_OK)
{
p_out_info->pcm_len = 0;
p_out_info->decoded_frames = 0;
}
ASSERT(out_len > sizeof(*p_out_info) + p_out_info->pcm_len, "%s: Bad out_len %u (should > %u + %u)", __func__, out_len, sizeof(*p_out_info), p_out_info->pcm_len);
dec_start = (uint8_t *)(p_out_info + 1) + p_out_info->pcm_len;
dec_len = out_len - (dec_start - (uint8_t *)out);
dec_sum = 0;
while (dec_sum < (int32_t)dec_len)
{
uint32_t lhdc_decode_temp = 0;
ret = a2dp_cp_get_in_frame((void **)&in_buf, &in_len);
if (ret)
{
TRACE_A2DP_DECODER_I("cp_get_int_frame fail, ret=%d", ret);
return 4;
}
ASSERT(in_len > sizeof(*p_in_info), "%s: Bad in_len %u (should > %u)", __func__, in_len, sizeof(*p_in_info));
p_in_info = (struct A2DP_CP_LHDC_IN_FRM_INFO_T *)in_buf;
in_buf += sizeof(*p_in_info);
in_len -= sizeof(*p_in_info);
#if defined(A2DP_LHDC_V3)
#if defined(A2DP_LHDC_LARC)
if (in_len == 0x55 && larcIsEnabled())
{
//TRACE_A2DP_DECODER_I("recover:sn(%d), tt(%d), csssn(%d), ttsssn(%d)", p_in_info->sequenceNumber, p_in_info->timestamp, p_in_info->curSubSequenceNumber, p_in_info->totalSubSequenceNumber);
in_len = 0;
}
#endif
lhdcPutData(in_buf, in_len);
//TRACE_A2DP_DECODER_I("%s:put length=%d, dec_len=%d", __func__, in_len, dec_len);
#else
lhdcPutData(in_buf, in_len);
#endif
uint32_t loop_cnt = 0;
do
{
//TRACE_A2DP_DECODER_I("loop %d , dec_start %p, dec_sum %d, lhdc_decode_temp %d", loop_cnt, dec_start, dec_sum, lhdc_decode_temp);
lhdc_out_len = lhdcDecodeProcess(dec_start + dec_sum + lhdc_decode_temp);
if (lhdc_out_len > 0)
{
lhdc_decode_temp += lhdc_out_len;
loop_cnt++;
}
else
{
//TRACE_A2DP_DECODER_I("decodeProcess error!!! ret=%d", lhdc_out_len);
break;
}
} while (lhdc_decode_temp < dec_len && lhdc_decode_temp > 0);
//TRACE_A2DP_DECODER_I("lhdc_cp_decode seq:%d len:%d err:%d", p_in_info->sequenceNumber,
// dec_len - dec_sum,
// lhdc_out_len);
//TRACE_A2DP_DECODER_I("%s:decode loop run times=%d, lhdc_decode_temp=%d", __func__, loop_cnt, lhdc_decode_temp);
dec_sum += lhdc_decode_temp;
if ((lhdc_decode_temp % LHDC_DECODED_FRAME_SIZE))
{
TRACE_A2DP_DECODER_I("error!!! dec_sum: %d decode_temp: %d", dec_sum, lhdc_decode_temp);
return -1;
}
ret = a2dp_cp_consume_in_frame();
if (ret != 0)
{
TRACE_A2DP_DECODER_I("%s: a2dp_cp_consume_in_frame() failed: ret=%d", __func__, ret);
}
ASSERT(ret == 0, "%s: a2dp_cp_consume_in_frame() failed: ret=%d", __func__, ret);
memcpy(&p_out_info->in_info, p_in_info, sizeof(*p_in_info));
p_out_info->decoded_frames++;
p_out_info->frame_samples = A2DP_LHDC_OUTPUT_FRAME_SAMPLES;
p_out_info->frame_idx = a2dp_cp_get_in_frame_index();
}
if ((dec_sum % LHDC_DECODED_FRAME_SIZE))
{
TRACE_A2DP_DECODER_I("error!!! dec_sum:%d != dec_len:%d", dec_sum, dec_len);
ASSERT(0, "%s", __func__);
}
p_out_info->pcm_len += dec_sum;
if (out_len <= sizeof(*p_out_info) + p_out_info->pcm_len)
{
ret = a2dp_cp_consume_emtpy_out_frame();
ASSERT(ret == 0, "%s: a2dp_cp_consume_emtpy_out_frame() failed: ret=%d", __func__, ret);
}
return 0;
}
#endif
#if 1
static int a2dp_audio_lhdc_list_checker(void)
{
list_t *list = a2dp_audio_context_p->audio_datapath.input_raw_packet_list;
list_node_t *node = NULL;
a2dp_audio_lhdc_decoder_frame_t *lhdc_decoder_frame_p = NULL;
int cnt = 0;
do
{
lhdc_decoder_frame_p = (a2dp_audio_lhdc_decoder_frame_t *)a2dp_audio_lhdc_frame_malloc(LHDC_READBUF_SIZE);
if (lhdc_decoder_frame_p)
{
a2dp_audio_list_append(list, lhdc_decoder_frame_p);
}
cnt++;
} while (lhdc_decoder_frame_p && cnt < LHDC_MTU_LIMITER);
do
{
node = a2dp_audio_list_begin(list);
if (node)
{
lhdc_decoder_frame_p = (a2dp_audio_lhdc_decoder_frame_t *)a2dp_audio_list_node(node);
a2dp_audio_list_remove(list, lhdc_decoder_frame_p);
}
} while (node);
TRACE_A2DP_DECODER_I("%s cnt:%d list:%d", __func__, cnt, a2dp_audio_list_length(list));
return 0;
}
#endif
int a2dp_audio_lhdc_mcu_decode_frame(uint8_t *buffer, uint32_t buffer_bytes)
{
list_t *list = a2dp_audio_context_p->audio_datapath.input_raw_packet_list;
list_node_t *node = NULL;
a2dp_audio_lhdc_decoder_frame_t *lhdc_decoder_frame_p = NULL;
bool cache_underflow = false;
int output_byte = 0;
uint32_t lhdc_out_len = 0;
node = a2dp_audio_list_begin(list);
if (!node)
{
TRACE_A2DP_DECODER_I("lhdc_decode cache underflow");
cache_underflow = true;
goto exit;
}
else
{
lhdc_decoder_frame_p = (a2dp_audio_lhdc_decoder_frame_t *)a2dp_audio_list_node(node);
#if defined(A2DP_LHDC_V3)
lhdcPutData(lhdc_decoder_frame_p->buffer, lhdc_decoder_frame_p->buffer_len);
#else
lhdcPutData(lhdc_decoder_frame_p->buffer, lhdc_decoder_frame_p->buffer_len);
#endif
do
{
lhdc_out_len = lhdcDecodeProcess(buffer + output_byte);
if (lhdc_out_len > 0)
{
output_byte += lhdc_out_len;
}
else
{
break;
}
} while (output_byte < (int)buffer_bytes && output_byte > 0);
if (output_byte != (int)buffer_bytes)
{
TRACE_A2DP_DECODER_I("[warning] lhdc_decode_frame output_byte:%d lhdc_out_len:%d buffer_bytes:%d", output_byte, lhdc_out_len, buffer_bytes);
TRACE_A2DP_DECODER_I("[warning] lhdc_decode_frame frame_len:%d rtp seq:%d timestamp:%d decoder_frame:%d/%d ",
lhdc_decoder_frame_p->buffer_len,
lhdc_decoder_frame_p->sequenceNumber,
lhdc_decoder_frame_p->timestamp,
lhdc_decoder_frame_p->curSubSequenceNumber,
lhdc_decoder_frame_p->totalSubSequenceNumber);
output_byte = buffer_bytes;
int32_t dump_byte = lhdc_decoder_frame_p->buffer_len;
int32_t dump_offset = 0;
while (1)
{
uint32_t dump_byte_output = 0;
dump_byte_output = dump_byte > 32 ? 32 : dump_byte;
DUMP8("%02x ", lhdc_decoder_frame_p->buffer + dump_offset, dump_byte_output);
dump_offset += dump_byte_output;
dump_byte -= dump_byte_output;
if (dump_byte <= 0)
{
break;
}
}
ASSERT(0, "%s", __func__);
}
a2dp_audio_lhdc_lastframe_info.sequenceNumber = lhdc_decoder_frame_p->sequenceNumber;
a2dp_audio_lhdc_lastframe_info.timestamp = lhdc_decoder_frame_p->timestamp;
a2dp_audio_lhdc_lastframe_info.curSubSequenceNumber = lhdc_decoder_frame_p->curSubSequenceNumber;
a2dp_audio_lhdc_lastframe_info.totalSubSequenceNumber = lhdc_decoder_frame_p->totalSubSequenceNumber;
a2dp_audio_lhdc_lastframe_info.frame_samples = A2DP_LHDC_OUTPUT_FRAME_SAMPLES;
a2dp_audio_lhdc_lastframe_info.decoded_frames++;
a2dp_audio_lhdc_lastframe_info.undecode_frames = a2dp_audio_list_length(list) - 1;
a2dp_audio_decoder_internal_lastframe_info_set(&a2dp_audio_lhdc_lastframe_info);
a2dp_audio_list_remove(list, lhdc_decoder_frame_p);
}
exit:
if (cache_underflow)
{
reset_lhdc_assmeble_packet();
a2dp_audio_lhdc_lastframe_info.undecode_frames = 0;
a2dp_audio_decoder_internal_lastframe_info_set(&a2dp_audio_lhdc_lastframe_info);
output_byte = A2DP_DECODER_CACHE_UNDERFLOW_ERROR;
}
return output_byte;
}
int a2dp_audio_lhdc_decode_frame(uint8_t *buffer, uint32_t buffer_bytes)
{
#ifdef A2DP_CP_ACCEL
return a2dp_cp_lhdc_mcu_decode(buffer, buffer_bytes);
#else
return a2dp_audio_lhdc_mcu_decode_frame(buffer, buffer_bytes);
#endif
}
int a2dp_audio_lhdc_preparse_packet(btif_media_header_t *header, uint8_t *buffer, uint32_t buffer_bytes)
{
a2dp_audio_lhdc_lastframe_info.sequenceNumber = header->sequenceNumber;
a2dp_audio_lhdc_lastframe_info.timestamp = header->timestamp;
a2dp_audio_lhdc_lastframe_info.curSubSequenceNumber = 0;
a2dp_audio_lhdc_lastframe_info.totalSubSequenceNumber = 0;
a2dp_audio_lhdc_lastframe_info.frame_samples = A2DP_LHDC_OUTPUT_FRAME_SAMPLES;
a2dp_audio_lhdc_lastframe_info.list_samples = A2DP_LHDC_OUTPUT_FRAME_SAMPLES;
a2dp_audio_lhdc_lastframe_info.decoded_frames = 0;
a2dp_audio_lhdc_lastframe_info.undecode_frames = 0;
a2dp_audio_decoder_internal_lastframe_info_set(&a2dp_audio_lhdc_lastframe_info);
TRACE_A2DP_DECODER_I("%s seq:%d timestamp:%08x", __func__, header->sequenceNumber, header->timestamp);
return A2DP_DECODER_NO_ERROR;
}
void a2dp_audio_lhdc_free(void *packet)
{
a2dp_audio_lhdc_decoder_frame_t *decoder_frame_p = (a2dp_audio_lhdc_decoder_frame_t *)packet;
a2dp_audio_heap_free(decoder_frame_p->buffer);
a2dp_audio_heap_free(decoder_frame_p);
}
int a2dp_audio_lhdc_store_packet(btif_media_header_t *header, uint8_t *buffer, uint32_t buffer_bytes)
{
list_t *list = a2dp_audio_context_p->audio_datapath.input_raw_packet_list;
int nRet = A2DP_DECODER_NO_ERROR;
uint32_t frame_num = 0;
uint32_t frame_cnt = 0;
uint32_t lSize = 0;
uint8_t *lPTR = NULL;
lhdc_frame_Info_t lhdc_frame_Info;
uint32_t ptr_offset = 0;
if ((frame_num = assemble_lhdc_packet(buffer, buffer_bytes, &lPTR, &lSize)) > 0)
{
if (lPTR != NULL && lSize != 0)
{
ptr_offset = 0;
//TRACE_A2DP_DECODER_I("%s: There are %d frames in packet", __func__, frame_num);
while (parse_lhdc_info(lPTR + ptr_offset, &lhdc_frame_Info) == 0 && ptr_offset < lSize && frame_cnt < frame_num)
{
a2dp_audio_lhdc_decoder_frame_t *decoder_frame_p = NULL;
if (!lhdc_frame_Info.frame_len)
{
DUMP8("%02x ", lPTR + ptr_offset, 32);
ASSERT(0, "lhdc_frame_Info error frame_len:%d offset:%d ptr:%08x/%08x", lhdc_frame_Info.frame_len, ptr_offset, (uint32_t)buffer, (uint32_t)lPTR);
}
ASSERT(lhdc_frame_Info.frame_len <= (lSize - ptr_offset), "%s frame_len:%d ptr_offset:%d buffer_bytes:%d",
__func__, lhdc_frame_Info.frame_len, ptr_offset, lSize);
uint32_t list_length = a2dp_audio_list_length(list);
if (list_length < lhdc_mtu_limiter)
{
decoder_frame_p = (a2dp_audio_lhdc_decoder_frame_t *)a2dp_audio_lhdc_frame_malloc(lhdc_frame_Info.frame_len);
}
else
{
nRet = A2DP_DECODER_MTU_LIMTER_ERROR;
break;
}
frame_cnt++;
decoder_frame_p->sequenceNumber = header->sequenceNumber;
decoder_frame_p->timestamp = header->timestamp;
decoder_frame_p->curSubSequenceNumber = frame_cnt;
decoder_frame_p->totalSubSequenceNumber = frame_num;
memcpy(decoder_frame_p->buffer, lPTR + ptr_offset, lhdc_frame_Info.frame_len);
decoder_frame_p->buffer_len = lhdc_frame_Info.frame_len;
a2dp_audio_list_append(list, decoder_frame_p);
ptr_offset += lhdc_frame_Info.frame_len;
#if 0
TRACE_A2DP_DECODER_I("lhdc_store_packet save seq:%d timestamp:%d len:%d lSize:%d list_length:%d frame_len:%d Split:%d/%d",
header->sequenceNumber,
header->timestamp,
buffer_bytes,
lSize,
list_length,
lhdc_frame_Info.frame_len,
lhdc_frame_Info.isSplit,
lhdc_frame_Info.isLeft);
#endif
}
}
}
else
{
// TRACE_A2DP_DECODER_I("lhdc_store_packet skip seq:%d timestamp:%d len:%d l:%d", header->sequenceNumber, header->timestamp,buffer_bytes, lSize);
}
return nRet;
}
int a2dp_audio_lhdc_discards_packet(uint32_t packets)
{
#ifdef A2DP_CP_ACCEL
a2dp_cp_reset_frame();
#endif
int nRet = a2dp_audio_context_p->audio_decoder.audio_decoder_synchronize_dest_packet_mut(a2dp_audio_context_p->dest_packet_mut);
reset_lhdc_assmeble_packet();
#if defined(A2DP_LHDC_LARC)
lhdc_drop_frame = 0;
#endif
return nRet;
}
static int a2dp_audio_lhdc_headframe_info_get(A2DP_AUDIO_HEADFRAME_INFO_T *headframe_info)
{
list_t *list = a2dp_audio_context_p->audio_datapath.input_raw_packet_list;
list_node_t *node = NULL;
a2dp_audio_lhdc_decoder_frame_t *decoder_frame_p = NULL;
if (a2dp_audio_list_length(list))
{
node = a2dp_audio_list_begin(list);
decoder_frame_p = (a2dp_audio_lhdc_decoder_frame_t *)a2dp_audio_list_node(node);
headframe_info->sequenceNumber = decoder_frame_p->sequenceNumber;
headframe_info->timestamp = decoder_frame_p->timestamp;
headframe_info->curSubSequenceNumber = 0;
headframe_info->totalSubSequenceNumber = 0;
}
else
{
memset(headframe_info, 0, sizeof(A2DP_AUDIO_HEADFRAME_INFO_T));
}
return A2DP_DECODER_NO_ERROR;
}
int a2dp_audio_lhdc_info_get(void *info)
{
return A2DP_DECODER_NO_ERROR;
}
extern uint32_t __lhdc_license_start[];
int a2dp_audio_lhdc_init(A2DP_AUDIO_OUTPUT_CONFIG_T *config, void *context)
{
TRACE_A2DP_DECODER_I("%s %s ch:%d freq:%d bits:%d", __func__, getVersionCode(),
config->num_channels,
config->sample_rate,
config->bits_depth);
uint8_t lhdc_license_key = 0;
uint8_t *lhdc_license_data=(uint8_t * )__lhdc_license_start + 0x98;
TRACE(5,"lhdc_license_data:%p, lhdc license %02x %02x %02x %02x",lhdc_license_data,
lhdc_license_data[0],lhdc_license_data[1],lhdc_license_data[2],lhdc_license_data[3]);
a2dp_audio_context_p = (A2DP_AUDIO_CONTEXT_T *)context;
memset(&a2dp_audio_lhdc_lastframe_info, 0, sizeof(A2DP_AUDIO_DECODER_LASTFRAME_INFO_T));
memcpy(&a2dp_audio_lhdc_output_config, config, sizeof(A2DP_AUDIO_OUTPUT_CONFIG_T));
a2dp_audio_lhdc_lastframe_info.stream_info = a2dp_audio_lhdc_output_config;
a2dp_audio_lhdc_lastframe_info.frame_samples = A2DP_LHDC_OUTPUT_FRAME_SAMPLES;
a2dp_audio_lhdc_lastframe_info.list_samples = A2DP_LHDC_OUTPUT_FRAME_SAMPLES;
a2dp_audio_decoder_internal_lastframe_info_set(&a2dp_audio_lhdc_lastframe_info);
lhdc_license_key = lhdcSetLicenseKeyTable(lhdc_license_data, bes_bt_local_info_get);
TRACE_A2DP_DECODER_I("lhdc_license_key:%d", lhdc_license_key);
#if defined(A2DP_LHDC_V3)
lhdcInit(config->bits_depth, config->sample_rate, 0, VERSION_3);
#else
lhdcInit(config->bits_depth, config->sample_rate, 0, VERSION_2);
#endif
initial_lhdc_assemble_packet(false);
#ifdef A2DP_CP_ACCEL
int ret;
ret = a2dp_cp_init(a2dp_cp_lhdc_cp_decode, CP_PROC_DELAY_2_FRAMES);
ASSERT(ret == 0, "%s: a2dp_cp_init() failed: ret=%d", __func__, ret);
#endif
a2dp_audio_lhdc_list_checker();
#if defined(A2DP_LHDC_LARC)
lhdc_last_time = 0;
lhdc_drop_frame = 0;
#endif
return A2DP_DECODER_NO_ERROR;
}
int a2dp_audio_lhdc_deinit(void)
{
#ifdef A2DP_CP_ACCEL
a2dp_cp_deinit();
#endif
lhdcDestroy();
return A2DP_DECODER_NO_ERROR;
}
int a2dp_audio_lhdc_synchronize_packet(A2DP_AUDIO_SYNCFRAME_INFO_T *sync_info, uint32_t mask)
{
int nRet = A2DP_DECODER_SYNC_ERROR;
list_t *list = a2dp_audio_context_p->audio_datapath.input_raw_packet_list;
list_node_t *node = NULL;
int list_len;
a2dp_audio_lhdc_decoder_frame_t *lhdc_decoder_frame;
#ifdef A2DP_CP_ACCEL
a2dp_cp_reset_frame();
#endif
list_len = a2dp_audio_list_length(list);
for (uint16_t i = 0; i < list_len; i++)
{
node = a2dp_audio_list_begin(list);
lhdc_decoder_frame = (a2dp_audio_lhdc_decoder_frame_t *)a2dp_audio_list_node(node);
if (A2DP_AUDIO_SYNCFRAME_CHK(lhdc_decoder_frame->sequenceNumber == sync_info->sequenceNumber, A2DP_AUDIO_SYNCFRAME_MASK_SEQ, mask) &&
A2DP_AUDIO_SYNCFRAME_CHK(lhdc_decoder_frame->curSubSequenceNumber == sync_info->curSubSequenceNumber, A2DP_AUDIO_SYNCFRAME_MASK_CURRSUBSEQ, mask) &&
A2DP_AUDIO_SYNCFRAME_CHK(lhdc_decoder_frame->totalSubSequenceNumber == sync_info->totalSubSequenceNumber, A2DP_AUDIO_SYNCFRAME_MASK_TOTALSUBSEQ, mask))
{
nRet = A2DP_DECODER_NO_ERROR;
break;
}
a2dp_audio_list_remove(list, lhdc_decoder_frame);
}
node = a2dp_audio_list_begin(list);
if (node)
{
lhdc_decoder_frame = (a2dp_audio_lhdc_decoder_frame_t *)a2dp_audio_list_node(node);
TRACE_A2DP_DECODER_I("%s nRet:%d SEQ:%d timestamp:%d %d/%d", __func__, nRet, lhdc_decoder_frame->sequenceNumber, lhdc_decoder_frame->timestamp,
lhdc_decoder_frame->curSubSequenceNumber, lhdc_decoder_frame->totalSubSequenceNumber);
}
else
{
TRACE_A2DP_DECODER_I("%s nRet:%d", __func__, nRet);
}
return nRet;
}
int a2dp_audio_lhdc_synchronize_dest_packet_mut(uint16_t packet_mut)
{
list_node_t *node = NULL;
uint32_t list_len = 0;
list_t *list = a2dp_audio_context_p->audio_datapath.input_raw_packet_list;
a2dp_audio_lhdc_decoder_frame_t *lhdc_decoder_frame_p = NULL;
list_len = a2dp_audio_list_length(list);
if (list_len > packet_mut)
{
do
{
node = a2dp_audio_list_begin(list);
lhdc_decoder_frame_p = (a2dp_audio_lhdc_decoder_frame_t *)a2dp_audio_list_node(node);
a2dp_audio_list_remove(list, lhdc_decoder_frame_p);
} while (a2dp_audio_list_length(list) > packet_mut);
}
TRACE_A2DP_DECODER_I("%s list:%d", __func__, a2dp_audio_list_length(list));
return A2DP_DECODER_NO_ERROR;
}
int a2dp_audio_lhdc_convert_list_to_samples(uint32_t *samples)
{
uint32_t list_len = 0;
list_t *list = a2dp_audio_context_p->audio_datapath.input_raw_packet_list;
list_len = a2dp_audio_list_length(list);
*samples = A2DP_LHDC_OUTPUT_FRAME_SAMPLES * list_len;
TRACE_A2DP_DECODER_I("%s list:%d samples:%d", __func__, list_len, *samples);
return A2DP_DECODER_NO_ERROR;
}
int a2dp_audio_lhdc_discards_samples(uint32_t samples)
{
int nRet = A2DP_DECODER_SYNC_ERROR;
list_t *list = a2dp_audio_context_p->audio_datapath.input_raw_packet_list;
a2dp_audio_lhdc_decoder_frame_t *lhdc_decoder_frame_p = NULL;
list_node_t *node = NULL;
int need_remove_list = 0;
uint32_t list_samples = 0;
ASSERT(!(samples % A2DP_LHDC_OUTPUT_FRAME_SAMPLES), "%s samples err:%d", __func__, samples);
a2dp_audio_lhdc_convert_list_to_samples(&list_samples);
if (list_samples >= samples)
{
need_remove_list = samples / A2DP_LHDC_OUTPUT_FRAME_SAMPLES;
for (int i = 0; i < need_remove_list; i++)
{
node = a2dp_audio_list_begin(list);
lhdc_decoder_frame_p = (a2dp_audio_lhdc_decoder_frame_t *)a2dp_audio_list_node(node);
a2dp_audio_list_remove(list, lhdc_decoder_frame_p);
}
nRet = A2DP_DECODER_NO_ERROR;
node = a2dp_audio_list_begin(list);
lhdc_decoder_frame_p = (a2dp_audio_lhdc_decoder_frame_t *)a2dp_audio_list_node(node);
TRACE_A2DP_DECODER_I("%s discard %d sample cur seq:%d", __func__, samples, lhdc_decoder_frame_p->sequenceNumber);
}
return nRet;
}
A2DP_AUDIO_DECODER_T a2dp_audio_lhdc_decoder_config = {
{96000, 2, 24},
1,
a2dp_audio_lhdc_init,
a2dp_audio_lhdc_deinit,
a2dp_audio_lhdc_decode_frame,
a2dp_audio_lhdc_preparse_packet,
a2dp_audio_lhdc_store_packet,
a2dp_audio_lhdc_discards_packet,
a2dp_audio_lhdc_synchronize_packet,
a2dp_audio_lhdc_synchronize_dest_packet_mut,
a2dp_audio_lhdc_convert_list_to_samples,
a2dp_audio_lhdc_discards_samples,
a2dp_audio_lhdc_headframe_info_get,
a2dp_audio_lhdc_info_get,
a2dp_audio_lhdc_free,
} ;