pinebuds/apps/audioplayers/a2dp_decoder/a2dp_decoder_lhdc.cpp

1221 lines
40 KiB
C++
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/***************************************************************************
*
* 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 "a2dp_decoder_internal.h"
#include "cmsis.h"
#include "hal_location.h"
#include "hal_timer.h"
#include "heap_api.h"
#include "plat_types.h"
#include <string.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,
};