2065 lines
67 KiB
C++
2065 lines
67 KiB
C++
/***************************************************************************
|
|
*
|
|
* 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.h"
|
|
#include "a2dp_decoder_internal.h"
|
|
#include "app_audio.h"
|
|
#include "app_bt.h"
|
|
#include "app_bt_media_manager.h"
|
|
#include "audioflinger.h"
|
|
#include "avdtp_api.h"
|
|
#include "bt_drv_reg_op.h"
|
|
#include "cmsis.h"
|
|
#include "cmsis_os.h"
|
|
#include "codec_sbc.h"
|
|
#include "hal_location.h"
|
|
#include "hal_timer.h"
|
|
#include "heap_api.h"
|
|
#include "plat_types.h"
|
|
#include <string.h>
|
|
#if defined(IBRT)
|
|
#include "app_ibrt_if.h"
|
|
#include "app_tws_ctrl_thread.h"
|
|
#include "app_tws_ibrt_audio_analysis.h"
|
|
#include "app_tws_ibrt_audio_sync.h"
|
|
#include "app_tws_ibrt_cmd_handler.h"
|
|
#include "bt_drv_interface.h"
|
|
#include "btapp.h"
|
|
#endif
|
|
#include "audio_prompt_sbc.h"
|
|
#include "crc32.h"
|
|
|
|
#ifndef A2DP_AUDIO_MEMPOOL_SIZE
|
|
#if defined(A2DP_LDAC_ON)
|
|
#define A2DP_AUDIO_LDAC_MEMPOOL_SIZE (76 * 1024)
|
|
#endif
|
|
#define A2DP_AUDIO_MEMPOOL_SIZE (72 * 1024)
|
|
#endif
|
|
#define A2DP_AUDIO_WAIT_TIMEOUT_MS (500)
|
|
#define A2DP_AUDIO_MUTE_FRAME_CNT_AFTER_NO_CACHE (25)
|
|
#define A2DP_AUDIO_SKIP_FRAME_LIMIT_AFTER_NO_CACHE (50)
|
|
|
|
#define A2DP_AUDIO_REFILL_AFTER_NO_CACHE (1)
|
|
|
|
#ifdef __A2DP_AUDIO_SYNC_FIX_DIFF_NOPID__
|
|
#define A2DP_AUDIO_SYNC_INTERVAL (25)
|
|
#else
|
|
#define A2DP_AUDIO_SYNC_INTERVAL (1000)
|
|
#endif
|
|
|
|
#define A2DP_AUDIO_LATENCY_LOW_FACTOR (1.0f)
|
|
#define A2DP_AUDIO_LATENCY_HIGH_FACTOR (1.6f)
|
|
#define A2DP_AUDIO_LATENCY_MORE_FACTOR (1.2f)
|
|
|
|
#define A2DP_AUDIO_SYNC_FACTOR_REFERENCE (1.0f)
|
|
#define A2DP_AUDIO_SYNC_FACTOR_FAST_LIMIT (0.00015f)
|
|
#define A2DP_AUDIO_SYNC_FACTOR_SLOW_LIMIT (-0.00035f)
|
|
#define A2DP_AUDIO_SYNC_FACTOR_NEED_FAST_CACHE (-0.001f)
|
|
|
|
#define A2DP_AUDIO_UNDERFLOW_CAUSE_AUDIO_RETRIGGER (1)
|
|
|
|
extern A2DP_AUDIO_DECODER_T a2dp_audio_sbc_decoder_config;
|
|
#if defined(A2DP_AAC_ON)
|
|
extern A2DP_AUDIO_DECODER_T a2dp_audio_aac_lc_decoder_config;
|
|
#endif
|
|
#if defined(A2DP_LHDC_ON)
|
|
extern A2DP_AUDIO_DECODER_T a2dp_audio_lhdc_decoder_config;
|
|
#endif
|
|
|
|
#if defined(A2DP_LDAC_ON)
|
|
extern A2DP_AUDIO_DECODER_T a2dp_audio_ldac_decoder_config;
|
|
#endif
|
|
|
|
#if defined(A2DP_SCALABLE_ON)
|
|
extern A2DP_AUDIO_DECODER_T a2dp_audio_scalable_decoder_config;
|
|
#endif
|
|
|
|
osSemaphoreDef(audio_buffer_semaphore);
|
|
osMutexDef(audio_buffer_mutex);
|
|
osMutexDef(audio_status_mutex);
|
|
osMutexDef(audio_stop_mutex);
|
|
|
|
#ifdef __A2DP_AUDIO_SYNC_FIX_DIFF_NOPID__
|
|
//#define A2DP_AUDIO_SYNC_FIX_DIFF_INTERVA_PRINT_FLOAT (1)
|
|
|
|
#define A2DP_AUDIO_SYNC_FIX_DIFF_INTERVAL (640)
|
|
#define A2DP_AUDIO_SYNC_FIX_DIFF_FAST_LIMIT (0.003f)
|
|
#define A2DP_AUDIO_SYNC_FIX_DIFF_SLOW_LIMIT (-0.003f)
|
|
|
|
extern uint32_t app_bt_stream_get_dma_buffer_samples(void);
|
|
static int a2dp_audio_sync_fix_diff_proc(uint32_t tick);
|
|
static int a2dp_audio_sync_fix_diff_stop(uint32_t tick);
|
|
static int32_t a2dp_audio_sync_fix_diff_reset(void);
|
|
static int32_t a2dp_audio_sync_fix_diff_start(uint32_t tick);
|
|
|
|
typedef enum {
|
|
A2DP_AUDIO_SYNC_FIX_DIFF_STATUS_IDLE,
|
|
A2DP_AUDIO_SYNC_FIX_DIFF_STATUS_START,
|
|
A2DP_AUDIO_SYNC_FIX_DIFF_STATUS_STOP,
|
|
} A2DP_AUDIO_SYNC_FIX_DIFF_STATUS;
|
|
|
|
typedef struct {
|
|
int32_t tick;
|
|
A2DP_AUDIO_SYNC_FIX_DIFF_STATUS status;
|
|
} A2DP_AUDIO_SYNC_FIX_DIFF_T;
|
|
|
|
static A2DP_AUDIO_SYNC_FIX_DIFF_T a2dp_audio_sync_fix_diff;
|
|
#endif
|
|
|
|
A2DP_AUDIO_CONTEXT_T a2dp_audio_context;
|
|
|
|
static heap_handle_t a2dp_audio_heap;
|
|
|
|
static A2DP_AUDIO_LASTFRAME_INFO_T a2dp_audio_lastframe_info;
|
|
|
|
static A2DP_AUDIO_DETECT_NEXT_PACKET_CALLBACK
|
|
a2dp_audio_detect_next_packet_callback = NULL;
|
|
static A2DP_AUDIO_DETECT_NEXT_PACKET_CALLBACK a2dp_audio_store_packet_callback =
|
|
NULL;
|
|
|
|
static float a2dp_audio_latency_factor = A2DP_AUDIO_LATENCY_LOW_FACTOR;
|
|
static float sync_tune_dest_ratio = 1.0f;
|
|
|
|
static uint32_t store_packet_history_loctime = 0;
|
|
|
|
static uint32_t check_sum_seed = 0;
|
|
|
|
static int a2dp_audio_internal_lastframe_info_ptr_get(
|
|
A2DP_AUDIO_LASTFRAME_INFO_T **lastframe_info);
|
|
|
|
float a2dp_audio_get_sample_reference(void);
|
|
int8_t a2dp_audio_get_current_buf_size(void);
|
|
|
|
void a2dp_audio_heap_init(void *begin_addr, uint32_t size) {
|
|
a2dp_audio_heap = heap_register(begin_addr, size);
|
|
}
|
|
|
|
void *a2dp_audio_heap_malloc(uint32_t size) {
|
|
void *ptr = heap_malloc(a2dp_audio_heap, size);
|
|
ASSERT_A2DP_DECODER(ptr, "%s size:%d", __func__, size);
|
|
return ptr;
|
|
}
|
|
|
|
void *a2dp_audio_heap_cmalloc(uint32_t size) {
|
|
void *ptr = heap_malloc(a2dp_audio_heap, size);
|
|
ASSERT_A2DP_DECODER(ptr, "%s size:%d", __func__, size);
|
|
memset(ptr, 0, size);
|
|
return ptr;
|
|
}
|
|
|
|
void *a2dp_audio_heap_realloc(void *rmem, uint32_t newsize) {
|
|
void *ptr = heap_realloc(a2dp_audio_heap, rmem, newsize);
|
|
ASSERT_A2DP_DECODER(ptr, "%s rmem:%p size:%d", __func__, rmem, newsize);
|
|
return ptr;
|
|
}
|
|
|
|
void a2dp_audio_heap_free(void *rmem) {
|
|
ASSERT_A2DP_DECODER(rmem, "%s rmem:%p", __func__, rmem);
|
|
heap_free(a2dp_audio_heap, rmem);
|
|
}
|
|
|
|
void a2dp_audio_heap_info(size_t *total, size_t *used, size_t *max_used) {
|
|
multi_heap_info_t info;
|
|
heap_get_info(a2dp_audio_heap, &info);
|
|
|
|
if (total != NULL)
|
|
*total = info.total_bytes;
|
|
|
|
if (used != NULL)
|
|
*used = info.total_allocated_bytes;
|
|
|
|
if (max_used != NULL)
|
|
*max_used = info.total_bytes - info.minimum_free_bytes;
|
|
}
|
|
|
|
int inline a2dp_audio_semaphore_init(void) {
|
|
if (a2dp_audio_context.audio_semaphore.semaphore == NULL) {
|
|
a2dp_audio_context.audio_semaphore.semaphore =
|
|
osSemaphoreCreate(osSemaphore(audio_buffer_semaphore), 0);
|
|
}
|
|
a2dp_audio_context.audio_semaphore.enalbe = false;
|
|
return 0;
|
|
}
|
|
|
|
int inline a2dp_audio_buffer_mutex_init(void) {
|
|
if (a2dp_audio_context.audio_buffer_mutex == NULL) {
|
|
a2dp_audio_context.audio_buffer_mutex =
|
|
osMutexCreate((osMutex(audio_buffer_mutex)));
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int inline a2dp_audio_buffer_mutex_lock(void) {
|
|
osMutexWait((osMutexId)a2dp_audio_context.audio_buffer_mutex, osWaitForever);
|
|
return 0;
|
|
}
|
|
|
|
int inline a2dp_audio_buffer_mutex_unlock(void) {
|
|
osMutexRelease((osMutexId)a2dp_audio_context.audio_buffer_mutex);
|
|
return 0;
|
|
}
|
|
|
|
int inline a2dp_audio_status_mutex_init(void) {
|
|
if (a2dp_audio_context.audio_status_mutex == NULL) {
|
|
a2dp_audio_context.audio_status_mutex =
|
|
osMutexCreate((osMutex(audio_status_mutex)));
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int inline a2dp_audio_status_mutex_lock(void) {
|
|
osMutexWait((osMutexId)a2dp_audio_context.audio_status_mutex, osWaitForever);
|
|
return 0;
|
|
}
|
|
|
|
int inline a2dp_audio_status_mutex_unlock(void) {
|
|
osMutexRelease((osMutexId)a2dp_audio_context.audio_status_mutex);
|
|
return 0;
|
|
}
|
|
|
|
int inline a2dp_audio_semaphore_wait(uint32_t timeout_ms) {
|
|
osSemaphoreId semaphore_id =
|
|
(osSemaphoreId)a2dp_audio_context.audio_semaphore.semaphore;
|
|
|
|
a2dp_audio_buffer_mutex_lock();
|
|
a2dp_audio_context.audio_semaphore.enalbe = true;
|
|
a2dp_audio_buffer_mutex_unlock();
|
|
|
|
int32_t nRet = osSemaphoreWait(semaphore_id, timeout_ms);
|
|
if ((0 == nRet) || (-1 == nRet)) {
|
|
TRACE_A2DP_DECODER_W("%s wait timerout", __func__);
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int inline a2dp_audio_semaphore_release(void) {
|
|
bool enalbe = false;
|
|
|
|
a2dp_audio_buffer_mutex_lock();
|
|
if (a2dp_audio_context.audio_semaphore.enalbe) {
|
|
a2dp_audio_context.audio_semaphore.enalbe = false;
|
|
enalbe = true;
|
|
}
|
|
a2dp_audio_buffer_mutex_unlock();
|
|
|
|
if (enalbe) {
|
|
osSemaphoreId semaphore_id =
|
|
(osSemaphoreId)a2dp_audio_context.audio_semaphore.semaphore;
|
|
osSemaphoreRelease(semaphore_id);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
list_node_t *a2dp_audio_list_begin(const list_t *list) {
|
|
a2dp_audio_buffer_mutex_lock();
|
|
list_node_t *node = list_begin(list);
|
|
a2dp_audio_buffer_mutex_unlock();
|
|
return node;
|
|
}
|
|
|
|
list_node_t *a2dp_audio_list_end(const list_t *list) {
|
|
a2dp_audio_buffer_mutex_lock();
|
|
list_node_t *node = list_end(list);
|
|
a2dp_audio_buffer_mutex_unlock();
|
|
return node;
|
|
}
|
|
|
|
uint32_t a2dp_audio_list_length(const list_t *list) {
|
|
a2dp_audio_buffer_mutex_lock();
|
|
uint32_t length = list_length(list);
|
|
a2dp_audio_buffer_mutex_unlock();
|
|
return length;
|
|
}
|
|
|
|
void *a2dp_audio_list_node(const list_node_t *node) {
|
|
a2dp_audio_buffer_mutex_lock();
|
|
void *data = list_node(node);
|
|
a2dp_audio_buffer_mutex_unlock();
|
|
return data;
|
|
}
|
|
|
|
list_node_t *a2dp_audio_list_next(const list_node_t *node) {
|
|
a2dp_audio_buffer_mutex_lock();
|
|
list_node_t *next = list_next(node);
|
|
a2dp_audio_buffer_mutex_unlock();
|
|
return next;
|
|
}
|
|
|
|
bool a2dp_audio_list_remove(list_t *list, void *data) {
|
|
a2dp_audio_buffer_mutex_lock();
|
|
bool nRet = list_remove(list, data);
|
|
a2dp_audio_buffer_mutex_unlock();
|
|
return nRet;
|
|
}
|
|
|
|
bool a2dp_audio_list_append(list_t *list, void *data) {
|
|
a2dp_audio_buffer_mutex_lock();
|
|
bool nRet = list_append(list, data);
|
|
a2dp_audio_buffer_mutex_unlock();
|
|
return nRet;
|
|
}
|
|
|
|
void a2dp_audio_list_clear(list_t *list) {
|
|
a2dp_audio_buffer_mutex_lock();
|
|
list_clear(list);
|
|
a2dp_audio_buffer_mutex_unlock();
|
|
}
|
|
|
|
void a2dp_audio_list_free(list_t *list) {
|
|
a2dp_audio_buffer_mutex_lock();
|
|
list_free(list);
|
|
a2dp_audio_buffer_mutex_unlock();
|
|
}
|
|
|
|
list_t *a2dp_audio_list_new(list_free_cb callback, list_mempool_zmalloc zmalloc,
|
|
list_mempool_free free) {
|
|
a2dp_audio_buffer_mutex_lock();
|
|
list_t *list = list_new(callback, zmalloc, free);
|
|
a2dp_audio_buffer_mutex_unlock();
|
|
return list;
|
|
}
|
|
|
|
uint32_t a2dp_audio_get_passed(uint32_t curr_ticks, uint32_t prev_ticks,
|
|
uint32_t max_ticks) {
|
|
if (curr_ticks < prev_ticks)
|
|
return ((max_ticks - prev_ticks + 1) + curr_ticks);
|
|
else
|
|
return (curr_ticks - prev_ticks);
|
|
}
|
|
|
|
#ifdef A2DP_CP_ACCEL
|
|
extern "C" uint32_t get_in_cp_frame_cnt(void);
|
|
extern "C" uint32_t get_in_cp_frame_delay(void);
|
|
#else
|
|
static uint32_t get_in_cp_frame_cnt(void) { return 0; }
|
|
|
|
static uint32_t get_in_cp_frame_delay(void) { return 0; }
|
|
#endif
|
|
|
|
int inline a2dp_audio_set_status(
|
|
enum A2DP_AUDIO_DECODER_STATUS decoder_status) {
|
|
a2dp_audio_status_mutex_lock();
|
|
a2dp_audio_context.audio_decoder_status = decoder_status;
|
|
a2dp_audio_status_mutex_unlock();
|
|
|
|
return 0;
|
|
}
|
|
|
|
enum A2DP_AUDIO_DECODER_STATUS inline a2dp_audio_get_status(void) {
|
|
enum A2DP_AUDIO_DECODER_STATUS decoder_status;
|
|
a2dp_audio_status_mutex_lock();
|
|
decoder_status = a2dp_audio_context.audio_decoder_status;
|
|
a2dp_audio_status_mutex_unlock();
|
|
|
|
return decoder_status;
|
|
}
|
|
|
|
int inline a2dp_audio_set_store_packet_status(
|
|
enum A2DP_AUDIO_DECODER_STORE_PACKET_STATUS store_packet_status) {
|
|
a2dp_audio_status_mutex_lock();
|
|
a2dp_audio_context.store_packet_status = store_packet_status;
|
|
a2dp_audio_status_mutex_unlock();
|
|
|
|
return 0;
|
|
}
|
|
|
|
enum A2DP_AUDIO_DECODER_STORE_PACKET_STATUS inline a2dp_audio_get_store_packet_status(
|
|
void) {
|
|
enum A2DP_AUDIO_DECODER_STORE_PACKET_STATUS store_packet_status;
|
|
a2dp_audio_status_mutex_lock();
|
|
store_packet_status = a2dp_audio_context.store_packet_status;
|
|
a2dp_audio_status_mutex_unlock();
|
|
|
|
return store_packet_status;
|
|
}
|
|
|
|
int inline a2dp_audio_set_playback_status(
|
|
enum A2DP_AUDIO_DECODER_PLAYBACK_STATUS playback_status) {
|
|
a2dp_audio_status_mutex_lock();
|
|
a2dp_audio_context.playback_status = playback_status;
|
|
a2dp_audio_status_mutex_unlock();
|
|
|
|
return 0;
|
|
}
|
|
|
|
enum A2DP_AUDIO_DECODER_PLAYBACK_STATUS inline a2dp_audio_get_playback_status(
|
|
void) {
|
|
enum A2DP_AUDIO_DECODER_PLAYBACK_STATUS playback_status;
|
|
a2dp_audio_status_mutex_lock();
|
|
playback_status = a2dp_audio_context.playback_status;
|
|
a2dp_audio_status_mutex_unlock();
|
|
|
|
return playback_status;
|
|
}
|
|
|
|
static void a2dp_decoder_info_checker(void) {
|
|
// app_bt_start_custom_function_in_app_thread(0, 0,
|
|
// (uint32_t)bt_drv_reg_op_bt_info_checker);
|
|
}
|
|
|
|
int a2dp_audio_sync_pid_config(void) {
|
|
A2DP_AUDIO_SYNC_T *audio_sync = &a2dp_audio_context.audio_sync;
|
|
A2DP_AUDIO_SYNC_PID_T *pid = &audio_sync->pid;
|
|
|
|
pid->proportiongain = 0.4f;
|
|
pid->integralgain = 0.1f;
|
|
pid->derivativegain = 0.6f;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_sync_reset_data(void) {
|
|
A2DP_AUDIO_SYNC_T *audio_sync = &a2dp_audio_context.audio_sync;
|
|
|
|
a2dp_audio_status_mutex_lock();
|
|
audio_sync->tick = 0;
|
|
audio_sync->cnt = 0;
|
|
a2dp_audio_sync_pid_config();
|
|
#ifdef __A2DP_AUDIO_SYNC_FIX_DIFF_NOPID__
|
|
a2dp_audio_sync_fix_diff_reset();
|
|
#endif
|
|
a2dp_audio_status_mutex_unlock();
|
|
|
|
TRACE_A2DP_DECODER_I("[SYNC]reset_data");
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_sync_init(double ratio) {
|
|
#ifdef __A2DP_AUDIO_SYNC_FIX_DIFF_NOPID__
|
|
a2dp_audio_sync_fix_diff_reset();
|
|
#endif
|
|
a2dp_audio_sync_reset_data();
|
|
a2dp_audio_sync_tune_sample_rate(ratio);
|
|
sync_tune_dest_ratio = (float)ratio;
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_sync_tune_sample_rate(double ratio) {
|
|
float resample_rate_ratio;
|
|
bool need_tune = false;
|
|
|
|
a2dp_audio_status_mutex_lock();
|
|
if (a2dp_audio_context.output_cfg.factor_reference != (float)ratio) {
|
|
a2dp_audio_context.output_cfg.factor_reference = (float)ratio;
|
|
resample_rate_ratio = (float)(ratio - (double)1.0);
|
|
need_tune = true;
|
|
}
|
|
// a2dp_audio_status_mutex_unlock();
|
|
|
|
if (need_tune) {
|
|
app_audio_manager_tune_samplerate_ratio(AUD_STREAM_PLAYBACK,
|
|
resample_rate_ratio);
|
|
TRACE_A2DP_DECODER_I("[SYNC] ppb:%d ratio:%08d",
|
|
(int32_t)(resample_rate_ratio * 10000000),
|
|
(int32_t)(ratio * 10000000));
|
|
}
|
|
a2dp_audio_status_mutex_unlock();
|
|
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_sync_direct_tune_sample_rate(double ratio) {
|
|
float resample_rate_ratio;
|
|
bool need_tune = false;
|
|
|
|
a2dp_audio_status_mutex_lock();
|
|
if (a2dp_audio_context.output_cfg.factor_reference != (float)ratio) {
|
|
a2dp_audio_context.output_cfg.factor_reference = (float)ratio;
|
|
resample_rate_ratio = (float)(ratio - (double)1.0);
|
|
need_tune = true;
|
|
}
|
|
|
|
if (need_tune) {
|
|
af_codec_direct_tune(AUD_STREAM_PLAYBACK, resample_rate_ratio);
|
|
|
|
TRACE_A2DP_DECODER_I("[SYNC] ppb:%d ratio:%08d",
|
|
(int32_t)(resample_rate_ratio * 10000000),
|
|
(int32_t)(ratio * 10000000));
|
|
}
|
|
a2dp_audio_status_mutex_unlock();
|
|
|
|
return 0;
|
|
}
|
|
|
|
bool a2dp_audio_sync_tune_onprocess(void) {
|
|
bool nRet = false;
|
|
;
|
|
if (a2dp_audio_context.output_cfg.factor_reference != sync_tune_dest_ratio) {
|
|
nRet = true;
|
|
}
|
|
return nRet;
|
|
}
|
|
|
|
int a2dp_audio_sync_tune(float ratio) {
|
|
int nRet = 0;
|
|
if (sync_tune_dest_ratio == ratio) {
|
|
goto exit;
|
|
}
|
|
sync_tune_dest_ratio = ratio;
|
|
#if defined(IBRT)
|
|
if (app_ibrt_ui_is_profile_exchanged()) {
|
|
if (app_tws_ibrt_mobile_link_connected()) {
|
|
APP_TWS_IBRT_AUDIO_SYNC_TUNE_T sync_tune;
|
|
sync_tune.factor_reference = ratio;
|
|
if (!app_tws_ibrt_audio_sync_tune_need_skip()) {
|
|
tws_ctrl_send_cmd(APP_TWS_CMD_SYNC_TUNE, (uint8_t *)&sync_tune,
|
|
sizeof(APP_TWS_IBRT_AUDIO_SYNC_TUNE_T));
|
|
} else {
|
|
a2dp_audio_sync_tune_cancel();
|
|
nRet = -1;
|
|
}
|
|
}
|
|
} else {
|
|
a2dp_audio_sync_tune_sample_rate(ratio);
|
|
}
|
|
#else
|
|
a2dp_audio_sync_tune_sample_rate(ratio);
|
|
#endif
|
|
exit:
|
|
return nRet;
|
|
}
|
|
|
|
int a2dp_audio_sync_tune_cancel(void) {
|
|
sync_tune_dest_ratio = a2dp_audio_context.output_cfg.factor_reference;
|
|
return 0;
|
|
}
|
|
|
|
float a2dp_audio_sync_pid_calc(A2DP_AUDIO_SYNC_PID_T *pid, float diff) {
|
|
float increment;
|
|
float pError, dError, iError;
|
|
|
|
pid->error[0] = diff;
|
|
pError = pid->error[0] - pid->error[1];
|
|
iError = pid->error[0];
|
|
dError = pid->error[0] - 2 * pid->error[1] + pid->error[2];
|
|
|
|
increment = pid->proportiongain * pError + pid->integralgain * iError +
|
|
pid->derivativegain * dError;
|
|
|
|
pid->error[2] = pid->error[1];
|
|
pid->error[1] = pid->error[0];
|
|
|
|
pid->result += increment;
|
|
|
|
return pid->result;
|
|
}
|
|
|
|
#ifdef __A2DP_AUDIO_SYNC_FIX_DIFF_NOPID__
|
|
static int a2dp_audio_sync_fix_diff_proc(uint32_t tick) {
|
|
if (a2dp_audio_sync_fix_diff.status ==
|
|
A2DP_AUDIO_SYNC_FIX_DIFF_STATUS_START) {
|
|
if (a2dp_audio_sync_fix_diff.tick > 0) {
|
|
a2dp_audio_sync_fix_diff.tick--;
|
|
} else {
|
|
a2dp_audio_sync_fix_diff_stop(0);
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int32_t a2dp_audio_sync_fix_diff_start(uint32_t tick) {
|
|
TRACE_A2DP_DECODER_I("[SYNC] fix diff start");
|
|
|
|
a2dp_audio_sync_fix_diff.status = A2DP_AUDIO_SYNC_FIX_DIFF_STATUS_START;
|
|
a2dp_audio_sync_fix_diff.tick = tick;
|
|
return 0;
|
|
}
|
|
|
|
static int a2dp_audio_sync_fix_diff_stop(uint32_t tick) {
|
|
TRACE_A2DP_DECODER_I("[SYNC] fix diff stop");
|
|
|
|
a2dp_audio_sync_fix_diff.status = A2DP_AUDIO_SYNC_FIX_DIFF_STATUS_STOP;
|
|
a2dp_audio_sync_fix_diff.tick = 0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int32_t a2dp_audio_sync_fix_diff_reset(void) {
|
|
TRACE_A2DP_DECODER_I("[SYNC] fix diff reset");
|
|
|
|
a2dp_audio_sync_fix_diff.status = A2DP_AUDIO_SYNC_FIX_DIFF_STATUS_IDLE;
|
|
a2dp_audio_sync_fix_diff.tick = 0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static A2DP_AUDIO_SYNC_FIX_DIFF_STATUS
|
|
a2dp_audio_sync_fix_diff_status_get(void) {
|
|
return a2dp_audio_sync_fix_diff.status;
|
|
}
|
|
|
|
int a2dp_audio_sync_handler(uint8_t *buffer, uint32_t buffer_bytes) {
|
|
A2DP_AUDIO_LASTFRAME_INFO_T *lastframe_info = NULL;
|
|
A2DP_AUDIO_SYNC_T *audio_sync = &a2dp_audio_context.audio_sync;
|
|
float diff_mtu = 0;
|
|
bool need_tune = false;
|
|
bool force_slow = false;
|
|
|
|
#if defined(IBRT)
|
|
if (!app_tws_ibrt_mobile_link_connected()) {
|
|
return -1;
|
|
}
|
|
#endif
|
|
if (a2dp_audio_internal_lastframe_info_ptr_get(&lastframe_info) < 0) {
|
|
return -1;
|
|
}
|
|
|
|
a2dp_audio_sync_fix_diff_proc(audio_sync->tick);
|
|
if (a2dp_audio_sync_fix_diff_status_get() ==
|
|
A2DP_AUDIO_SYNC_FIX_DIFF_STATUS_STOP) {
|
|
#if defined(IBRT)
|
|
if (!app_tws_ibrt_audio_sync_tune_onprocess() &&
|
|
!a2dp_audio_sync_tune_onprocess()
|
|
#else
|
|
if (!a2dp_audio_sync_tune_onprocess()
|
|
#endif
|
|
) {
|
|
int sync_tune_result = 0;
|
|
if (a2dp_audio_context.output_cfg.factor_reference !=
|
|
a2dp_audio_context.init_factor_reference) {
|
|
sync_tune_result =
|
|
a2dp_audio_sync_tune(a2dp_audio_context.init_factor_reference);
|
|
}
|
|
if (!sync_tune_result) {
|
|
a2dp_audio_sync_fix_diff_reset();
|
|
#ifdef A2DP_AUDIO_SYNC_FIX_DIFF_INTERVA_PRINT_FLOAT
|
|
TRACE_A2DP_DECODER_I("[SYNC] tune ratio normal %f mut:%5.3f->%d",
|
|
(double)a2dp_audio_context.init_factor_reference,
|
|
(double)a2dp_audio_context.average_packet_mut,
|
|
a2dp_audio_context.dest_packet_mut);
|
|
#else
|
|
TRACE_A2DP_DECODER_I(
|
|
"[SYNC] tune ratio normal %d mut:%d->%d",
|
|
(int32_t)(a2dp_audio_context.init_factor_reference * 10000000),
|
|
(int32_t)(a2dp_audio_context.average_packet_mut + 0.5f),
|
|
a2dp_audio_context.dest_packet_mut);
|
|
#endif
|
|
} else {
|
|
TRACE_A2DP_DECODER_I("[SYNC] tune ratio normal busy");
|
|
}
|
|
|
|
} else {
|
|
#ifdef A2DP_AUDIO_SYNC_FIX_DIFF_INTERVA_PRINT_FLOAT
|
|
TRACE_A2DP_DECODER_I("[SYNC] tune ratio busy %f mut:%5.3f->%d",
|
|
(double)a2dp_audio_context.init_factor_reference,
|
|
(double)a2dp_audio_context.average_packet_mut,
|
|
a2dp_audio_context.dest_packet_mut);
|
|
#else
|
|
TRACE_A2DP_DECODER_I(
|
|
"[SYNC] tune ratio busy %d mut:%d->%d",
|
|
(int32_t)(a2dp_audio_context.init_factor_reference * 10000000),
|
|
(int32_t)(a2dp_audio_context.average_packet_mut + 0.5f),
|
|
a2dp_audio_context.dest_packet_mut);
|
|
#endif
|
|
}
|
|
goto exit;
|
|
}
|
|
|
|
if (audio_sync->tick % A2DP_AUDIO_SYNC_INTERVAL == 0) {
|
|
diff_mtu = a2dp_audio_context.average_packet_mut -
|
|
(float)a2dp_audio_context.dest_packet_mut;
|
|
if (ABS(diff_mtu) < 0.6f) {
|
|
#ifdef A2DP_AUDIO_SYNC_FIX_DIFF_INTERVA_PRINT_FLOAT
|
|
TRACE_A2DP_DECODER_I("[SYNC] skip mut:%5.3f", (double)diff_mtu);
|
|
#else
|
|
TRACE_A2DP_DECODER_I("[SYNC] skip mut:0.%d", (int32_t)(diff_mtu * 10));
|
|
#endif
|
|
goto exit;
|
|
} else if (diff_mtu / a2dp_audio_context.dest_packet_mut < -0.2f) {
|
|
float curr_ratio = a2dp_audio_context.output_cfg.factor_reference;
|
|
float ref_ratio = a2dp_audio_context.init_factor_reference;
|
|
if (curr_ratio != (ref_ratio + A2DP_AUDIO_SYNC_FIX_DIFF_SLOW_LIMIT)) {
|
|
#ifdef A2DP_AUDIO_SYNC_FIX_DIFF_INTERVA_PRINT_FLOAT
|
|
TRACE_A2DP_DECODER_I("[SYNC] force slow mut:%5.3f", (double)diff_mtu);
|
|
#else
|
|
TRACE_A2DP_DECODER_I("[SYNC] force slow mut:0.%d",
|
|
(int32_t)(diff_mtu * 10));
|
|
#endif
|
|
force_slow = true;
|
|
}
|
|
}
|
|
if (a2dp_audio_sync_fix_diff_status_get() ==
|
|
A2DP_AUDIO_SYNC_FIX_DIFF_STATUS_IDLE ||
|
|
force_slow) {
|
|
uint32_t dma_buffer_samples, samples, dma_interval;
|
|
double ratio = 1.0, limit_ratio = 1.0;
|
|
float sampleRate, ref_ratio, curr_ratio;
|
|
float ref_us = 0;
|
|
float dest_us = 0;
|
|
float sample_us = 0;
|
|
|
|
dma_buffer_samples = app_bt_stream_get_dma_buffer_samples() / 2;
|
|
dma_interval = A2DP_AUDIO_SYNC_FIX_DIFF_INTERVAL;
|
|
|
|
ref_ratio = a2dp_audio_context.init_factor_reference;
|
|
curr_ratio = a2dp_audio_context.output_cfg.factor_reference;
|
|
|
|
samples = dma_interval * dma_buffer_samples;
|
|
sampleRate = lastframe_info->stream_info.sample_rate * ref_ratio;
|
|
|
|
sample_us = 1e6 / sampleRate;
|
|
ref_us = sample_us * (float)samples;
|
|
dest_us = sample_us * (float)(lastframe_info->frame_samples * diff_mtu);
|
|
ratio = ref_us / (ref_us - dest_us) * curr_ratio;
|
|
|
|
if (ratio > (double)(ref_ratio + A2DP_AUDIO_SYNC_FIX_DIFF_FAST_LIMIT)) {
|
|
limit_ratio = ref_ratio + A2DP_AUDIO_SYNC_FIX_DIFF_FAST_LIMIT;
|
|
} else if (ratio <
|
|
double(ref_ratio + A2DP_AUDIO_SYNC_FIX_DIFF_SLOW_LIMIT)) {
|
|
limit_ratio = ref_ratio + A2DP_AUDIO_SYNC_FIX_DIFF_SLOW_LIMIT;
|
|
} else {
|
|
limit_ratio = ratio;
|
|
}
|
|
TRACE_A2DP_DECODER_I(
|
|
"[SYNC] sampleRate:%d ref_ratio:%d samples:%d %d->%d",
|
|
(int32_t)sampleRate, (int32_t)(ref_ratio * 10000000), samples,
|
|
(int32_t)ref_us, (int32_t)dest_us);
|
|
need_tune = true;
|
|
#if defined(IBRT)
|
|
if (!app_tws_ibrt_audio_sync_tune_onprocess() &&
|
|
!a2dp_audio_sync_tune_onprocess() &&
|
|
#else
|
|
if (!a2dp_audio_sync_tune_onprocess() &&
|
|
#endif
|
|
need_tune) {
|
|
if (!a2dp_audio_sync_tune((float)limit_ratio)) {
|
|
a2dp_audio_sync_fix_diff_start(dma_interval);
|
|
#ifdef A2DP_AUDIO_SYNC_FIX_DIFF_INTERVA_PRINT_FLOAT
|
|
TRACE_A2DP_DECODER_I("[SYNC] tune ratio %f mut:%5.3f->%d",
|
|
limit_ratio,
|
|
(double)a2dp_audio_context.average_packet_mut,
|
|
a2dp_audio_context.dest_packet_mut);
|
|
#else
|
|
TRACE_A2DP_DECODER_I(
|
|
"[SYNC] tune ratio %d mut:%d->%d",
|
|
(int32_t)(limit_ratio * 10000000),
|
|
(int32_t)(a2dp_audio_context.average_packet_mut + 0.5f),
|
|
a2dp_audio_context.dest_packet_mut);
|
|
#endif
|
|
} else {
|
|
TRACE_A2DP_DECODER_I("[SYNC] tune ratio busy");
|
|
}
|
|
}
|
|
} else {
|
|
TRACE_A2DP_DECODER_I(
|
|
"[SYNC] avg_mut:%d dest_mtu:%d",
|
|
(int32_t)(a2dp_audio_context.average_packet_mut + 0.5f),
|
|
a2dp_audio_context.dest_packet_mut);
|
|
}
|
|
} else {
|
|
}
|
|
exit:
|
|
audio_sync->tick++;
|
|
return 0;
|
|
}
|
|
#else
|
|
int a2dp_audio_sync_handler(uint8_t *buffer, uint32_t buffer_bytes) {
|
|
A2DP_AUDIO_LASTFRAME_INFO_T *lastframe_info = NULL;
|
|
A2DP_AUDIO_SYNC_T *audio_sync = &a2dp_audio_context.audio_sync;
|
|
float dest_pid_result = .0f;
|
|
float diff_mtu = 0;
|
|
int32_t frame_mtu = 0;
|
|
int32_t total_mtu = 0;
|
|
float diff_factor = 0;
|
|
|
|
#if defined(IBRT)
|
|
if (!app_tws_ibrt_mobile_link_connected()) {
|
|
return -1;
|
|
}
|
|
#endif
|
|
if (a2dp_audio_internal_lastframe_info_ptr_get(&lastframe_info) < 0) {
|
|
return -1;
|
|
}
|
|
|
|
if (audio_sync->tick++ % A2DP_AUDIO_SYNC_INTERVAL == 0) {
|
|
list_t *list = a2dp_audio_context.audio_datapath.input_raw_packet_list;
|
|
A2DP_AUDIO_SYNC_PID_T *pid = &audio_sync->pid;
|
|
// valid limter 0x80000
|
|
if (audio_sync->cnt < 0x80000) {
|
|
audio_sync->cnt += A2DP_AUDIO_SYNC_INTERVAL;
|
|
}
|
|
frame_mtu = lastframe_info->stream_info.frame_samples /
|
|
lastframe_info->frame_samples;
|
|
total_mtu = audio_sync->cnt * frame_mtu;
|
|
diff_mtu = a2dp_audio_context.average_packet_mut -
|
|
(float)a2dp_audio_context.dest_packet_mut;
|
|
#if 1
|
|
TRACE_A2DP_DECODER_I(
|
|
"[SYNC] sample:%d/%d diff:%d/%d/%d/%d curr:%d",
|
|
lastframe_info->frame_samples,
|
|
lastframe_info->stream_info.frame_samples, (int32_t)(diff_mtu + 0.5f),
|
|
(int32_t)(a2dp_audio_context.average_packet_mut + 0.5f),
|
|
a2dp_audio_context.dest_packet_mut, total_mtu,
|
|
a2dp_audio_list_length(list) + get_in_cp_frame_cnt());
|
|
#else
|
|
TRACE_A2DP_DECODER_I(
|
|
"[SYNC] diff:%10.9f/%10.9f frame_mut:%d dest:%d total:%d curr:%d",
|
|
(double)diff_mtu, (double)a2dp_audio_context.average_packet_mut,
|
|
frame_mtu, a2dp_audio_context.dest_packet_mut, total_mtu,
|
|
a2dp_audio_list_length(list));
|
|
TRACE_A2DP_DECODER_I("[SYNC] try tune:%d, %10.9f %10.9f",
|
|
diff_mtu != 0.f && audio_sync->tick != 1
|
|
? ABS(diff_mtu) > ((float)frame_mtu * 0.2f)
|
|
: 0,
|
|
(double)ABS(diff_mtu),
|
|
(double)((float)frame_mtu * 0.2f));
|
|
#endif
|
|
// TRACE(2,"audio_sync tune %d/%d tick",
|
|
// app_tws_ibrt_audio_sync_tune_onprocess(),
|
|
// a2dp_audio_sync_tune_onprocess(), audio_sync->tick);
|
|
|
|
#if defined(IBRT)
|
|
if ((!app_tws_ibrt_audio_sync_tune_onprocess() &&
|
|
!a2dp_audio_sync_tune_onprocess()) &&
|
|
#else
|
|
if (!a2dp_audio_sync_tune_onprocess() &&
|
|
#endif
|
|
diff_mtu != 0.f && audio_sync->tick != 1 &&
|
|
((ABS(diff_mtu) > ((float)frame_mtu * 0.25f) && diff_mtu > 0) ||
|
|
(ABS(diff_mtu) > ((float)frame_mtu * 0.1f) && diff_mtu < 0))) {
|
|
diff_factor = diff_mtu / a2dp_audio_context.average_packet_mut;
|
|
if (a2dp_audio_sync_pid_calc(pid, diff_factor)) {
|
|
dest_pid_result =
|
|
a2dp_audio_context.output_cfg.factor_reference + pid->result;
|
|
|
|
if (dest_pid_result > (A2DP_AUDIO_SYNC_FACTOR_REFERENCE +
|
|
A2DP_AUDIO_SYNC_FACTOR_FAST_LIMIT)) {
|
|
dest_pid_result = A2DP_AUDIO_SYNC_FACTOR_REFERENCE +
|
|
A2DP_AUDIO_SYNC_FACTOR_FAST_LIMIT;
|
|
} else if (dest_pid_result < (A2DP_AUDIO_SYNC_FACTOR_REFERENCE +
|
|
A2DP_AUDIO_SYNC_FACTOR_SLOW_LIMIT)) {
|
|
dest_pid_result = A2DP_AUDIO_SYNC_FACTOR_REFERENCE +
|
|
A2DP_AUDIO_SYNC_FACTOR_SLOW_LIMIT;
|
|
}
|
|
|
|
if (a2dp_audio_context.output_cfg.factor_reference != dest_pid_result) {
|
|
if (!a2dp_audio_sync_tune(dest_pid_result)) {
|
|
audio_sync->cnt = 0;
|
|
}
|
|
TRACE_A2DP_DECODER_I(
|
|
"[SYNC] tune diff_factor:%10.9f pid:%10.9f tune:%10.9f",
|
|
(double)diff_factor, (double)pid->result,
|
|
(double)dest_pid_result);
|
|
} else {
|
|
TRACE_A2DP_DECODER_I("[SYNC] tune skip same");
|
|
}
|
|
}
|
|
} else {
|
|
/*
|
|
TRACE_A2DP_DECODER_I("[SYNC] tune busy skip proc:%d/%d mtu:%d
|
|
tick:%d >0:%d <0:%d tick:%d", app_tws_ibrt_audio_sync_tune_onprocess(),
|
|
a2dp_audio_sync_tune_onprocess(),
|
|
diff_mtu != 0.f,
|
|
audio_sync->tick != 1,
|
|
(ABS(diff_mtu) >
|
|
((float)frame_mtu * 0.25f) && diff_mtu > 0), (ABS(diff_mtu) >
|
|
((float)frame_mtu * 0.1f) && diff_mtu < 0), audio_sync->tick);
|
|
*/
|
|
}
|
|
} else {
|
|
bool need_tune = false;
|
|
if (lastframe_info->undecode_min_frames * 10 <=
|
|
a2dp_audio_context.dest_packet_mut * 10 / 3) {
|
|
dest_pid_result = a2dp_audio_context.init_factor_reference +
|
|
A2DP_AUDIO_SYNC_FACTOR_NEED_FAST_CACHE;
|
|
need_tune = true;
|
|
} else if (lastframe_info->undecode_min_frames * 10 <=
|
|
a2dp_audio_context.dest_packet_mut * 20 / 3) {
|
|
dest_pid_result = a2dp_audio_context.init_factor_reference +
|
|
A2DP_AUDIO_SYNC_FACTOR_SLOW_LIMIT;
|
|
need_tune = true;
|
|
}
|
|
#if defined(IBRT)
|
|
if (!app_tws_ibrt_audio_sync_tune_onprocess() &&
|
|
!a2dp_audio_sync_tune_onprocess() &&
|
|
#else
|
|
if (!a2dp_audio_sync_tune_onprocess() &&
|
|
#endif
|
|
|
|
need_tune) {
|
|
if (a2dp_audio_context.output_cfg.factor_reference != dest_pid_result) {
|
|
a2dp_audio_sync_reset_data();
|
|
a2dp_audio_sync_tune(dest_pid_result);
|
|
TRACE_A2DP_DECODER_I("[SYNC] tune ratio force slow %d/%d->%d",
|
|
lastframe_info->undecode_min_frames,
|
|
lastframe_info->undecode_max_frames,
|
|
a2dp_audio_context.dest_packet_mut);
|
|
}
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#if A2DP_DECODER_HISTORY_SEQ_SAVE
|
|
static int a2dp_audio_reset_history_seq(void) {
|
|
a2dp_audio_status_mutex_lock();
|
|
for (uint8_t i = 0; i < A2DP_DECODER_HISTORY_SEQ_SAVE; i++) {
|
|
a2dp_audio_context.historySeq[i] = 0;
|
|
#ifdef A2DP_DECODER_HISTORY_LOCTIME_SAVE
|
|
a2dp_audio_context.historyLoctime[i] = 0;
|
|
#endif
|
|
#ifdef A2DP_DECODER_HISTORY_CHECK_SUM_SAVE
|
|
a2dp_audio_context.historyChecksum[i] = 0;
|
|
#endif
|
|
}
|
|
a2dp_audio_context.historySeq_idx = 0;
|
|
a2dp_audio_status_mutex_unlock();
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int a2dp_audio_save_history_seq(btif_media_header_t *header,
|
|
unsigned char *buf, unsigned int len) {
|
|
uint16_t historySeqPre = 0;
|
|
uint8_t historySeq_idx = 0;
|
|
|
|
a2dp_audio_status_mutex_lock();
|
|
historySeq_idx = a2dp_audio_context.historySeq_idx;
|
|
if (historySeq_idx) {
|
|
historySeq_idx = (historySeq_idx - 1) % A2DP_DECODER_HISTORY_SEQ_SAVE;
|
|
historySeqPre = a2dp_audio_context.historySeq[historySeq_idx];
|
|
if ((header->sequenceNumber - historySeqPre) != 1) {
|
|
TRACE_A2DP_DECODER_W("[INPUT] SEQ ERR %d/%d", historySeqPre,
|
|
header->sequenceNumber);
|
|
a2dp_audio_show_history_seq();
|
|
}
|
|
}
|
|
historySeq_idx =
|
|
a2dp_audio_context.historySeq_idx % A2DP_DECODER_HISTORY_SEQ_SAVE;
|
|
a2dp_audio_context.historySeq[historySeq_idx] = header->sequenceNumber;
|
|
#ifdef A2DP_DECODER_HISTORY_LOCTIME_SAVE
|
|
a2dp_audio_context.historyLoctime[historySeq_idx] = hal_fast_sys_timer_get();
|
|
#endif
|
|
#ifdef A2DP_DECODER_HISTORY_CHECK_SUM_SAVE
|
|
a2dp_audio_context.historyChecksum[historySeq_idx] = crc32(0, buf, len);
|
|
#endif
|
|
a2dp_audio_context.historySeq_idx++;
|
|
a2dp_audio_status_mutex_unlock();
|
|
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_show_history_seq(void) {
|
|
uint8_t i = 0, j = 1;
|
|
uint16_t reordHistorySeq[A2DP_DECODER_HISTORY_SEQ_SAVE];
|
|
#ifdef A2DP_DECODER_HISTORY_LOCTIME_SAVE
|
|
int32_t diff_max_idx = 0;
|
|
int32_t diff_max_ms = 0;
|
|
int64_t diff_avg_ms = 0;
|
|
int32_t diff_avg_cnt = 0;
|
|
uint32_t historyLoctime[A2DP_DECODER_HISTORY_SEQ_SAVE];
|
|
#endif
|
|
#ifdef A2DP_DECODER_HISTORY_CHECK_SUM_SAVE
|
|
uint32_t historyChecksum[A2DP_DECODER_HISTORY_SEQ_SAVE];
|
|
#endif
|
|
|
|
a2dp_audio_status_mutex_lock();
|
|
for (i = 0; i < A2DP_DECODER_HISTORY_SEQ_SAVE; i++) {
|
|
reordHistorySeq[i] = a2dp_audio_context.historySeq[i];
|
|
#ifdef A2DP_DECODER_HISTORY_LOCTIME_SAVE
|
|
historyLoctime[i] = a2dp_audio_context.historyLoctime[i];
|
|
#endif
|
|
#ifdef A2DP_DECODER_HISTORY_CHECK_SUM_SAVE
|
|
historyChecksum[i] = a2dp_audio_context.historyChecksum[i];
|
|
#endif
|
|
}
|
|
|
|
for (i = 0; i < A2DP_DECODER_HISTORY_SEQ_SAVE - 1; i++) {
|
|
for (j = 0; j < A2DP_DECODER_HISTORY_SEQ_SAVE - 1 - i; j++) {
|
|
if (reordHistorySeq[j] > reordHistorySeq[j + 1]) {
|
|
uint16_t temp_seq = reordHistorySeq[j];
|
|
reordHistorySeq[j] = reordHistorySeq[j + 1];
|
|
reordHistorySeq[j + 1] = temp_seq;
|
|
#ifdef A2DP_DECODER_HISTORY_LOCTIME_SAVE
|
|
uint32_t temp_Loctime = historyLoctime[j];
|
|
historyLoctime[j] = historyLoctime[j + 1];
|
|
historyLoctime[j + 1] = temp_Loctime;
|
|
#endif
|
|
#ifdef A2DP_DECODER_HISTORY_CHECK_SUM_SAVE
|
|
uint32_t temp_Checksum = historyChecksum[j];
|
|
historyChecksum[j] = historyChecksum[j + 1];
|
|
historyChecksum[j + 1] = temp_Checksum;
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
#ifdef A2DP_DECODER_HISTORY_LOCTIME_SAVE
|
|
for (i = 0, j = 1; i < A2DP_DECODER_HISTORY_SEQ_SAVE - 1; i++, j++) {
|
|
int32_t tmp_ms = historyLoctime[j] - historyLoctime[i];
|
|
diff_avg_ms += tmp_ms;
|
|
diff_avg_cnt++;
|
|
if (tmp_ms > diff_max_ms) {
|
|
diff_max_ms = tmp_ms;
|
|
diff_max_idx = i;
|
|
}
|
|
if (tmp_ms > (int32_t)MS_TO_FAST_TICKS(50)) {
|
|
TRACE_A2DP_DECODER_I("[INPUT] > 30ms seq:%d diff:%d :%d / %d",
|
|
reordHistorySeq[i], FAST_TICKS_TO_MS(tmp_ms),
|
|
historyLoctime[i], historyLoctime[j]);
|
|
}
|
|
}
|
|
diff_avg_ms /= diff_avg_cnt;
|
|
TRACE_A2DP_DECODER_I("[INPUT] max_diff:%dms idx:%d avg:%dus",
|
|
FAST_TICKS_TO_MS(diff_max_ms), diff_max_idx,
|
|
FAST_TICKS_TO_US(diff_avg_ms));
|
|
#endif
|
|
// DUMP16("%d ", reordHistorySeq, A2DP_DECODER_HISTORY_SEQ_SAVE);
|
|
#ifdef A2DP_DECODER_HISTORY_LOCTIME_SAVE
|
|
DUMP32("%x ", historyLoctime, A2DP_DECODER_HISTORY_SEQ_SAVE);
|
|
#endif
|
|
#ifdef A2DP_DECODER_HISTORY_CHECK_SUM_SAVE
|
|
DUMP32("%x ", historyChecksum, A2DP_DECODER_HISTORY_SEQ_SAVE);
|
|
#endif
|
|
a2dp_audio_status_mutex_unlock();
|
|
|
|
return 0;
|
|
}
|
|
|
|
#endif
|
|
|
|
/*
|
|
1, 2^1
|
|
3, 2^2
|
|
7, 2^3
|
|
15, 2^4
|
|
31, 2^5
|
|
*/
|
|
#define AUDIO_ALPHA_PRAMS_1 (3)
|
|
#define AUDIO_ALPHA_PRAMS_2 (4)
|
|
|
|
static inline float a2dp_audio_alpha_filter(float y, float x) {
|
|
if (y) {
|
|
y = ((AUDIO_ALPHA_PRAMS_1 * y) + x) / AUDIO_ALPHA_PRAMS_2;
|
|
} else {
|
|
y = x;
|
|
}
|
|
return y;
|
|
}
|
|
|
|
static void inline a2dp_audio_convert_16bit_to_24bit(int32_t *out, int16_t *in,
|
|
int len) {
|
|
for (int i = len - 1; i >= 0; i--) {
|
|
out[i] = ((int32_t)in[i] << 8);
|
|
}
|
|
}
|
|
|
|
static void inline a2dp_audio_channel_select(
|
|
A2DP_AUDIO_CHANNEL_SELECT_E chnl_sel, uint8_t *buffer,
|
|
uint32_t buffer_bytes) {
|
|
uint32_t samples;
|
|
uint32_t i;
|
|
|
|
ASSERT_A2DP_DECODER(a2dp_audio_context.output_cfg.num_channels == 2,
|
|
"%s num_channels:%d", __func__,
|
|
a2dp_audio_context.output_cfg.num_channels);
|
|
|
|
if (a2dp_audio_context.output_cfg.bits_depth == 24) {
|
|
int32_t *buf_l_p = (int32_t *)buffer;
|
|
int32_t *buf_r_p = (int32_t *)buffer + 1;
|
|
|
|
samples = buffer_bytes / 4 / 2;
|
|
switch (chnl_sel) {
|
|
case A2DP_AUDIO_CHANNEL_SELECT_LRMERGE:
|
|
for (i = 0; i < samples; i++, buf_l_p += 2, buf_r_p += 2) {
|
|
int32_t tmp_sample = (*buf_l_p + *buf_r_p) >> 1;
|
|
*buf_l_p = tmp_sample;
|
|
*buf_r_p = tmp_sample;
|
|
}
|
|
break;
|
|
case A2DP_AUDIO_CHANNEL_SELECT_LCHNL:
|
|
for (i = 0; i < samples; i++, buf_l_p += 2, buf_r_p += 2) {
|
|
*buf_r_p = *buf_l_p;
|
|
}
|
|
break;
|
|
case A2DP_AUDIO_CHANNEL_SELECT_RCHNL:
|
|
for (i = 0; i < samples; i++, buf_l_p += 2, buf_r_p += 2) {
|
|
*buf_l_p = *buf_r_p;
|
|
}
|
|
break;
|
|
case A2DP_AUDIO_CHANNEL_SELECT_STEREO:
|
|
default:
|
|
break;
|
|
}
|
|
} else {
|
|
int16_t *buf_l_p = (int16_t *)buffer;
|
|
int16_t *buf_r_p = (int16_t *)buffer + 1;
|
|
|
|
samples = buffer_bytes / 2 / 2;
|
|
switch (chnl_sel) {
|
|
case A2DP_AUDIO_CHANNEL_SELECT_LRMERGE:
|
|
for (i = 0; i < samples; i++, buf_l_p += 2, buf_r_p += 2) {
|
|
int16_t tmp_sample = ((int32_t)*buf_l_p + (int32_t)*buf_r_p) >> 1;
|
|
*buf_l_p = tmp_sample;
|
|
*buf_r_p = tmp_sample;
|
|
}
|
|
break;
|
|
case A2DP_AUDIO_CHANNEL_SELECT_LCHNL:
|
|
for (i = 0; i < samples; i++, buf_l_p += 2, buf_r_p += 2) {
|
|
*buf_r_p = *buf_l_p;
|
|
}
|
|
break;
|
|
case A2DP_AUDIO_CHANNEL_SELECT_RCHNL:
|
|
for (i = 0; i < samples; i++, buf_l_p += 2, buf_r_p += 2) {
|
|
*buf_l_p = *buf_r_p;
|
|
}
|
|
break;
|
|
case A2DP_AUDIO_CHANNEL_SELECT_STEREO:
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef A2DP_CP_ACCEL
|
|
extern "C" bool is_cp_need_reset(void);
|
|
extern uint32_t app_bt_stream_get_dma_buffer_samples(void);
|
|
|
|
static uint32_t get_cp_frame_mtus(A2DP_AUDIO_LASTFRAME_INFO_T *info) {
|
|
uint32_t cp_frame_mtus = app_bt_stream_get_dma_buffer_samples() / 2;
|
|
if (cp_frame_mtus % (info->frame_samples)) {
|
|
cp_frame_mtus = cp_frame_mtus / (info->frame_samples) + 1;
|
|
} else {
|
|
cp_frame_mtus = cp_frame_mtus / (info->frame_samples);
|
|
}
|
|
|
|
TRACE_A2DP_DECODER_I("[CP] cp_frame_mtus:%d", cp_frame_mtus);
|
|
|
|
return cp_frame_mtus;
|
|
}
|
|
|
|
#else
|
|
bool is_cp_need_reset(void) { return false; }
|
|
|
|
#endif
|
|
|
|
#define A2DP_AUDIO_SYSFREQ_BOOST_RESUME_CNT (20)
|
|
uint32_t a2dp_audio_sysfreq_cnt = UINT32_MAX;
|
|
uint32_t a2dp_audio_sysfreq_dest_boost_cnt = 0;
|
|
APP_SYSFREQ_FREQ_T a2dp_audio_sysfreq_normalfreq = APP_SYSFREQ_52M;
|
|
|
|
int a2dp_audio_sysfreq_boost_init(uint32_t normalfreq) {
|
|
a2dp_audio_sysfreq_cnt = UINT32_MAX;
|
|
a2dp_audio_sysfreq_dest_boost_cnt = 0;
|
|
a2dp_audio_sysfreq_normalfreq = (APP_SYSFREQ_FREQ_T)normalfreq;
|
|
TRACE_A2DP_DECODER_I("[BOOST] freq:%d", normalfreq);
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_sysfreq_boost_start(uint32_t boost_cnt) {
|
|
enum APP_SYSFREQ_FREQ_T sysfreq = APP_SYSFREQ_104M;
|
|
|
|
a2dp_audio_sysfreq_cnt = 0;
|
|
a2dp_audio_sysfreq_dest_boost_cnt = boost_cnt;
|
|
if (a2dp_audio_sysfreq_normalfreq >= APP_SYSFREQ_52M) {
|
|
sysfreq = APP_SYSFREQ_104M;
|
|
} else {
|
|
sysfreq = APP_SYSFREQ_52M;
|
|
}
|
|
TRACE_A2DP_DECODER_I("[BOOST] freq:%d cnt:%d", sysfreq, boost_cnt);
|
|
app_sysfreq_req(APP_SYSFREQ_USER_BT_A2DP, sysfreq);
|
|
return 0;
|
|
}
|
|
|
|
static int a2dp_audio_sysfreq_boost_porc(void) {
|
|
if (a2dp_audio_sysfreq_cnt == UINT32_MAX) {
|
|
// do nothing
|
|
} else if (a2dp_audio_sysfreq_cnt >= a2dp_audio_sysfreq_dest_boost_cnt) {
|
|
a2dp_audio_sysfreq_cnt = UINT32_MAX;
|
|
TRACE_A2DP_DECODER_I("[BOOST] freq:%d", a2dp_audio_sysfreq_normalfreq);
|
|
app_sysfreq_req(APP_SYSFREQ_USER_BT_A2DP, a2dp_audio_sysfreq_normalfreq);
|
|
} else {
|
|
a2dp_audio_sysfreq_cnt++;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_sysfreq_boost_running(void) {
|
|
return a2dp_audio_sysfreq_cnt == UINT32_MAX ? 0 : 1;
|
|
}
|
|
|
|
int a2dp_audio_store_packet_checker_start(void) {
|
|
store_packet_history_loctime = 0;
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_store_packet_checker(btif_media_header_t *header) {
|
|
bool show_info = false;
|
|
uint32_t fast_sys_tick = hal_fast_sys_timer_get();
|
|
int32_t tmp_ms = 0;
|
|
|
|
if (store_packet_history_loctime) {
|
|
tmp_ms = fast_sys_tick - store_packet_history_loctime;
|
|
if (tmp_ms > (int32_t)MS_TO_FAST_TICKS(50)) {
|
|
show_info = true;
|
|
}
|
|
}
|
|
|
|
if (show_info) {
|
|
// TRACE_A2DP_DECODER_I("[INPUT] >50ms seq:%d diff:%d",
|
|
// header->sequenceNumber, FAST_TICKS_TO_MS(tmp_ms));
|
|
a2dp_decoder_info_checker();
|
|
}
|
|
|
|
store_packet_history_loctime = fast_sys_tick;
|
|
|
|
if (a2dp_audio_context.historySeq_idx &&
|
|
((a2dp_audio_context.historySeq_idx - 1) %
|
|
A2DP_DECODER_HISTORY_SEQ_SAVE ==
|
|
0)) {
|
|
a2dp_audio_show_history_seq();
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
//#define DECODER_ERR_TEST
|
|
int a2dp_audio_store_packet(btif_media_header_t *header, unsigned char *buf,
|
|
unsigned int len) {
|
|
int nRet = A2DP_DECODER_NO_ERROR;
|
|
|
|
if (a2dp_audio_get_status() ==
|
|
A2DP_AUDIO_DECODER_STATUS_NULL) { // if mem deinit , drop data
|
|
return nRet;
|
|
}
|
|
a2dp_audio_status_mutex_lock();
|
|
|
|
#ifdef DECODER_ERR_TEST
|
|
static uint16_t cnt = 0;
|
|
static uint16_t limit = 500;
|
|
cnt++;
|
|
if (cnt % limit == 0) {
|
|
cnt = 0;
|
|
limit = rand() % 500;
|
|
uint16_t len2 = rand() % len;
|
|
memset(buf + len - len2, 0, len2);
|
|
TRACE_A2DP_DECODER_I("[INPUT] Fill Err!!!! seq:%d", header->sequenceNumber);
|
|
}
|
|
#endif
|
|
#if A2DP_DECODER_HISTORY_SEQ_SAVE
|
|
a2dp_audio_save_history_seq(header, buf, len);
|
|
#endif
|
|
a2dp_audio_set_store_packet_status(
|
|
A2DP_AUDIO_DECODER_STORE_PACKET_STATUS_BUSY);
|
|
if (a2dp_audio_get_status() == A2DP_AUDIO_DECODER_STATUS_START) {
|
|
a2dp_audio_store_packet_checker(header);
|
|
if (a2dp_audio_context.need_detect_first_packet) {
|
|
a2dp_audio_context.need_detect_first_packet = false;
|
|
a2dp_audio_context.audio_decoder.audio_decoder_preparse_packet(header,
|
|
buf, len);
|
|
}
|
|
|
|
if (a2dp_audio_detect_next_packet_callback) {
|
|
a2dp_audio_detect_next_packet_callback(header, buf, len);
|
|
}
|
|
|
|
nRet = a2dp_audio_context.audio_decoder.audio_decoder_store_packet(
|
|
header, buf, len);
|
|
#if defined(IBRT)
|
|
if (is_cp_need_reset()) {
|
|
TRACE_A2DP_DECODER_I("[CP] find cp error need restart");
|
|
app_ibrt_if_force_audio_retrigger();
|
|
}
|
|
if (nRet == A2DP_DECODER_MTU_LIMTER_ERROR) {
|
|
if (app_tws_ibrt_mobile_link_connected()) {
|
|
// try again
|
|
// a2dp_audio_semaphore_wait(A2DP_AUDIO_WAIT_TIMEOUT_MS);
|
|
nRet = a2dp_audio_context.audio_decoder.audio_decoder_store_packet(
|
|
header, buf, len);
|
|
}
|
|
if (nRet == A2DP_DECODER_MTU_LIMTER_ERROR) {
|
|
int dest_discards_samples = 0;
|
|
ibrt_ctrl_t *p_ibrt_ctrl = app_tws_ibrt_get_bt_ctrl_ctx();
|
|
if (p_ibrt_ctrl == NULL) {
|
|
return A2DP_DECODER_NOT_SUPPORT;
|
|
}
|
|
if (app_tws_ibrt_mobile_link_connected()) {
|
|
bt_syn_trig_checker(p_ibrt_ctrl->mobile_conhandle);
|
|
} else if (app_tws_ibrt_slave_ibrt_link_connected()) {
|
|
bt_syn_trig_checker(p_ibrt_ctrl->ibrt_conhandle);
|
|
}
|
|
dest_discards_samples = app_bt_stream_get_dma_buffer_samples() / 2;
|
|
a2dp_audio_discards_samples(dest_discards_samples * 2);
|
|
a2dp_audio_context.audio_decoder.audio_decoder_store_packet(header, buf,
|
|
len);
|
|
TRACE_A2DP_DECODER_I("[INPUT] MTU_LIMTER so discards_packet");
|
|
}
|
|
}
|
|
#else
|
|
if (is_cp_need_reset()) {
|
|
TRACE_A2DP_DECODER_I("[CP] find cp error need restart");
|
|
app_audio_decode_err_force_trigger();
|
|
}
|
|
|
|
if (nRet == A2DP_DECODER_MTU_LIMTER_ERROR) {
|
|
a2dp_audio_synchronize_dest_packet_mut(
|
|
a2dp_audio_context.dest_packet_mut);
|
|
a2dp_audio_context.audio_decoder.audio_decoder_store_packet(header, buf,
|
|
len);
|
|
TRACE_A2DP_DECODER_W("[INPUT] MTU_LIMTER so discards_packet");
|
|
}
|
|
#endif
|
|
} else {
|
|
TRACE(2, "[INPUT] skip packet status:%d", a2dp_audio_get_status());
|
|
}
|
|
|
|
if (a2dp_audio_store_packet_callback) {
|
|
a2dp_audio_store_packet_callback(header, buf, len);
|
|
}
|
|
|
|
a2dp_audio_set_store_packet_status(
|
|
A2DP_AUDIO_DECODER_STORE_PACKET_STATUS_IDLE);
|
|
a2dp_audio_status_mutex_unlock();
|
|
return 0;
|
|
}
|
|
|
|
uint32_t a2dp_audio_playback_handler(uint8_t *buffer, uint32_t buffer_bytes) {
|
|
uint32_t len = buffer_bytes;
|
|
int nRet = A2DP_DECODER_NO_ERROR;
|
|
A2DP_AUDIO_LASTFRAME_INFO_T *lastframe_info = NULL;
|
|
list_t *list = a2dp_audio_context.audio_datapath.input_raw_packet_list;
|
|
|
|
a2dp_audio_set_playback_status(A2DP_AUDIO_DECODER_PLAYBACK_STATUS_BUSY);
|
|
if (a2dp_audio_get_status() != A2DP_AUDIO_DECODER_STATUS_START) {
|
|
TRACE_A2DP_DECODER_W("[PLAYBACK] skip handler status:%d",
|
|
a2dp_audio_get_status());
|
|
goto exit;
|
|
}
|
|
|
|
a2dp_audio_sysfreq_boost_porc();
|
|
if (a2dp_audio_context.average_packet_mut == 0) {
|
|
A2DP_AUDIO_HEADFRAME_INFO_T headframe_info;
|
|
a2dp_audio_decoder_headframe_info_get(&headframe_info);
|
|
a2dp_audio_context.average_packet_mut = a2dp_audio_list_length(list);
|
|
TRACE_A2DP_DECODER_I(
|
|
"[PLAYBACK] init average_packet_mut:%d seq:%d",
|
|
(uint16_t)(a2dp_audio_context.average_packet_mut + 0.5f),
|
|
headframe_info.sequenceNumber);
|
|
} else {
|
|
if (!a2dp_audio_refill_packet()) {
|
|
uint16_t packet_mut = 0;
|
|
if (!a2dp_audio_internal_lastframe_info_ptr_get(&lastframe_info)) {
|
|
packet_mut = a2dp_audio_list_length(list) + get_in_cp_frame_cnt() +
|
|
get_in_cp_frame_delay() * (lastframe_info->frame_samples /
|
|
lastframe_info->list_samples);
|
|
a2dp_audio_context.average_packet_mut = a2dp_audio_alpha_filter(
|
|
(float)a2dp_audio_context.average_packet_mut, (float)packet_mut);
|
|
a2dp_audio_sync_handler(buffer, buffer_bytes);
|
|
}
|
|
}
|
|
}
|
|
#if defined(A2DP_AUDIO_REFILL_AFTER_NO_CACHE)
|
|
if (a2dp_audio_context.skip_frame_cnt_after_no_cache) {
|
|
#if defined(A2DP_CP_ACCEL)
|
|
uint32_t cp_delay_mtus = get_in_cp_frame_delay();
|
|
cp_delay_mtus *= get_cp_frame_mtus(&a2dp_audio_lastframe_info);
|
|
if (a2dp_audio_list_length(list) >=
|
|
(a2dp_audio_context.dest_packet_mut - cp_delay_mtus)) {
|
|
a2dp_audio_context.skip_frame_cnt_after_no_cache = 0;
|
|
}
|
|
#else
|
|
if (a2dp_audio_list_length(list) >= a2dp_audio_context.dest_packet_mut) {
|
|
a2dp_audio_context.skip_frame_cnt_after_no_cache = 0;
|
|
}
|
|
#endif
|
|
|
|
memset(buffer, 0, buffer_bytes);
|
|
TRACE_A2DP_DECODER_I("[PLAYBACK] decode refill skip_cnt:%d, list:%d",
|
|
a2dp_audio_context.skip_frame_cnt_after_no_cache,
|
|
a2dp_audio_list_length(list));
|
|
a2dp_decoder_info_checker();
|
|
if (a2dp_audio_context.skip_frame_cnt_after_no_cache > 0) {
|
|
a2dp_audio_context.skip_frame_cnt_after_no_cache--;
|
|
} else {
|
|
a2dp_audio_context.mute_frame_cnt_after_no_cache =
|
|
A2DP_AUDIO_MUTE_FRAME_CNT_AFTER_NO_CACHE;
|
|
}
|
|
} else
|
|
#endif
|
|
{
|
|
if (a2dp_audio_context.output_cfg.bits_depth == 24 &&
|
|
16 == bt_sbc_player_get_sample_bit()) {
|
|
|
|
len = len / (sizeof(int32_t) / sizeof(int16_t));
|
|
|
|
nRet = a2dp_audio_context.audio_decoder.audio_decoder_decode_frame(buffer,
|
|
len);
|
|
if (nRet < 0 || a2dp_audio_context.mute_frame_cnt_after_no_cache) {
|
|
TRACE_A2DP_DECODER_I("[PLAYBACK] decode failed nRet=%d mute_cnt:%d",
|
|
nRet,
|
|
a2dp_audio_context.mute_frame_cnt_after_no_cache);
|
|
a2dp_decoder_info_checker();
|
|
// mute frame
|
|
memset(buffer, 0, len);
|
|
}
|
|
a2dp_audio_convert_16bit_to_24bit((int32_t *)buffer, (int16_t *)buffer,
|
|
len / sizeof(int16_t));
|
|
|
|
// Restore len to 24-bit sample buffer length
|
|
len = len * (sizeof(int32_t) / sizeof(int16_t));
|
|
|
|
} else if (a2dp_audio_context.output_cfg.bits_depth ==
|
|
a2dp_audio_context.audio_decoder.stream_info.bits_depth) {
|
|
|
|
nRet = a2dp_audio_context.audio_decoder.audio_decoder_decode_frame(buffer,
|
|
len);
|
|
if (nRet < 0 || a2dp_audio_context.mute_frame_cnt_after_no_cache) {
|
|
// mute frame
|
|
TRACE_A2DP_DECODER_I("[PLAYBACK] decode failed nRet=%d mute_cnt:%d",
|
|
nRet,
|
|
a2dp_audio_context.mute_frame_cnt_after_no_cache);
|
|
a2dp_decoder_info_checker();
|
|
memset(buffer, 0, len);
|
|
}
|
|
}
|
|
a2dp_audio_channel_select(a2dp_audio_context.chnl_sel, buffer,
|
|
buffer_bytes);
|
|
}
|
|
a2dp_audio_semaphore_release();
|
|
|
|
if (nRet == A2DP_DECODER_CACHE_UNDERFLOW_ERROR) {
|
|
if (a2dp_audio_internal_lastframe_info_ptr_get(&lastframe_info) < 0) {
|
|
goto exit;
|
|
}
|
|
TRACE(2, "CACHE_UNDERFLOW lastseq:%d ftick:%d",
|
|
lastframe_info->sequenceNumber, hal_fast_sys_timer_get());
|
|
a2dp_audio_show_history_seq();
|
|
uint32_t mute_frames = A2DP_AUDIO_MUTE_FRAME_CNT_AFTER_NO_CACHE;
|
|
uint32_t skip_frames =
|
|
A2DP_AUDIO_SKIP_FRAME_LIMIT_AFTER_NO_CACHE - get_in_cp_frame_delay();
|
|
a2dp_audio_context.mute_frame_cnt_after_no_cache =
|
|
(uint32_t)((float)mute_frames * a2dp_audio_latency_factor_get());
|
|
a2dp_audio_context.skip_frame_cnt_after_no_cache =
|
|
(uint32_t)((float)skip_frames * a2dp_audio_latency_factor_get());
|
|
|
|
a2dp_audio_context.average_packet_mut = 0;
|
|
a2dp_audio_sync_reset_data();
|
|
#if defined(IBRT)
|
|
|
|
a2dp_audio_sync_tune_sample_rate(
|
|
app_tws_ibrt_audio_sync_config_factor_reference_get());
|
|
#else
|
|
a2dp_audio_sync_tune_sample_rate(a2dp_audio_context.init_factor_reference);
|
|
#endif
|
|
a2dp_decoder_info_checker();
|
|
} else {
|
|
if (a2dp_audio_context.mute_frame_cnt_after_no_cache > 0) {
|
|
a2dp_audio_context.mute_frame_cnt_after_no_cache--;
|
|
a2dp_audio_context.average_packet_mut = 0;
|
|
if (a2dp_audio_context.mute_frame_cnt_after_no_cache >= 1) {
|
|
a2dp_audio_synchronize_dest_packet_mut(
|
|
a2dp_audio_context.dest_packet_mut);
|
|
}
|
|
}
|
|
}
|
|
if (!a2dp_audio_internal_lastframe_info_ptr_get(&lastframe_info)) {
|
|
lastframe_info->stream_info.factor_reference =
|
|
a2dp_audio_context.output_cfg.factor_reference;
|
|
lastframe_info->average_frames =
|
|
(uint32_t)(a2dp_audio_context.average_packet_mut + 0.5f);
|
|
}
|
|
exit:
|
|
a2dp_audio_set_playback_status(A2DP_AUDIO_DECODER_PLAYBACK_STATUS_IDLE);
|
|
#if defined(IBRT)
|
|
if (nRet == A2DP_DECODER_CACHE_UNDERFLOW_ERROR) {
|
|
#if defined(A2DP_AUDIO_UNDERFLOW_CAUSE_AUDIO_RETRIGGER)
|
|
bool force_audio_retrigger = true;
|
|
#else
|
|
bool force_audio_retrigger = false;
|
|
#endif
|
|
if (a2dp_audio_latency_factor_get() == A2DP_AUDIO_LATENCY_LOW_FACTOR &&
|
|
app_tws_ibrt_mobile_link_connected()) {
|
|
a2dp_audio_latency_factor_sethigh();
|
|
if (app_tws_ibrt_tws_link_connected() &&
|
|
app_ibrt_ui_is_profile_exchanged()) {
|
|
float latency_factor = a2dp_audio_latency_factor_get();
|
|
tws_ctrl_send_cmd(APP_TWS_CMD_SET_LATENCYFACTOR,
|
|
(uint8_t *)&latency_factor, sizeof(latency_factor));
|
|
force_audio_retrigger = true;
|
|
}
|
|
}
|
|
if (force_audio_retrigger && !a2dp_audio_context.underflow_onporcess) {
|
|
a2dp_audio_context.underflow_onporcess = true;
|
|
app_ibrt_if_force_audio_retrigger();
|
|
}
|
|
}
|
|
#else
|
|
if (nRet == A2DP_DECODER_CACHE_UNDERFLOW_ERROR) {
|
|
app_audio_decode_err_force_trigger();
|
|
}
|
|
#endif
|
|
return len;
|
|
}
|
|
|
|
static void a2dp_audio_packet_free(void *packet) {
|
|
if (a2dp_audio_context.audio_decoder.audio_decoder_packet_free) {
|
|
a2dp_audio_context.audio_decoder.audio_decoder_packet_free(packet);
|
|
} else {
|
|
a2dp_audio_heap_free(packet);
|
|
}
|
|
}
|
|
|
|
void a2dp_audio_clear_input_raw_packet_list(void) {
|
|
// just clean the packet list to start receive ai data again
|
|
if (a2dp_audio_context.audio_datapath.input_raw_packet_list)
|
|
a2dp_audio_list_clear(
|
|
a2dp_audio_context.audio_datapath.input_raw_packet_list);
|
|
}
|
|
|
|
int a2dp_audio_init(uint32_t sysfreq, A2DP_AUDIO_CODEC_TYPE codec_type,
|
|
A2DP_AUDIO_OUTPUT_CONFIG_T *config,
|
|
A2DP_AUDIO_CHANNEL_SELECT_E chnl_sel,
|
|
uint16_t dest_packet_mut) {
|
|
uint8_t *heap_buff = NULL;
|
|
uint32_t heap_size = 0;
|
|
double ratio = 0;
|
|
|
|
A2DP_AUDIO_OUTPUT_CONFIG_T decoder_output_config;
|
|
TRACE_A2DP_DECODER_I("[INIT] freq:%d codec:%d chnl:%d", sysfreq, codec_type,
|
|
chnl_sel);
|
|
TRACE_A2DP_DECODER_I("[INIT] out:%d-%d-%d smp:%d dest:%d",
|
|
config->sample_rate, config->num_channels,
|
|
config->bits_depth, config->frame_samples,
|
|
dest_packet_mut);
|
|
a2dp_audio_sysfreq_boost_init(sysfreq);
|
|
a2dp_audio_sysfreq_boost_start(A2DP_AUDIO_SYSFREQ_BOOST_RESUME_CNT);
|
|
a2dp_audio_semaphore_init();
|
|
a2dp_audio_buffer_mutex_init();
|
|
a2dp_audio_status_mutex_init();
|
|
|
|
a2dp_audio_status_mutex_lock();
|
|
|
|
a2dp_audio_detect_next_packet_callback_register(NULL);
|
|
a2dp_audio_detect_store_packet_callback_register(NULL);
|
|
|
|
#if defined(A2DP_LDAC_ON)
|
|
if (bt_sbc_player_get_codec_type() == BTIF_AVDTP_CODEC_TYPE_NON_A2DP) {
|
|
heap_size = A2DP_AUDIO_LDAC_MEMPOOL_SIZE;
|
|
} else
|
|
#endif
|
|
{
|
|
heap_size = A2DP_AUDIO_MEMPOOL_SIZE;
|
|
}
|
|
app_audio_mempool_get_buff(&heap_buff, heap_size);
|
|
ASSERT_A2DP_DECODER(heap_buff, "%s size:%d", __func__, heap_size);
|
|
a2dp_audio_heap_init(heap_buff, heap_size);
|
|
|
|
memset(&a2dp_audio_lastframe_info, 0, sizeof(A2DP_AUDIO_LASTFRAME_INFO_T));
|
|
|
|
a2dp_audio_context.audio_datapath.input_raw_packet_list = a2dp_audio_list_new(
|
|
a2dp_audio_packet_free, (list_mempool_zmalloc)a2dp_audio_heap_cmalloc,
|
|
(list_mempool_free)a2dp_audio_heap_free);
|
|
|
|
a2dp_audio_context.audio_datapath.output_pcm_packet_list =
|
|
a2dp_audio_list_new(a2dp_audio_packet_free,
|
|
(list_mempool_zmalloc)a2dp_audio_heap_cmalloc,
|
|
(list_mempool_free)a2dp_audio_heap_free);
|
|
|
|
memcpy(&(a2dp_audio_context.output_cfg), config,
|
|
sizeof(A2DP_AUDIO_OUTPUT_CONFIG_T));
|
|
ratio = a2dp_audio_context.output_cfg.factor_reference;
|
|
a2dp_audio_context.output_cfg.factor_reference = 0;
|
|
|
|
a2dp_audio_context.init_factor_reference = config->factor_reference;
|
|
a2dp_audio_context.chnl_sel = chnl_sel;
|
|
a2dp_audio_context.dest_packet_mut = dest_packet_mut;
|
|
a2dp_audio_context.average_packet_mut = 0;
|
|
|
|
switch (codec_type) {
|
|
case A2DP_AUDIO_CODEC_TYPE_SBC:
|
|
decoder_output_config.sample_rate = config->sample_rate;
|
|
decoder_output_config.num_channels = 2;
|
|
decoder_output_config.bits_depth = 16;
|
|
decoder_output_config.frame_samples = config->frame_samples;
|
|
decoder_output_config.factor_reference = 1.0f;
|
|
memcpy(&(a2dp_audio_context.audio_decoder), &a2dp_audio_sbc_decoder_config,
|
|
sizeof(A2DP_AUDIO_DECODER_T));
|
|
break;
|
|
#if defined(A2DP_AAC_ON)
|
|
case A2DP_AUDIO_CODEC_TYPE_MPEG2_4_AAC:
|
|
decoder_output_config.sample_rate = config->sample_rate;
|
|
decoder_output_config.num_channels = 2;
|
|
decoder_output_config.bits_depth = 16;
|
|
decoder_output_config.frame_samples = config->frame_samples;
|
|
decoder_output_config.factor_reference = 1.0f;
|
|
memcpy(&(a2dp_audio_context.audio_decoder),
|
|
&a2dp_audio_aac_lc_decoder_config, sizeof(A2DP_AUDIO_DECODER_T));
|
|
break;
|
|
#endif
|
|
#if defined(A2DP_SCALABLE_ON)
|
|
case A2DP_AUDIO_CODEC_TYPE_SCALABL:
|
|
decoder_output_config.sample_rate = config->sample_rate;
|
|
decoder_output_config.num_channels = 2;
|
|
decoder_output_config.bits_depth = config->curr_bits;
|
|
decoder_output_config.frame_samples = config->frame_samples;
|
|
decoder_output_config.factor_reference = 1.0f;
|
|
memcpy(&(a2dp_audio_context.audio_decoder),
|
|
&a2dp_audio_scalable_decoder_config, sizeof(A2DP_AUDIO_DECODER_T));
|
|
break;
|
|
#endif
|
|
#if defined(A2DP_LHDC_ON)
|
|
case A2DP_AUDIO_CODEC_TYPE_LHDC:
|
|
decoder_output_config.sample_rate = config->sample_rate;
|
|
decoder_output_config.num_channels = 2;
|
|
decoder_output_config.bits_depth = config->curr_bits;
|
|
decoder_output_config.frame_samples = config->frame_samples;
|
|
decoder_output_config.factor_reference = 1.0f;
|
|
memcpy(&(a2dp_audio_context.audio_decoder), &a2dp_audio_lhdc_decoder_config,
|
|
sizeof(A2DP_AUDIO_DECODER_T));
|
|
break;
|
|
#endif
|
|
#if defined(A2DP_LDAC_ON)
|
|
case A2DP_AUDIO_CODEC_TYPE_LDAC:
|
|
decoder_output_config.sample_rate = config->sample_rate;
|
|
decoder_output_config.num_channels = 2;
|
|
decoder_output_config.bits_depth = config->curr_bits;
|
|
decoder_output_config.frame_samples = config->frame_samples;
|
|
decoder_output_config.factor_reference = 1.0f;
|
|
memcpy(&(a2dp_audio_context.audio_decoder), &a2dp_audio_ldac_decoder_config,
|
|
sizeof(A2DP_AUDIO_DECODER_T));
|
|
break;
|
|
#endif
|
|
default:
|
|
ASSERT_A2DP_DECODER(0, "%s invalid codec_type:%d", __func__, codec_type);
|
|
break;
|
|
}
|
|
|
|
a2dp_audio_context.audio_decoder.audio_decoder_init(
|
|
&decoder_output_config, (void *)&a2dp_audio_context);
|
|
a2dp_audio_context.need_detect_first_packet = true;
|
|
a2dp_audio_context.underflow_onporcess = false;
|
|
a2dp_audio_context.skip_frame_cnt_after_no_cache = 0;
|
|
a2dp_audio_context.mute_frame_cnt_after_no_cache = 0;
|
|
|
|
a2dp_audio_context.audio_decoder_status = A2DP_AUDIO_DECODER_STATUS_READY;
|
|
|
|
a2dp_audio_context.store_packet_status =
|
|
A2DP_AUDIO_DECODER_STORE_PACKET_STATUS_IDLE;
|
|
a2dp_audio_context.playback_status = A2DP_AUDIO_DECODER_PLAYBACK_STATUS_IDLE;
|
|
|
|
a2dp_audio_sync_init(ratio);
|
|
|
|
#if A2DP_DECODER_HISTORY_SEQ_SAVE
|
|
a2dp_audio_reset_history_seq();
|
|
#endif
|
|
a2dp_audio_store_packet_checker_start();
|
|
|
|
a2dp_audio_status_mutex_unlock();
|
|
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_deinit(void) {
|
|
TRACE_A2DP_DECODER_I("[DEINIT]");
|
|
|
|
a2dp_audio_status_mutex_lock();
|
|
|
|
a2dp_audio_detect_next_packet_callback_register(NULL);
|
|
a2dp_audio_detect_store_packet_callback_register(NULL);
|
|
|
|
a2dp_audio_context.audio_decoder.audio_decoder_deinit();
|
|
memset(&(a2dp_audio_context.audio_decoder), 0, sizeof(A2DP_AUDIO_DECODER_T));
|
|
memset(&(a2dp_audio_context.output_cfg), 0,
|
|
sizeof(A2DP_AUDIO_OUTPUT_CONFIG_T));
|
|
a2dp_audio_list_clear(
|
|
a2dp_audio_context.audio_datapath.input_raw_packet_list);
|
|
a2dp_audio_list_free(a2dp_audio_context.audio_datapath.input_raw_packet_list);
|
|
a2dp_audio_context.audio_datapath.input_raw_packet_list = NULL;
|
|
a2dp_audio_list_clear(
|
|
a2dp_audio_context.audio_datapath.output_pcm_packet_list);
|
|
a2dp_audio_list_free(
|
|
a2dp_audio_context.audio_datapath.output_pcm_packet_list);
|
|
a2dp_audio_context.audio_datapath.output_pcm_packet_list = NULL;
|
|
|
|
size_t total = 0, used = 0, max_used = 0;
|
|
a2dp_audio_heap_info(&total, &used, &max_used);
|
|
TRACE_A2DP_DECODER_I(
|
|
"[DEINIT] heap info: total - %d, used - %d, max_used - %d.", total, used,
|
|
max_used);
|
|
// ASSERT_A2DP_DECODER(used == 0, "[%s] used != 0", __func__);
|
|
|
|
a2dp_audio_set_store_packet_status(
|
|
A2DP_AUDIO_DECODER_STORE_PACKET_STATUS_IDLE);
|
|
a2dp_audio_set_playback_status(A2DP_AUDIO_DECODER_PLAYBACK_STATUS_IDLE);
|
|
a2dp_audio_set_status(A2DP_AUDIO_DECODER_STATUS_NULL);
|
|
#if A2DP_DECODER_HISTORY_SEQ_SAVE
|
|
a2dp_audio_reset_history_seq();
|
|
#endif
|
|
a2dp_audio_status_mutex_unlock();
|
|
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_stop(void) {
|
|
TRACE_A2DP_DECODER_I("[STOP]");
|
|
int cnt = 0;
|
|
|
|
a2dp_audio_set_status(A2DP_AUDIO_DECODER_STATUS_STOP);
|
|
a2dp_audio_semaphore_release();
|
|
|
|
cnt = 50;
|
|
do {
|
|
if (a2dp_audio_get_playback_status() ==
|
|
A2DP_AUDIO_DECODER_PLAYBACK_STATUS_IDLE) {
|
|
TRACE_A2DP_DECODER_I("[DEINIT]PLAYBACK_STATUS_IDLE cnt:%d", cnt);
|
|
break;
|
|
} else {
|
|
osThreadYield();
|
|
}
|
|
} while (--cnt > 0);
|
|
cnt = 50;
|
|
do {
|
|
if (a2dp_audio_get_store_packet_status() ==
|
|
A2DP_AUDIO_DECODER_STORE_PACKET_STATUS_IDLE) {
|
|
TRACE_A2DP_DECODER_I("[DEINIT] STORE_PACKET_STATUS_IDLE cnt:%d", cnt);
|
|
break;
|
|
} else {
|
|
osThreadYield();
|
|
}
|
|
} while (--cnt > 0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_start(void) {
|
|
TRACE_A2DP_DECODER_I("[START]");
|
|
a2dp_audio_status_mutex_lock();
|
|
a2dp_audio_set_status(A2DP_AUDIO_DECODER_STATUS_START);
|
|
a2dp_audio_status_mutex_unlock();
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_detect_next_packet_callback_register(
|
|
A2DP_AUDIO_DETECT_NEXT_PACKET_CALLBACK callback) {
|
|
a2dp_audio_status_mutex_lock();
|
|
a2dp_audio_detect_next_packet_callback = callback;
|
|
a2dp_audio_status_mutex_unlock();
|
|
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_detect_store_packet_callback_register(
|
|
A2DP_AUDIO_DETECT_NEXT_PACKET_CALLBACK callback) {
|
|
a2dp_audio_status_mutex_lock();
|
|
a2dp_audio_store_packet_callback = callback;
|
|
a2dp_audio_status_mutex_unlock();
|
|
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_detect_first_packet(void) {
|
|
a2dp_audio_status_mutex_lock();
|
|
a2dp_audio_context.need_detect_first_packet = true;
|
|
a2dp_audio_status_mutex_unlock();
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_detect_first_packet_clear(void) {
|
|
a2dp_audio_status_mutex_lock();
|
|
a2dp_audio_context.need_detect_first_packet = false;
|
|
a2dp_audio_status_mutex_unlock();
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_discards_packet(uint32_t packets) {
|
|
int nRet = 0;
|
|
|
|
if (a2dp_audio_get_status() == A2DP_AUDIO_DECODER_STATUS_START) {
|
|
a2dp_audio_status_mutex_lock();
|
|
nRet =
|
|
a2dp_audio_context.audio_decoder.audio_decoder_discards_packet(packets);
|
|
a2dp_audio_status_mutex_unlock();
|
|
} else {
|
|
nRet = -1;
|
|
}
|
|
|
|
return nRet;
|
|
}
|
|
|
|
int a2dp_audio_synchronize_dest_packet_mut(uint32_t mtu) {
|
|
int nRet = 0;
|
|
int cnt = 50;
|
|
if (a2dp_audio_get_status() == A2DP_AUDIO_DECODER_STATUS_START) {
|
|
do {
|
|
if (a2dp_audio_get_playback_status() ==
|
|
A2DP_AUDIO_DECODER_PLAYBACK_STATUS_IDLE) {
|
|
nRet = a2dp_audio_context.audio_decoder
|
|
.audio_decoder_synchronize_dest_packet_mut(mtu);
|
|
break;
|
|
} else {
|
|
osThreadYield();
|
|
}
|
|
} while (--cnt > 0);
|
|
} else {
|
|
nRet = -1;
|
|
}
|
|
|
|
return nRet;
|
|
}
|
|
|
|
int a2dp_audio_discards_samples(uint32_t samples) {
|
|
return a2dp_audio_context.audio_decoder.a2dp_audio_discards_samples(samples);
|
|
}
|
|
|
|
int a2dp_audio_convert_list_to_samples(uint32_t *samples) {
|
|
return a2dp_audio_context.audio_decoder.a2dp_audio_convert_list_to_samples(
|
|
samples);
|
|
}
|
|
|
|
int a2dp_audio_get_packet_samples(void) {
|
|
A2DP_AUDIO_LASTFRAME_INFO_T lastframe_info;
|
|
uint32_t packet_samples = 0;
|
|
uint16_t totalSubSequenceNumber = 1;
|
|
|
|
a2dp_audio_lastframe_info_get(&lastframe_info);
|
|
|
|
if (lastframe_info.totalSubSequenceNumber) {
|
|
totalSubSequenceNumber = lastframe_info.totalSubSequenceNumber;
|
|
}
|
|
|
|
packet_samples = totalSubSequenceNumber * lastframe_info.frame_samples;
|
|
|
|
return packet_samples;
|
|
}
|
|
|
|
static int a2dp_audio_internal_lastframe_info_ptr_get(
|
|
A2DP_AUDIO_LASTFRAME_INFO_T **lastframe_info) {
|
|
int nRet = 0;
|
|
|
|
if (a2dp_audio_get_status() == A2DP_AUDIO_DECODER_STATUS_START) {
|
|
*lastframe_info = &a2dp_audio_lastframe_info;
|
|
} else {
|
|
*lastframe_info = NULL;
|
|
nRet = -1;
|
|
}
|
|
|
|
return nRet;
|
|
}
|
|
|
|
int a2dp_audio_lastframe_info_get(A2DP_AUDIO_LASTFRAME_INFO_T *lastframe_info) {
|
|
int nRet = 0;
|
|
|
|
if (a2dp_audio_get_status() == A2DP_AUDIO_DECODER_STATUS_START) {
|
|
a2dp_audio_buffer_mutex_lock();
|
|
memcpy(lastframe_info, &a2dp_audio_lastframe_info,
|
|
sizeof(A2DP_AUDIO_LASTFRAME_INFO_T));
|
|
a2dp_audio_buffer_mutex_unlock();
|
|
} else {
|
|
memset(lastframe_info, 0, sizeof(A2DP_AUDIO_LASTFRAME_INFO_T));
|
|
nRet = -1;
|
|
}
|
|
|
|
return nRet;
|
|
}
|
|
|
|
int a2dp_audio_decoder_internal_check_sum_reset(void) {
|
|
check_sum_seed = 0;
|
|
return 0;
|
|
}
|
|
|
|
uint32_t a2dp_audio_decoder_internal_check_sum_generate(const uint8_t *buf,
|
|
uint32_t len) {
|
|
#if A2DP_DECODER_CHECKER
|
|
check_sum_seed = crc32(check_sum_seed, buf, len);
|
|
#else
|
|
check_sum_seed = 0;
|
|
#endif
|
|
return check_sum_seed;
|
|
}
|
|
|
|
int a2dp_audio_lastframe_info_reset_undecodeframe(void) {
|
|
int nRet = 0;
|
|
|
|
if (a2dp_audio_get_status() == A2DP_AUDIO_DECODER_STATUS_START) {
|
|
a2dp_audio_buffer_mutex_lock();
|
|
a2dp_audio_lastframe_info.undecode_frames = 0;
|
|
a2dp_audio_lastframe_info.undecode_max_frames = 0;
|
|
a2dp_audio_lastframe_info.undecode_min_frames = 0xffff;
|
|
a2dp_audio_decoder_internal_check_sum_reset();
|
|
a2dp_audio_lastframe_info.check_sum = 0;
|
|
a2dp_audio_buffer_mutex_unlock();
|
|
} else {
|
|
nRet = -1;
|
|
}
|
|
|
|
return nRet;
|
|
}
|
|
|
|
int a2dp_audio_decoder_internal_lastframe_info_set(
|
|
A2DP_AUDIO_DECODER_LASTFRAME_INFO_T *lastframe_info) {
|
|
a2dp_audio_buffer_mutex_lock();
|
|
a2dp_audio_lastframe_info.sequenceNumber = lastframe_info->sequenceNumber;
|
|
a2dp_audio_lastframe_info.timestamp = lastframe_info->timestamp;
|
|
a2dp_audio_lastframe_info.curSubSequenceNumber =
|
|
lastframe_info->curSubSequenceNumber;
|
|
a2dp_audio_lastframe_info.totalSubSequenceNumber =
|
|
lastframe_info->totalSubSequenceNumber;
|
|
a2dp_audio_lastframe_info.frame_samples = lastframe_info->frame_samples;
|
|
a2dp_audio_lastframe_info.list_samples = lastframe_info->list_samples;
|
|
a2dp_audio_lastframe_info.decoded_frames = lastframe_info->decoded_frames;
|
|
a2dp_audio_lastframe_info.undecode_frames = lastframe_info->undecode_frames;
|
|
|
|
a2dp_audio_lastframe_info.undecode_max_frames =
|
|
MAX(a2dp_audio_lastframe_info.undecode_frames,
|
|
a2dp_audio_lastframe_info.undecode_max_frames);
|
|
a2dp_audio_lastframe_info.undecode_min_frames =
|
|
MIN(a2dp_audio_lastframe_info.undecode_frames,
|
|
a2dp_audio_lastframe_info.undecode_min_frames);
|
|
|
|
a2dp_audio_lastframe_info.stream_info = lastframe_info->stream_info;
|
|
a2dp_audio_lastframe_info.check_sum = lastframe_info->check_sum;
|
|
a2dp_audio_buffer_mutex_unlock();
|
|
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_synchronize_packet(A2DP_AUDIO_SYNCFRAME_INFO_T *sync_info,
|
|
uint32_t mask) {
|
|
int nRet = 0;
|
|
|
|
if (a2dp_audio_get_status() == A2DP_AUDIO_DECODER_STATUS_START) {
|
|
a2dp_audio_status_mutex_lock();
|
|
nRet = a2dp_audio_context.audio_decoder.audio_decoder_synchronize_packet(
|
|
sync_info, mask);
|
|
if (nRet == A2DP_DECODER_NOT_SUPPORT) {
|
|
// can't support synchronize packet, so return fake val;
|
|
nRet = A2DP_DECODER_NO_ERROR;
|
|
}
|
|
a2dp_audio_status_mutex_unlock();
|
|
} else {
|
|
nRet = -1;
|
|
}
|
|
|
|
return nRet;
|
|
}
|
|
|
|
int a2dp_audio_decoder_headframe_info_get(
|
|
A2DP_AUDIO_HEADFRAME_INFO_T *headframe_info) {
|
|
int nRet = 0;
|
|
|
|
if (a2dp_audio_get_status() == A2DP_AUDIO_DECODER_STATUS_START) {
|
|
nRet = a2dp_audio_context.audio_decoder.audio_decoder_headframe_info_get(
|
|
headframe_info);
|
|
} else {
|
|
memset(headframe_info, 0, sizeof(A2DP_AUDIO_HEADFRAME_INFO_T));
|
|
nRet = -1;
|
|
}
|
|
|
|
return nRet;
|
|
}
|
|
|
|
int a2dp_audio_refill_packet(void) {
|
|
int refill_cnt = 0;
|
|
|
|
#if defined(A2DP_AUDIO_REFILL_AFTER_NO_CACHE)
|
|
refill_cnt += a2dp_audio_context.skip_frame_cnt_after_no_cache;
|
|
#endif
|
|
refill_cnt += a2dp_audio_context.mute_frame_cnt_after_no_cache;
|
|
|
|
return refill_cnt;
|
|
}
|
|
|
|
bool a2dp_audio_auto_synchronize_support(void) {
|
|
bool nRet = 0;
|
|
|
|
if (a2dp_audio_get_status() == A2DP_AUDIO_DECODER_STATUS_START) {
|
|
a2dp_audio_buffer_mutex_lock();
|
|
nRet = a2dp_audio_context.audio_decoder.auto_synchronize_support > 0
|
|
? true
|
|
: false;
|
|
a2dp_audio_buffer_mutex_unlock();
|
|
} else {
|
|
nRet = 0;
|
|
}
|
|
|
|
return nRet;
|
|
}
|
|
|
|
A2DP_AUDIO_OUTPUT_CONFIG_T *a2dp_audio_get_output_config(void) {
|
|
A2DP_AUDIO_OUTPUT_CONFIG_T *output_config = NULL;
|
|
|
|
if (a2dp_audio_get_status() == A2DP_AUDIO_DECODER_STATUS_START) {
|
|
a2dp_audio_buffer_mutex_lock();
|
|
output_config = &a2dp_audio_context.output_cfg;
|
|
a2dp_audio_buffer_mutex_unlock();
|
|
} else {
|
|
output_config = NULL;
|
|
}
|
|
|
|
return output_config;
|
|
}
|
|
|
|
int a2dp_audio_latency_factor_setlow(void) {
|
|
a2dp_audio_latency_factor = A2DP_AUDIO_LATENCY_LOW_FACTOR;
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_latency_factor_sethigh(void) {
|
|
a2dp_audio_latency_factor = A2DP_AUDIO_LATENCY_HIGH_FACTOR;
|
|
return 0;
|
|
}
|
|
|
|
float a2dp_audio_latency_factor_get(void) { return a2dp_audio_latency_factor; }
|
|
|
|
int a2dp_audio_latency_factor_set(float factor) {
|
|
a2dp_audio_latency_factor = factor;
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_latency_factor_status_get(
|
|
A2DP_AUDIO_LATENCY_STATUS_E *latency_status, float *more_latency_factor) {
|
|
if (a2dp_audio_latency_factor == A2DP_AUDIO_LATENCY_HIGH_FACTOR) {
|
|
*latency_status = A2DP_AUDIO_LATENCY_STATUS_HIGH;
|
|
*more_latency_factor = A2DP_AUDIO_LATENCY_MORE_FACTOR;
|
|
} else {
|
|
*latency_status = A2DP_AUDIO_LATENCY_STATUS_LOW;
|
|
*more_latency_factor = 1.0f;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int a2dp_audio_frame_delay_get(void) { return get_in_cp_frame_delay(); }
|
|
|
|
int a2dp_audio_dest_packet_mut_get(void) {
|
|
return a2dp_audio_context.dest_packet_mut;
|
|
}
|
|
|
|
int a2dp_audio_set_channel_select(A2DP_AUDIO_CHANNEL_SELECT_E chnl_sel) {
|
|
int nRet = A2DP_DECODER_NO_ERROR;
|
|
|
|
if (a2dp_audio_get_status() == A2DP_AUDIO_DECODER_STATUS_START) {
|
|
a2dp_audio_context.chnl_sel = chnl_sel;
|
|
if (a2dp_audio_context.audio_decoder.audio_decoder_channel_select) {
|
|
nRet = a2dp_audio_context.audio_decoder.audio_decoder_channel_select(
|
|
chnl_sel);
|
|
}
|
|
}
|
|
|
|
return nRet;
|
|
}
|
|
|
|
float a2dp_audio_get_sample_reference(void) {
|
|
TRACE(1, "a2dp_audio_get_sample_reference:%d",
|
|
(int32_t)(a2dp_audio_context.output_cfg.factor_reference * 10000000));
|
|
return a2dp_audio_context.output_cfg.factor_reference;
|
|
}
|
|
|
|
int8_t a2dp_audio_get_current_buf_size(void) {
|
|
TRACE(1, "a2dp_audio_get_current_buf_size:%d",
|
|
(int8_t)(a2dp_audio_context.average_packet_mut + 0.5f));
|
|
return (int8_t)(a2dp_audio_context.average_packet_mut + 0.5f);
|
|
}
|