pinebuds/apps/audioplayers/a2dp_decoder/a2dp_decoder_lhdc.cpp

1221 lines
40 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 "a2dp_decoder_internal.h"
2022-08-15 04:20:27 -05:00
#include "cmsis.h"
#include "hal_location.h"
#include "hal_timer.h"
#include "heap_api.h"
#include "plat_types.h"
#include <string.h>
2022-08-15 04:20:27 -05:00
typedef struct {
uint16_t sequenceNumber;
uint32_t timestamp;
uint16_t curSubSequenceNumber;
uint16_t totalSubSequenceNumber;
uint8_t *buffer;
uint32_t buffer_len;
2022-08-15 04:20:27 -05:00
} 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)
2022-08-15 04:20:27 -05:00
#else
#define PACKET_BUFFER_LENGTH (4 * 1024)
2022-08-15 04:20:27 -05:00
#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)
2022-08-15 04:20:27 -05:00
#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)
2022-08-15 04:20:27 -05:00
#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,
2022-08-15 04:20:27 -05:00
} ASM_PKT_STATUS;
typedef enum { VERSION_2 = 200, VERSION_3 = 300 } lhdc_ver_t;
2022-08-15 04:20:27 -05:00
/**
* 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
2022-08-15 04:20:27 -05:00
} 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;
2022-08-15 04:20:27 -05:00
};
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;
2022-08-15 04:20:27 -05:00
};
#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);
2022-08-15 04:20:27 -05:00
#if defined(A2DP_LHDC_V3)
uint32_t lhdcPutData(uint8_t *pIn, uint32_t len);
2022-08-15 04:20:27 -05:00
#else
uint32_t lhdcPutData(uint8_t *pIn, int32_t len);
2022-08-15 04:20:27 -05:00
#endif
uint32_t lhdcDecodeProcess(uint8_t *pOutBuf);
void lhdcDestroy();
bool lhdcSetLicenseKeyTable(uint8_t *licTable, LHDC_GET_BT_INFO pFunc);
const char *getVersionCode();
bool larcIsEnabled();
2022-08-15 04:20:27 -05:00
}
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;
2022-08-15 04:20:27 -05:00
#endif
extern int bes_bt_local_info_get(bes_bt_local_info *local_info);
// Local API declare
2022-08-15 04:20:27 -05:00
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
2022-08-15 04:20:27 -05:00
} 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) {
2022-08-15 04:20:27 -05:00
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);
2022-08-15 04:20:27 -05:00
statistic.sum_bytes = 0;
statistic.last_times = time_current;
}
2022-08-15 04:20:27 -05:00
}
#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) {
2022-08-15 04:20:27 -05:00
memcpy(&last_frame_info, f_info, sizeof(struct A2DP_CP_LHDC_IN_FRM_INFO_T));
2022-08-15 04:20:27 -05:00
}
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;
}
2022-08-15 04:20:27 -05:00
}
#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;
2022-08-15 04:20:27 -05:00
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;
2022-08-15 04:20:27 -05:00
}
static void reset_lhdc_assmeble_packet(void) {
is_synced = false;
asm_pkt_st = ASM_PKT_WAT_STR;
packet_buf_len = 0;
2022-08-15 04:20:27 -05:00
}
static void initial_lhdc_assemble_packet(bool splitFlg) {
memset(packet_buffer, 0, PACKET_BUFFER_LENGTH);
reset_lhdc_assmeble_packet();
serial_no = 0xff;
2022-08-15 04:20:27 -05:00
}
/**
* 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;
2022-08-15 04:20:27 -05:00
}
#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;
2022-08-15 04:20:27 -05:00
}
serial_no = seqno + 1;
}
2022-08-15 04:20:27 -05:00
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);
2022-08-15 04:20:27 -05:00
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;
}
2022-08-15 04:20:27 -05:00
}
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;
2022-08-15 04:20:27 -05:00
}
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;
2022-08-15 04:20:27 -05:00
}
break;
}
default:
ret = 0;
break;
}
return ret;
2022-08-15 04:20:27 -05:00
}
#endif
static int parse_lhdc_info(uint8_t *in, lhdc_frame_Info_t *h) {
2022-08-15 04:20:27 -05:00
#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;
2022-08-15 04:20:27 -05:00
}
#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;
2022-08-15 04:20:27 -05:00
#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);
2022-08-15 04:20:27 -05:00
#endif
return ret;
2022-08-15 04:20:27 -05:00
}
// 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;
2022-08-15 04:20:27 -05:00
#if defined(A2DP_LHDC_LARC)
// bool forceRecovery = false;
bool larcEnabled = larcIsEnabled();
uint32_t frame_number = 0;
2022-08-15 04:20:27 -05:00
if (larcEnabled) {
/* The LARC is enabled. */
2022-08-15 04:20:27 -05:00
uint32_t time_diff = 0, time_now = GET_CURRENT_MS();
2022-08-15 04:20:27 -05:00
if (lhdc_last_time == 0) {
lhdc_last_time = time_now;
2022-08-15 04:20:27 -05:00
}
// 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);
// }
}
2022-08-15 04:20:27 -05:00
#endif
cp_buffer_frames_max = app_bt_stream_get_dma_buffer_samples() / 2;
2022-08-15 04:20:27 -05:00
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);
}
2022-08-15 04:20:27 -05:00
out_frame_len = sizeof(*p_out_info) + buffer_bytes;
2022-08-15 04:20:27 -05:00
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;
}
2022-08-15 04:20:27 -05:00
#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;
2022-08-15 04:20:27 -05:00
}
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;
2022-08-15 04:20:27 -05:00
#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;
2022-08-15 04:20:27 -05:00
#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++;
}
2022-08-15 04:20:27 -05:00
#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;
2022-08-15 04:20:27 -05:00
}
#endif
a2dp_audio_list_remove(list, lhdc_decoder_frame_p);
node = a2dp_audio_list_begin(list);
}
2022-08-15 04:20:27 -05:00
#if defined(A2DP_LHDC_LARC)
TRACE_A2DP_DECODER_I("Put %d frames, drop %d frames", put_cnt, drop_cnt);
2022-08-15 04:20:27 -05:00
#endif
ret = a2dp_cp_get_full_out_frame((void **)&out, &out_len);
2022-08-15 04:20:27 -05:00
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;
}
2022-08-15 04:20:27 -05:00
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;
2022-08-15 04:20:27 -05:00
}
#ifdef __CP_EXCEPTION_TEST__
static bool _cp_assert = false;
int cp_assert(void) {
_cp_assert = true;
return 0;
2022-08-15 04:20:27 -05:00
}
#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;
2022-08-15 04:20:27 -05:00
#ifdef __CP_EXCEPTION_TEST__
if (_cp_assert) {
_cp_assert = false;
*(int *)0 = 1;
// ASSERT(0, "ASSERT %s %d", __func__, __LINE__);
}
2022-08-15 04:20:27 -05:00
#endif
out_frm_st = a2dp_cp_get_emtpy_out_frame((void **)&out, &out_len);
2022-08-15 04:20:27 -05:00
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));
2022-08-15 04:20:27 -05:00
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;
}
2022-08-15 04:20:27 -05:00
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);
2022-08-15 04:20:27 -05:00
dec_start = (uint8_t *)(p_out_info + 1) + p_out_info->pcm_len;
dec_len = out_len - (dec_start - (uint8_t *)out);
2022-08-15 04:20:27 -05:00
dec_sum = 0;
2022-08-15 04:20:27 -05:00
while (dec_sum < (int32_t)dec_len) {
uint32_t lhdc_decode_temp = 0;
2022-08-15 04:20:27 -05:00
ret = a2dp_cp_get_in_frame((void **)&in_buf, &in_len);
2022-08-15 04:20:27 -05:00
if (ret) {
TRACE_A2DP_DECODER_I("cp_get_int_frame fail, ret=%d", ret);
return 4;
}
2022-08-15 04:20:27 -05:00
ASSERT(in_len > sizeof(*p_in_info), "%s: Bad in_len %u (should > %u)",
__func__, in_len, sizeof(*p_in_info));
2022-08-15 04:20:27 -05:00
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);
2022-08-15 04:20:27 -05:00
#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;
}
2022-08-15 04:20:27 -05:00
#endif
lhdcPutData(in_buf, in_len);
// TRACE_A2DP_DECODER_I("%s:put length=%d, dec_len=%d", __func__, in_len,
// dec_len);
2022-08-15 04:20:27 -05:00
#else
lhdcPutData(in_buf, in_len);
2022-08-15 04:20:27 -05:00
#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;
2022-08-15 04:20:27 -05:00
}
ret = a2dp_cp_consume_in_frame();
2022-08-15 04:20:27 -05:00
if (ret != 0) {
TRACE_A2DP_DECODER_I("%s: a2dp_cp_consume_in_frame() failed: ret=%d",
__func__, ret);
2022-08-15 04:20:27 -05:00
}
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;
2022-08-15 04:20:27 -05:00
}
#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);
2022-08-15 04:20:27 -05:00
TRACE_A2DP_DECODER_I("%s cnt:%d list:%d", __func__, cnt,
a2dp_audio_list_length(list));
2022-08-15 04:20:27 -05:00
return 0;
2022-08-15 04:20:27 -05:00
}
#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;
2022-08-15 04:20:27 -05:00
bool cache_underflow = false;
int output_byte = 0;
2022-08-15 04:20:27 -05:00
uint32_t lhdc_out_len = 0;
2022-08-15 04:20:27 -05:00
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);
2022-08-15 04:20:27 -05:00
#if defined(A2DP_LHDC_V3)
lhdcPutData(lhdc_decoder_frame_p->buffer, lhdc_decoder_frame_p->buffer_len);
2022-08-15 04:20:27 -05:00
#else
lhdcPutData(lhdc_decoder_frame_p->buffer, lhdc_decoder_frame_p->buffer_len);
2022-08-15 04:20:27 -05:00
#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;
2022-08-15 04:20:27 -05:00
}
}
ASSERT(0, "%s", __func__);
2022-08-15 04:20:27 -05:00
}
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);
}
2022-08-15 04:20:27 -05:00
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;
2022-08-15 04:20:27 -05:00
}
int a2dp_audio_lhdc_decode_frame(uint8_t *buffer, uint32_t buffer_bytes) {
2022-08-15 04:20:27 -05:00
#ifdef A2DP_CP_ACCEL
return a2dp_cp_lhdc_mcu_decode(buffer, buffer_bytes);
2022-08-15 04:20:27 -05:00
#else
return a2dp_audio_lhdc_mcu_decode_frame(buffer, buffer_bytes);
2022-08-15 04:20:27 -05:00
#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;
2022-08-15 04:20:27 -05:00
}
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);
2022-08-15 04:20:27 -05:00
}
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;
2022-08-15 04:20:27 -05:00
#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
}
2022-08-15 04:20:27 -05:00
}
} else {
// TRACE_A2DP_DECODER_I("lhdc_store_packet skip seq:%d timestamp:%d
// len:%d l:%d", header->sequenceNumber,
// header->timestamp,buffer_bytes, lSize);
}
2022-08-15 04:20:27 -05:00
return nRet;
2022-08-15 04:20:27 -05:00
}
int a2dp_audio_lhdc_discards_packet(uint32_t packets) {
2022-08-15 04:20:27 -05:00
#ifdef A2DP_CP_ACCEL
a2dp_cp_reset_frame();
2022-08-15 04:20:27 -05:00
#endif
int nRet = a2dp_audio_context_p->audio_decoder
.audio_decoder_synchronize_dest_packet_mut(
a2dp_audio_context_p->dest_packet_mut);
2022-08-15 04:20:27 -05:00
reset_lhdc_assmeble_packet();
2022-08-15 04:20:27 -05:00
#if defined(A2DP_LHDC_LARC)
lhdc_drop_frame = 0;
2022-08-15 04:20:27 -05:00
#endif
return nRet;
2022-08-15 04:20:27 -05:00
}
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;
2022-08-15 04:20:27 -05:00
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;
2022-08-15 04:20:27 -05:00
}
int a2dp_audio_lhdc_info_get(void *info) { return A2DP_DECODER_NO_ERROR; }
2022-08-15 04:20:27 -05:00
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);
2022-08-15 04:20:27 -05:00
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]);
2022-08-15 04:20:27 -05:00
a2dp_audio_context_p = (A2DP_AUDIO_CONTEXT_T *)context;
2022-08-15 04:20:27 -05:00
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);
2022-08-15 04:20:27 -05:00
lhdc_license_key =
lhdcSetLicenseKeyTable(lhdc_license_data, bes_bt_local_info_get);
2022-08-15 04:20:27 -05:00
TRACE_A2DP_DECODER_I("lhdc_license_key:%d", lhdc_license_key);
2022-08-15 04:20:27 -05:00
#if defined(A2DP_LHDC_V3)
lhdcInit(config->bits_depth, config->sample_rate, 0, VERSION_3);
2022-08-15 04:20:27 -05:00
#else
lhdcInit(config->bits_depth, config->sample_rate, 0, VERSION_2);
2022-08-15 04:20:27 -05:00
#endif
initial_lhdc_assemble_packet(false);
2022-08-15 04:20:27 -05:00
#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);
2022-08-15 04:20:27 -05:00
#endif
a2dp_audio_lhdc_list_checker();
2022-08-15 04:20:27 -05:00
#if defined(A2DP_LHDC_LARC)
lhdc_last_time = 0;
lhdc_drop_frame = 0;
2022-08-15 04:20:27 -05:00
#endif
return A2DP_DECODER_NO_ERROR;
2022-08-15 04:20:27 -05:00
}
int a2dp_audio_lhdc_deinit(void) {
2022-08-15 04:20:27 -05:00
#ifdef A2DP_CP_ACCEL
a2dp_cp_deinit();
2022-08-15 04:20:27 -05:00
#endif
lhdcDestroy();
return A2DP_DECODER_NO_ERROR;
2022-08-15 04:20:27 -05:00
}
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;
2022-08-15 04:20:27 -05:00
#ifdef A2DP_CP_ACCEL
a2dp_cp_reset_frame();
2022-08-15 04:20:27 -05:00
#endif
list_len = a2dp_audio_list_length(list);
2022-08-15 04:20:27 -05:00
for (uint16_t i = 0; i < list_len; i++) {
2022-08-15 04:20:27 -05:00
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;
2022-08-15 04:20:27 -05:00
}
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;
2022-08-15 04:20:27 -05:00
}
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;
2022-08-15 04:20:27 -05:00
}
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;
2022-08-15 04:20:27 -05:00
list_len = a2dp_audio_list_length(list);
2022-08-15 04:20:27 -05:00
*samples = A2DP_LHDC_OUTPUT_FRAME_SAMPLES * list_len;
2022-08-15 04:20:27 -05:00
TRACE_A2DP_DECODER_I("%s list:%d samples:%d", __func__, list_len, *samples);
2022-08-15 04:20:27 -05:00
return A2DP_DECODER_NO_ERROR;
2022-08-15 04:20:27 -05:00
}
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);
2022-08-15 04:20:27 -05:00
}
nRet = A2DP_DECODER_NO_ERROR;
2022-08-15 04:20:27 -05:00
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;
2022-08-15 04:20:27 -05:00
}
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,
};