pinebuds/services/bt_app/app_bt.cpp
Ben V. Brown dca92cf01f Removing FPGA dev support
As we will never get their FGPA source code. Zero loss.
2023-02-02 17:42:33 +11:00

5095 lines
179 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.
*
****************************************************************************/
#include "app_bt.h"
#include "a2dp_api.h"
#include "app_a2dp.h"
#include "app_ai_manager_api.h"
#include "app_bt.h"
#include "app_bt_func.h"
#include "app_dip.h"
#include "app_hfp.h"
#include "app_status_ind.h"
#include "app_thread.h"
#include "apps.h"
#include "besbt.h"
#include "besbt_cfg.h"
#include "bluetooth.h"
#include "bt_drv_interface.h"
#include "bt_drv_reg_op.h"
#include "btapp.h"
#include "dip_api.h"
#include "hal_aud.h"
#include "hal_chipid.h"
#include "hal_trace.h"
#include "hci_api.h"
#include "hfp_api.h"
#include "l2cap_api.h"
#include "me_api.h"
#include "mei_api.h"
#include "nvrecord.h"
#include "os_api.h"
#if defined(IBRT)
#include "app_ibrt_a2dp.h"
#include "app_ibrt_if.h"
#include "app_ibrt_ui.h"
#include "app_tws_ibrt.h"
#endif
#ifdef __THIRDPARTY
#include "app_thirdparty.h"
#endif
#include "app_ble_mode_switch.h"
#ifdef GFPS_ENABLED
#include "app_gfps.h"
#endif
#ifdef __AI_VOICE__
#include "ai_manager.h"
#include "ai_spp.h"
#include "ai_thread.h"
#endif
#ifdef __INTERCONNECTION__
#include "app_interconnection.h"
#include "app_interconnection_ble.h"
#include "app_interconnection_ccmp.h"
#include "app_interconnection_spp.h"
#include "app_spp.h"
#include "spp_api.h"
#endif
#ifdef __INTERACTION__
#include "app_interaction.h"
#endif
#ifdef BISTO_ENABLED
#include "gsound_custom_bt.h"
#endif
#if defined(__EARPHONE_STAY_BCR_SLAVE__) && defined(__BT_ONE_BRING_TWO__)
#error can not defined at the same time.
#endif
#ifdef GFPS_ENABLED
#include "app_gfps.h"
#endif
#ifdef TILE_DATAPATH
#include "rwip_config.h"
#include "tile_target_ble.h"
#endif
#if (A2DP_DECODER_VER >= 2)
#include "a2dp_decoder.h"
#endif
extern "C" {
#include "ddbif.h"
}
extern struct BT_DEVICE_T app_bt_device;
extern "C" bool app_anc_work_status(void);
extern uint8_t avrcp_get_media_status(void);
void avrcp_set_media_status(uint8_t status);
#ifdef GFPS_ENABLED
extern "C" void
app_gfps_handling_on_mobile_link_disconnection(btif_remote_device_t *pRemDev);
#endif
static btif_remote_device_t *connectedMobile = NULL;
static btif_remote_device_t *sppOpenMobile = NULL;
U16 bt_accessory_feature_feature = BTIF_HF_CUSTOM_FEATURE_SUPPORT;
#ifdef __BT_ONE_BRING_TWO__
btif_device_record_t record2_copy;
uint8_t record2_avalible;
#endif
enum bt_profile_reconnect_mode {
bt_profile_reconnect_null,
bt_profile_reconnect_openreconnecting,
bt_profile_reconnect_reconnecting,
bt_profile_reconnect_reconnect_pending,
};
enum bt_profile_connect_status {
bt_profile_connect_status_unknow,
bt_profile_connect_status_success,
bt_profile_connect_status_failure,
};
struct app_bt_profile_manager {
bool has_connected;
enum bt_profile_connect_status hfp_connect;
enum bt_profile_connect_status hsp_connect;
enum bt_profile_connect_status a2dp_connect;
bt_bdaddr_t rmt_addr;
bt_profile_reconnect_mode reconnect_mode;
bt_profile_reconnect_mode saved_reconnect_mode;
a2dp_stream_t *stream;
// HfChannel *chan;
hf_chan_handle_t chan;
#if defined(__HSP_ENABLE__)
HsChannel *hs_chan;
#endif
uint16_t reconnect_cnt;
osTimerId connect_timer;
void (*connect_timer_cb)(void const *);
APP_BT_CONNECTING_STATE_E connectingState;
};
// reconnect = (INTERVAL+PAGETO)*CNT = (3000ms+5000ms)*15 = 120s
#define APP_BT_PROFILE_RECONNECT_RETRY_INTERVAL_MS (3000)
#define APP_BT_PROFILE_OPENNING_RECONNECT_RETRY_LIMIT_CNT (2)
#define APP_BT_PROFILE_RECONNECT_RETRY_LIMIT_CNT (15)
#define APP_BT_PROFILE_CONNECT_RETRY_MS (10000)
static struct app_bt_profile_manager bt_profile_manager[BT_DEVICE_NUM];
static int8_t app_bt_profile_reconnect_pending_process(void);
void app_bt_connectable_mode_stop_reconnecting(void);
btif_accessible_mode_t g_bt_access_mode = BTIF_BAM_NOT_ACCESSIBLE;
#define APP_BT_PROFILE_BOTH_SCAN_MS (11000)
#define APP_BT_PROFILE_PAGE_SCAN_MS (4000)
osTimerId app_bt_accessmode_timer = NULL;
btif_accessible_mode_t app_bt_accessmode_timer_argument =
BTIF_BAM_NOT_ACCESSIBLE;
static int app_bt_accessmode_timehandler(void const *param);
osTimerDef(APP_BT_ACCESSMODE_TIMER,
(void (*)(void const *))
app_bt_accessmode_timehandler); // define timers
#define A2DP_CONN_CLOSED 10
#ifdef __IAG_BLE_INCLUDE__
#define APP_FAST_BLE_ADV_TIMEOUT_IN_MS 30000
osTimerId app_fast_ble_adv_timeout_timer = NULL;
static int app_fast_ble_adv_timeout_timehandler(void const *param);
osTimerDef(APP_FAST_BLE_ADV_TIMEOUT_TIMER,
(void (*)(void const *))
app_fast_ble_adv_timeout_timehandler); // define timers
/*---------------------------------------------------------------------------
* app_start_fast_connectable_ble_adv
*---------------------------------------------------------------------------
*
*Synopsis:
* start fast connectable BLE adv
*
* Parameters:
* advInterval - adv interval
*
* Return:
* void
*/
static void app_start_fast_connectable_ble_adv(uint16_t advInterval);
#endif
#if defined(__INTERCONNECTION__)
btif_accessible_mode_t app_bt_get_current_access_mode(void) {
return g_bt_access_mode;
}
bool app_bt_is_connected() {
uint8_t i = 0;
bool connceted_value = false;
for (i = 0; i < BT_DEVICE_NUM; i++) {
if (bt_profile_manager[i].has_connected) {
connceted_value = true;
break;
}
}
TRACE(1, "bt_is_connected is %d", connceted_value);
return connceted_value;
}
#endif
bool app_device_bt_is_connected() {
uint8_t i = 0;
bool connceted_value = false;
for (i = 0; i < BT_DEVICE_NUM; i++) {
if (bt_profile_manager[i].has_connected) {
connceted_value = true;
break;
}
}
// TRACE("bt_is_connected : %d, a2dp : %d, hfp : %d",
// connceted_value,bt_profile_manager[i].a2dp_connect,bt_profile_manager[i].hfp_connect);
return connceted_value;
}
static void app_bt_precheck_before_starting_connecting(uint8_t isBtConnected);
static void app_bt_accessmode_handler(btif_accessible_mode_t accMode) {
const btif_access_mode_info_t info = {
BTIF_BT_DEFAULT_INQ_SCAN_INTERVAL, BTIF_BT_DEFAULT_INQ_SCAN_WINDOW,
BTIF_BT_DEFAULT_PAGE_SCAN_INTERVAL, BTIF_BT_DEFAULT_PAGE_SCAN_WINDOW};
osapi_lock_stack();
if (accMode == BTIF_BAM_CONNECTABLE_ONLY) {
app_bt_accessmode_timer_argument = BTIF_BAM_GENERAL_ACCESSIBLE;
osTimerStart(app_bt_accessmode_timer, APP_BT_PROFILE_PAGE_SCAN_MS);
} else if (accMode == BTIF_BAM_GENERAL_ACCESSIBLE) {
app_bt_accessmode_timer_argument = BTIF_BAM_CONNECTABLE_ONLY;
osTimerStart(app_bt_accessmode_timer, APP_BT_PROFILE_BOTH_SCAN_MS);
}
app_bt_ME_SetAccessibleMode(accMode, &info);
TRACE(1, "app_bt_accessmode_timehandler accMode=%x", accMode);
osapi_unlock_stack();
}
static int app_bt_accessmode_timehandler(void const *param) {
btif_accessible_mode_t accMode = *(btif_accessible_mode_t *)(param);
app_bt_start_custom_function_in_bt_thread(
(uint32_t)accMode, 0, (uint32_t)app_bt_accessmode_handler);
return 0;
}
void app_bt_accessmode_set(btif_accessible_mode_t mode) {
const btif_access_mode_info_t info = {
BTIF_BT_DEFAULT_INQ_SCAN_INTERVAL, BTIF_BT_DEFAULT_INQ_SCAN_WINDOW,
BTIF_BT_DEFAULT_PAGE_SCAN_INTERVAL, BTIF_BT_DEFAULT_PAGE_SCAN_WINDOW};
#if defined(IBRT)
return;
#endif
TRACE(2, "%s %d", __func__, mode);
osapi_lock_stack();
g_bt_access_mode = mode;
if (g_bt_access_mode == BTIF_BAM_GENERAL_ACCESSIBLE) {
app_bt_accessmode_timehandler(&g_bt_access_mode);
} else {
osTimerStop(app_bt_accessmode_timer);
app_bt_ME_SetAccessibleMode(g_bt_access_mode, &info);
TRACE(1, "app_bt_accessmode_set access_mode=%x", g_bt_access_mode);
}
osapi_unlock_stack();
}
extern "C" uint8_t app_bt_get_act_cons(void) {
int activeCons;
osapi_lock_stack();
activeCons = btif_me_get_activeCons();
osapi_unlock_stack();
TRACE(2, "%s %d", __func__, activeCons);
return activeCons;
}
enum {
INITIATE_PAIRING_NONE = 0,
INITIATE_PAIRING_RUN = 1,
};
static uint8_t initiate_pairing = INITIATE_PAIRING_NONE;
void app_bt_connectable_state_set(uint8_t set) { initiate_pairing = set; }
bool is_app_bt_pairing_running(void) {
return (initiate_pairing == INITIATE_PAIRING_RUN) ? (true) : (false);
}
#define APP_DISABLE_PAGE_SCAN_AFTER_CONN
#ifdef APP_DISABLE_PAGE_SCAN_AFTER_CONN
osTimerId disable_page_scan_check_timer = NULL;
static void disable_page_scan_check_timer_handler(void const *param);
osTimerDef(DISABLE_PAGE_SCAN_CHECK_TIMER,
(void (*)(void const *))
disable_page_scan_check_timer_handler); // define timers
static void disable_page_scan_check_timer_handler(void const *param) {
#ifdef __BT_ONE_BRING_TWO__
if ((btif_me_get_activeCons() > 1) &&
(initiate_pairing == INITIATE_PAIRING_NONE)) {
#else
if ((btif_me_get_activeCons() > 0) &&
(initiate_pairing == INITIATE_PAIRING_NONE)) {
#endif
app_bt_accessmode_set_req(BTIF_BAM_NOT_ACCESSIBLE);
}
}
static void disable_page_scan_check_timer_start(void) {
if (disable_page_scan_check_timer == NULL) {
disable_page_scan_check_timer = osTimerCreate(
osTimer(DISABLE_PAGE_SCAN_CHECK_TIMER), osTimerOnce, NULL);
}
osTimerStart(disable_page_scan_check_timer, 4000);
}
#endif
void PairingTransferToConnectable(void) {
int activeCons;
osapi_lock_stack();
activeCons = btif_me_get_activeCons();
osapi_unlock_stack();
TRACE(1, "%s", __func__);
app_bt_connectable_state_set(INITIATE_PAIRING_NONE);
if (activeCons == 0) {
TRACE(0, "!!!PairingTransferToConnectable BAM_CONNECTABLE_ONLY\n");
app_bt_accessmode_set_req(BTIF_BAM_CONNECTABLE_ONLY);
}
}
int app_bt_get_audio_up_id(void) {
uint8_t i;
btif_remote_device_t *remDev = NULL;
btif_cmgr_handler_t *cmgrHandler;
for (i = 0; i < BT_DEVICE_NUM; i++) {
remDev = btif_me_enumerate_remote_devices(i);
if (remDev != NULL) {
cmgrHandler = btif_cmgr_get_acl_handler(remDev);
if (btif_cmgr_is_audio_up(cmgrHandler) == true)
break;
}
}
return i;
}
#define HFP_DEBUG
void hfp_call_state_checker(void);
#if defined(IBRT)
int app_bt_ibrt_profile_checker(const char *str, btif_remote_device_t *remDev,
btif_cmgr_handler_t *cmgrHandler,
a2dp_stream_t *a2dp_stream,
hf_chan_handle_t hf_channel) {
if (remDev && cmgrHandler) {
TRACE(6, "checker: %s remDev state:%d mode:%d role:%d sniffInterval:%d/%d",
str, btif_me_get_remote_device_state(remDev),
btif_me_get_remote_device_mode(remDev),
btif_me_get_remote_device_role(remDev),
btif_cmgr_get_cmgrhandler_sniff_Interval(cmgrHandler),
btif_cmgr_get_cmgrhandler_sniff_info(cmgrHandler)->maxInterval);
TRACE(2, "checker: %s remDev:%p remote_dev_address:", str, remDev);
DUMP8("0x%02x ", btif_me_get_remote_device_bdaddr(remDev)->address,
BTIF_BD_ADDR_SIZE);
} else {
TRACE(3, "checker: %s remDev:%p cmgrHandler:%p", str, remDev, cmgrHandler);
}
if (a2dp_stream) {
TRACE(1, "a2dp State:%d", btif_a2dp_get_stream_state(a2dp_stream));
if (btif_a2dp_get_stream_state(a2dp_stream) > BTIF_AVDTP_STRM_STATE_IDLE) {
TRACE(1, "a2dp cmgrHandler remDev:%p",
btif_a2dp_get_stream_devic_cmgrHandler_remdev(a2dp_stream));
}
} else {
TRACE(1, "%s a2dp stream NULL", str);
}
if (hf_channel) {
TRACE(4, "hf_channel Connected:%d IsAudioUp:%d/%d remDev:%p",
btif_hf_is_acl_connected(hf_channel), app_bt_device.hf_audio_state[0],
btif_hf_check_AudioConnect_status(hf_channel),
btif_hf_cmgr_get_remote_device(hf_channel));
} else {
TRACE(1, "%s hf_channel NULL", str);
}
return 0;
}
#endif
int app_bt_state_checker(void) {
btif_remote_device_t *remDev = NULL;
btif_cmgr_handler_t *cmgrHandler;
osapi_lock_stack();
#if defined(IBRT)
ibrt_ctrl_t *p_ibrt_ctrl = app_tws_ibrt_get_bt_ctrl_ctx();
if (app_tws_ibrt_mobile_link_connected()) {
TRACE(1, "checker: IBRT_MASTER activeCons:%d", btif_me_get_activeCons());
remDev = p_ibrt_ctrl->p_tws_remote_dev;
if (remDev != NULL) {
cmgrHandler = btif_cmgr_get_acl_handler(remDev);
if (cmgrHandler) {
app_bt_ibrt_profile_checker("tws peers", remDev, cmgrHandler, NULL,
NULL);
} else {
TRACE(0, "checker: cmgrhandler not handle p_tws_remote_dev!");
}
} else {
TRACE(0, "checker: tws_remote_dev is NULL!");
}
remDev = p_ibrt_ctrl->p_mobile_remote_dev;
if (remDev != NULL) {
cmgrHandler = btif_cmgr_get_acl_handler(remDev);
if (cmgrHandler) {
app_bt_ibrt_profile_checker(
"master mobile", remDev, cmgrHandler,
app_bt_device.a2dp_connected_stream[0],
app_bt_device.hf_conn_flag[0] ? app_bt_device.hf_channel[0] : NULL);
} else {
TRACE(0, "checker: cmgrhandler not handle mobile_remote_dev");
}
} else {
TRACE(0, "checker: mobile_remote_dev is NULL!");
}
} else if (app_tws_ibrt_slave_ibrt_link_connected()) {
TRACE(1, "checker: IBRT_SLAVE activeCons:%d", btif_me_get_activeCons());
remDev = p_ibrt_ctrl->p_tws_remote_dev;
if (remDev != NULL) {
cmgrHandler = btif_cmgr_get_acl_handler(remDev);
if (cmgrHandler) {
app_bt_ibrt_profile_checker("tws peers", remDev, cmgrHandler, NULL,
NULL);
} else {
TRACE(0, "checker: cmgrhandler not handle p_tws_remote_dev!");
}
} else {
TRACE(0, "checker: tws_remote_dev is NULL!");
}
if (app_ibrt_ui_is_profile_exchanged()) {
app_bt_ibrt_profile_checker(
"ibrt mobile", NULL, NULL, app_bt_device.a2dp_connected_stream[0],
app_bt_device.hf_conn_flag[0] ? app_bt_device.hf_channel[0] : NULL);
}
} else {
TRACE(1, "checker: IBRT_UNKNOW activeCons:%d", btif_me_get_activeCons());
}
app_ibrt_if_ctx_checker();
#if defined(ENHANCED_STACK)
btif_me_cobuf_state_dump();
btif_me_hcibuff_state_dump();
#endif
// BT controller state checker
ASSERT(bt_drv_reg_op_check_bt_controller_state(), "BT controller dead!");
#else
for (uint8_t i = 0; i < BT_DEVICE_NUM; i++) {
remDev = btif_me_enumerate_remote_devices(i);
if (remDev != NULL) {
cmgrHandler = btif_cmgr_get_acl_handler(remDev);
if (cmgrHandler) {
TRACE(8,
"checker: id:%d state:%d mode:%d role:%d cmghdl:%p "
"sniffInterva:%d/%d IsAudioUp:%d",
i, btif_me_get_remote_device_state(remDev),
btif_me_get_remote_device_mode(remDev),
btif_me_get_remote_device_role(remDev), cmgrHandler,
btif_cmgr_get_cmgrhandler_sniff_Interval(cmgrHandler),
btif_cmgr_get_cmgrhandler_sniff_info(cmgrHandler)->maxInterval,
app_bt_device.hf_audio_state[i]);
DUMP8("0x%02x ", btif_me_get_remote_device_bdaddr(remDev)->address,
BTIF_BD_ADDR_SIZE);
TRACE(4, "remDev:%p a2dp State:%d hf_channel Connected:%d remDev:%p ",
remDev,
app_bt_device.a2dp_connected_stream[i]
? btif_a2dp_get_stream_state(
app_bt_device.a2dp_connected_stream[i])
: BTIF_A2DP_STREAM_STATE_CLOSED,
app_bt_device.hf_conn_flag[i],
app_bt_device.hf_conn_flag[i]
? (btif_remote_device_t *)btif_hf_cmgr_get_remote_device(
app_bt_device.hf_channel[i])
: NULL);
}
} else {
TRACE(1, "checker: id:%d remDev is NULL!", i);
}
#ifdef __AI_VOICE__
TRACE(1, "ai_setup_complete %d", app_ai_is_setup_complete());
#endif
#ifdef IS_MULTI_AI_ENABLED
TRACE(1, "current_spec %d", app_ai_manager_get_current_spec());
#endif
#if defined(__HSP_ENABLE__)
TRACE(2, "hs_channel Connected:%d remDev:%p ",
app_bt_device.hs_conn_flag[i],
app_bt_device.hs_channel[i].cmgrHandler.remDev);
#endif
TRACE(2, "%s btif_me_get_activeCons = %d", __func__,
btif_me_get_activeCons());
#ifdef HFP_DEBUG
hfp_call_state_checker();
#endif
}
#if defined(ENHANCED_STACK)
btif_me_cobuf_state_dump();
btif_me_hcibuff_state_dump();
#endif
#endif
osapi_unlock_stack();
return 0;
}
static uint8_t app_bt_get_devId_from_RemDev(btif_remote_device_t *remDev) {
uint8_t connectedDevId = 0;
for (uint8_t devId = 0; devId < BT_DEVICE_NUM; devId++) {
if (btif_me_enumerate_remote_devices(devId) == remDev) {
TRACE(2, "%s %d", __func__, devId);
connectedDevId = devId;
break;
}
}
return connectedDevId;
}
void app_bt_accessible_manager_process(const btif_event_t *Event) {
#if defined(IBRT)
// IBRT device's access mode will be controlled by UI
return;
#else
btif_event_type_t etype = btif_me_get_callback_event_type(Event);
#ifdef __BT_ONE_BRING_TWO__
static uint8_t opening_reconnect_cnf_cnt = 0;
// uint8_t disconnectedDevId =
// app_bt_get_devId_from_RemDev(btif_me_get_callback_event_rem_dev( Event));
if (app_bt_profile_connect_openreconnecting(NULL)) {
if (etype == BTIF_BTEVENT_LINK_CONNECT_CNF) {
opening_reconnect_cnf_cnt++;
}
if (record2_avalible) {
if (opening_reconnect_cnf_cnt < 2) {
return;
}
}
}
#endif
switch (etype) {
case BTIF_BTEVENT_LINK_CONNECT_CNF:
case BTIF_BTEVENT_LINK_CONNECT_IND:
TRACE(1, "BTEVENT_LINK_CONNECT_IND/CNF activeCons:%d",
btif_me_get_activeCons());
#if defined(__BTIF_EARPHONE__)
app_stop_10_second_timer(APP_PAIR_TIMER_ID);
#endif
#ifdef __BT_ONE_BRING_TWO__
if (btif_me_get_activeCons() == 0) {
#ifdef __EARPHONE_STAY_BOTH_SCAN__
app_bt_accessmode_set_req(BTIF_BT_DEFAULT_ACCESS_MODE_PAIR);
#else
app_bt_accessmode_set_req(BTIF_BAM_CONNECTABLE_ONLY);
#endif
} else if (btif_me_get_activeCons() == 1) {
app_bt_accessmode_set_req(BTIF_BAM_CONNECTABLE_ONLY);
} else if (btif_me_get_activeCons() >= 2) {
app_bt_accessmode_set_req(BTIF_BAM_NOT_ACCESSIBLE);
}
#else
if (btif_me_get_activeCons() == 0) {
#ifdef __EARPHONE_STAY_BOTH_SCAN__
app_bt_accessmode_set_req(BTIF_BT_DEFAULT_ACCESS_MODE_PAIR);
#else
app_bt_accessmode_set_req(BTIF_BAM_CONNECTABLE_ONLY);
#endif
} else if (btif_me_get_activeCons() >= 1) {
app_bt_accessmode_set_req(BTIF_BAM_NOT_ACCESSIBLE);
}
#endif
break;
case BTIF_BTEVENT_LINK_DISCONNECT:
TRACE(1, "DISCONNECT activeCons:%d", btif_me_get_activeCons());
#ifdef __EARPHONE_STAY_BOTH_SCAN__
#ifdef __BT_ONE_BRING_TWO__
if (btif_me_get_activeCons() == 0) {
app_bt_accessmode_set_req(BTIF_BT_DEFAULT_ACCESS_MODE_PAIR);
} else if (btif_me_get_activeCons() == 1) {
app_bt_accessmode_set_req(BTIF_BAM_CONNECTABLE_ONLY);
} else if (btif_me_get_activeCons() >= 2) {
app_bt_accessmode_set_req(BTIF_BAM_NOT_ACCESSIBLE);
}
#else
app_bt_accessmode_set_req(BTIF_BT_DEFAULT_ACCESS_MODE_PAIR);
#endif
#else
app_bt_accessmode_set_req(BTIF_BAM_CONNECTABLE_ONLY);
#endif
break;
#ifdef __BT_ONE_BRING_TWO__
case BTIF_BTEVENT_SCO_CONNECT_IND:
case BTIF_BTEVENT_SCO_CONNECT_CNF:
if (btif_me_get_activeCons() == 1) {
app_bt_accessmode_set_req(BTIF_BAM_NOT_ACCESSIBLE);
}
break;
case BTIF_BTEVENT_SCO_DISCONNECT:
if (btif_me_get_activeCons() == 1) {
app_bt_accessmode_set_req(BTIF_BAM_CONNECTABLE_ONLY);
}
break;
#endif
default:
break;
}
#endif
}
#define APP_BT_SWITCHROLE_LIMIT (2)
// #define __SET_OUR_AS_MASTER__
void app_bt_role_manager_process(const btif_event_t *Event) {
#if defined(IBRT) || defined(APP_LINEIN_A2DP_SOURCE) || \
defined(APP_I2S_A2DP_SOURCE) || defined(__APP_A2DP_SOURCE__)
return;
#else
static btif_remote_device_t *opRemDev = NULL;
static uint8_t switchrole_cnt = 0;
btif_remote_device_t *remDev = NULL;
btif_event_type_t etype = btif_me_get_callback_event_type(Event);
// on phone connecting
switch (etype) {
case BTIF_BTEVENT_LINK_CONNECT_IND:
if (btif_me_get_callback_event_err_code(Event) == BTIF_BEC_NO_ERROR) {
if (btif_me_get_activeCons() == 1) {
switch (btif_me_get_callback_event_rem_dev_role(Event)) {
#if defined(__SET_OUR_AS_MASTER__)
case BTIF_BCR_SLAVE:
case BTIF_BCR_PSLAVE:
#else
case BTIF_BCR_MASTER:
case BTIF_BCR_PMASTER:
#endif
TRACE(1, "CONNECT_IND try to role %p\n",
btif_me_get_callback_event_rem_dev(Event));
// curr connectrot try to role
opRemDev = btif_me_get_callback_event_rem_dev(Event);
switchrole_cnt = 0;
app_bt_Me_SetLinkPolicy(btif_me_get_callback_event_rem_dev(Event),
BTIF_BLP_MASTER_SLAVE_SWITCH |
BTIF_BLP_SNIFF_MODE);
break;
#if defined(__SET_OUR_AS_MASTER__)
case BTIF_BCR_MASTER:
case BTIF_BCR_PMASTER:
#else
case BTIF_BCR_SLAVE:
case BTIF_BCR_PSLAVE:
#endif
case BTIF_BCR_ANY:
case BTIF_BCR_UNKNOWN:
default:
TRACE(1, "CONNECT_IND disable role %p\n",
btif_me_get_callback_event_rem_dev(Event));
// disable roleswitch when 1 connect
app_bt_Me_SetLinkPolicy(btif_me_get_callback_event_rem_dev(Event),
BTIF_BLP_SNIFF_MODE);
break;
}
// set next connector to master
app_bt_ME_SetConnectionRole(BTIF_BCR_MASTER);
} else if (btif_me_get_activeCons() > 1) {
switch (btif_me_get_callback_event_rem_dev_role(Event)) {
case BTIF_BCR_MASTER:
case BTIF_BCR_PMASTER:
TRACE(1, "CONNECT_IND disable role %p\n",
btif_me_get_callback_event_rem_dev(Event));
// disable roleswitch
app_bt_Me_SetLinkPolicy(btif_me_get_callback_event_rem_dev(Event),
BTIF_BLP_SNIFF_MODE);
break;
case BTIF_BCR_SLAVE:
case BTIF_BCR_PSLAVE:
case BTIF_BCR_ANY:
case BTIF_BCR_UNKNOWN:
default:
// disconnect slave
TRACE(1, "CONNECT_IND disconnect slave %p\n",
btif_me_get_callback_event_rem_dev(Event));
app_bt_MeDisconnectLink(btif_me_get_callback_event_rem_dev(Event));
break;
}
// set next connector to master
app_bt_ME_SetConnectionRole(BTIF_BCR_MASTER);
}
}
break;
case BTIF_BTEVENT_LINK_CONNECT_CNF:
if (btif_me_get_activeCons() == 1) {
switch (btif_me_get_callback_event_rem_dev_role(Event)) {
#if defined(__SET_OUR_AS_MASTER__)
case BTIF_BCR_SLAVE:
case BTIF_BCR_PSLAVE:
#else
case BTIF_BCR_MASTER:
case BTIF_BCR_PMASTER:
#endif
TRACE(1, "CONNECT_CNF try to role %p\n",
btif_me_get_callback_event_rem_dev(Event));
// curr connectrot try to role
opRemDev = btif_me_get_callback_event_rem_dev(Event);
switchrole_cnt = 0;
app_bt_Me_SetLinkPolicy(btif_me_get_callback_event_rem_dev(Event),
BTIF_BLP_MASTER_SLAVE_SWITCH |
BTIF_BLP_SNIFF_MODE);
app_bt_ME_SwitchRole(btif_me_get_callback_event_rem_dev(Event));
break;
#if defined(__SET_OUR_AS_MASTER__)
case BTIF_BCR_MASTER:
case BTIF_BCR_PMASTER:
#else
case BTIF_BCR_SLAVE:
case BTIF_BCR_PSLAVE:
#endif
case BTIF_BCR_ANY:
case BTIF_BCR_UNKNOWN:
default:
TRACE(1, "CONNECT_CNF disable role %p\n",
btif_me_get_callback_event_rem_dev(Event));
// disable roleswitch
app_bt_Me_SetLinkPolicy(btif_me_get_callback_event_rem_dev(Event),
BTIF_BLP_SNIFF_MODE);
break;
}
// set next connector to master
app_bt_ME_SetConnectionRole(BTIF_BCR_MASTER);
} else if (btif_me_get_activeCons() > 1) {
switch (btif_me_get_callback_event_rem_dev_role(Event)) {
case BTIF_BCR_MASTER:
case BTIF_BCR_PMASTER:
TRACE(1, "CONNECT_CNF disable role %p\n",
btif_me_get_callback_event_rem_dev(Event));
// disable roleswitch
app_bt_Me_SetLinkPolicy(btif_me_get_callback_event_rem_dev(Event),
BTIF_BLP_SNIFF_MODE);
break;
case BTIF_BCR_SLAVE:
case BTIF_BCR_ANY:
case BTIF_BCR_UNKNOWN:
default:
// disconnect slave
TRACE(1, "CONNECT_CNF disconnect slave %p\n",
btif_me_get_callback_event_rem_dev(Event));
app_bt_MeDisconnectLink(btif_me_get_callback_event_rem_dev(Event));
break;
}
// set next connector to master
app_bt_ME_SetConnectionRole(BTIF_BCR_MASTER);
}
break;
case BTIF_BTEVENT_LINK_DISCONNECT:
if (opRemDev == btif_me_get_callback_event_rem_dev(Event)) {
opRemDev = NULL;
switchrole_cnt = 0;
}
if (btif_me_get_activeCons() == 0) {
for (uint8_t i = 0; i < BT_DEVICE_NUM; i++) {
if (app_bt_device.a2dp_connected_stream[i])
app_bt_A2DP_SetMasterRole(app_bt_device.a2dp_connected_stream[i],
FALSE);
app_bt_HF_SetMasterRole(app_bt_device.hf_channel[i], FALSE);
}
app_bt_ME_SetConnectionRole(BTIF_BCR_ANY);
} else if (btif_me_get_activeCons() == 1) {
// set next connector to master
app_bt_ME_SetConnectionRole(BTIF_BCR_MASTER);
}
break;
case BTIF_BTEVENT_ROLE_CHANGE:
if (opRemDev == btif_me_get_callback_event_rem_dev(Event)) {
switch (btif_me_get_callback_event_role_change_new_role(Event)) {
#if defined(__SET_OUR_AS_MASTER__)
case BTIF_BCR_SLAVE:
#else
case BTIF_BCR_MASTER:
#endif
if (++switchrole_cnt <= APP_BT_SWITCHROLE_LIMIT) {
app_bt_ME_SwitchRole(btif_me_get_callback_event_rem_dev(Event));
} else {
#if defined(__SET_OUR_AS_MASTER__)
TRACE(2, "ROLE TO MASTER FAILED remDev %p cnt:%d\n",
btif_me_get_callback_event_rem_dev(Event), switchrole_cnt);
#else
TRACE(2, "ROLE TO SLAVE FAILED remDev %p cnt:%d\n",
btif_me_get_callback_event_rem_dev(Event), switchrole_cnt);
#endif
opRemDev = NULL;
switchrole_cnt = 0;
}
break;
#if defined(__SET_OUR_AS_MASTER__)
case BTIF_BCR_MASTER:
TRACE(2, "ROLE TO MASTER SUCCESS remDev %p cnt:%d\n",
btif_me_get_callback_event_rem_dev(Event), switchrole_cnt);
#else
case BTIF_BCR_SLAVE:
TRACE(2, "ROLE TO SLAVE SUCCESS remDev %p cnt:%d\n",
btif_me_get_callback_event_rem_dev(Event), switchrole_cnt);
#endif
opRemDev = NULL;
switchrole_cnt = 0;
app_bt_Me_SetLinkPolicy(btif_me_get_callback_event_rem_dev(Event),
BTIF_BLP_SNIFF_MODE);
break;
case BTIF_BCR_ANY:
break;
case BTIF_BCR_UNKNOWN:
break;
default:
break;
}
}
if (btif_me_get_activeCons() > 1) {
uint8_t slave_cnt = 0;
for (uint8_t i = 0; i < BT_DEVICE_NUM; i++) {
remDev = btif_me_enumerate_remote_devices(i);
if (btif_me_get_current_role(remDev) == BTIF_BCR_SLAVE) {
slave_cnt++;
}
}
if (slave_cnt > 1) {
TRACE(1, "ROLE_CHANGE disconnect slave %p\n",
btif_me_get_callback_event_rem_dev(Event));
app_bt_MeDisconnectLink(btif_me_get_callback_event_rem_dev(Event));
}
}
break;
default:
break;
}
#endif
}
void app_bt_role_manager_process_dual_slave(const btif_event_t *Event) {
#if defined(IBRT) || defined(APP_LINEIN_A2DP_SOURCE) || \
defined(APP_I2S_A2DP_SOURCE) || defined(__APP_A2DP_SOURCE__)
return;
#else
static btif_remote_device_t *opRemDev = NULL;
static uint8_t switchrole_cnt = 0;
// btif_remote_device_t *remDev = NULL;
// on phone connecting
switch (btif_me_get_callback_event_type(Event)) {
case BTIF_BTEVENT_LINK_CONNECT_IND:
case BTIF_BTEVENT_LINK_CONNECT_CNF:
if (btif_me_get_callback_event_err_code(Event) == BTIF_BEC_NO_ERROR) {
switch (btif_me_get_callback_event_rem_dev_role(Event)) {
#if defined(__SET_OUR_AS_MASTER__)
case BTIF_BCR_SLAVE:
case BTIF_BCR_PSLAVE:
#else
case BTIF_BCR_MASTER:
case BTIF_BCR_PMASTER:
#endif
TRACE(1, "CONNECT_IND/CNF try to role %p\n",
btif_me_get_callback_event_rem_dev(Event));
opRemDev = btif_me_get_callback_event_rem_dev(Event);
switchrole_cnt = 0;
app_bt_Me_SetLinkPolicy(btif_me_get_callback_event_rem_dev(Event),
BTIF_BLP_MASTER_SLAVE_SWITCH |
BTIF_BLP_SNIFF_MODE);
app_bt_ME_SwitchRole(btif_me_get_callback_event_rem_dev(Event));
break;
#if defined(__SET_OUR_AS_MASTER__)
case BTIF_BCR_MASTER:
case BTIF_BCR_PMASTER:
#else
case BTIF_BCR_SLAVE:
case BTIF_BCR_PSLAVE:
#endif
case BTIF_BCR_ANY:
case BTIF_BCR_UNKNOWN:
default:
TRACE(1, "CONNECT_IND disable role %p\n",
btif_me_get_callback_event_rem_dev(Event));
app_bt_Me_SetLinkPolicy(btif_me_get_callback_event_rem_dev(Event),
BTIF_BLP_SNIFF_MODE);
break;
}
app_bt_ME_SetConnectionRole(BTIF_BCR_SLAVE);
}
break;
case BTIF_BTEVENT_LINK_DISCONNECT:
if (opRemDev == btif_me_get_callback_event_rem_dev(Event)) {
opRemDev = NULL;
switchrole_cnt = 0;
}
if (btif_me_get_activeCons() == 0) {
for (uint8_t i = 0; i < BT_DEVICE_NUM; i++) {
if (app_bt_device.a2dp_connected_stream[i])
app_bt_A2DP_SetMasterRole(app_bt_device.a2dp_connected_stream[i],
FALSE);
app_bt_HF_SetMasterRole(app_bt_device.hf_channel[i], FALSE);
}
app_bt_ME_SetConnectionRole(BTIF_BCR_ANY);
} else if (btif_me_get_activeCons() == 1) {
app_bt_ME_SetConnectionRole(BTIF_BCR_SLAVE);
}
break;
case BTIF_BTEVENT_ROLE_CHANGE:
if (opRemDev == btif_me_get_callback_event_rem_dev(Event)) {
switch (btif_me_get_callback_event_role_change_new_role(Event)) {
#if defined(__SET_OUR_AS_MASTER__)
case BTIF_BCR_SLAVE:
#else
case BTIF_BCR_MASTER:
#endif
if (++switchrole_cnt <= APP_BT_SWITCHROLE_LIMIT) {
TRACE(1, "ROLE_CHANGE try to role again: %d", switchrole_cnt);
app_bt_Me_SetLinkPolicy(btif_me_get_callback_event_rem_dev(Event),
BTIF_BLP_MASTER_SLAVE_SWITCH |
BTIF_BLP_SNIFF_MODE);
app_bt_ME_SwitchRole(btif_me_get_callback_event_rem_dev(Event));
} else {
#if defined(__SET_OUR_AS_MASTER__)
TRACE(2, "ROLE TO MASTER FAILED remDev %p cnt:%d\n",
btif_me_get_callback_event_rem_dev(Event), switchrole_cnt);
#else
TRACE(2, "ROLE TO SLAVE FAILED remDev %p cnt:%d\n",
btif_me_get_callback_event_rem_dev(Event), switchrole_cnt);
#endif
opRemDev = NULL;
switchrole_cnt = 0;
}
break;
#if defined(__SET_OUR_AS_MASTER__)
case BTIF_BCR_MASTER:
TRACE(2, "ROLE TO MASTER SUCCESS remDev %p cnt:%d\n",
btif_me_get_callback_event_rem_dev(Event), switchrole_cnt);
#else
case BTIF_BCR_SLAVE:
TRACE(2, "ROLE TO SLAVE SUCCESS remDev %p cnt:%d\n",
btif_me_get_callback_event_rem_dev(Event), switchrole_cnt);
#endif
opRemDev = NULL;
switchrole_cnt = 0;
// workaround for power reset opening reconnect sometime unsuccessfully
// in sniff mode, only after authentication completes, enable sniff
// mode.
opRemDev = btif_me_get_callback_event_rem_dev(Event);
if (btif_me_get_remote_device_auth_state(opRemDev) ==
BTIF_BAS_AUTHENTICATED) {
app_bt_Me_SetLinkPolicy(opRemDev, BTIF_BLP_SNIFF_MODE);
} else {
app_bt_Me_SetLinkPolicy(opRemDev, BTIF_BLP_DISABLE_ALL);
}
break;
case BTIF_BCR_ANY:
break;
case BTIF_BCR_UNKNOWN:
break;
default:
break;
}
}
break;
}
#endif
}
static int app_bt_sniff_manager_init(void) {
btif_sniff_info_t sniffInfo;
btif_remote_device_t *remDev = NULL;
for (uint8_t i = 0; i < BT_DEVICE_NUM; i++) {
remDev = btif_me_enumerate_remote_devices(i);
sniffInfo.maxInterval = BTIF_CMGR_SNIFF_MAX_INTERVAL;
sniffInfo.minInterval = BTIF_CMGR_SNIFF_MIN_INTERVAL;
sniffInfo.attempt = BTIF_CMGR_SNIFF_ATTEMPT;
sniffInfo.timeout = BTIF_CMGR_SNIFF_TIMEOUT;
app_bt_CMGR_SetSniffInfoToAllHandlerByRemDev(&sniffInfo, remDev);
app_bt_HF_EnableSniffMode(app_bt_device.hf_channel[i], FALSE);
#if defined(__HSP_ENABLE__)
app_bt_HS_EnableSniffMode(&app_bt_device.hs_channel[i], FALSE);
#endif
}
return 0;
}
void app_bt_sniff_config(btif_remote_device_t *remDev) {
btif_sniff_info_t sniffInfo;
sniffInfo.maxInterval = BTIF_CMGR_SNIFF_MAX_INTERVAL;
sniffInfo.minInterval = BTIF_CMGR_SNIFF_MIN_INTERVAL;
sniffInfo.attempt = BTIF_CMGR_SNIFF_ATTEMPT;
sniffInfo.timeout = BTIF_CMGR_SNIFF_TIMEOUT;
app_bt_CMGR_SetSniffInfoToAllHandlerByRemDev(&sniffInfo, remDev);
#if !defined(IBRT)
if (btif_me_get_activeCons() > 1) {
btif_remote_device_t *tmpRemDev = NULL;
btif_cmgr_handler_t *currbtif_cmgr_handler_t = NULL;
btif_cmgr_handler_t *otherbtif_cmgr_handler_t = NULL;
currbtif_cmgr_handler_t = btif_cmgr_get_conn_ind_handler(remDev);
for (uint8_t i = 0; i < BT_DEVICE_NUM; i++) {
tmpRemDev = btif_me_enumerate_remote_devices(i);
if (remDev != tmpRemDev && tmpRemDev != NULL) {
otherbtif_cmgr_handler_t = btif_cmgr_get_acl_handler(tmpRemDev);
if (otherbtif_cmgr_handler_t && currbtif_cmgr_handler_t) {
if (btif_cmgr_get_cmgrhandler_sniff_info(otherbtif_cmgr_handler_t)
->maxInterval ==
btif_cmgr_get_cmgrhandler_sniff_info(currbtif_cmgr_handler_t)
->maxInterval) {
sniffInfo.maxInterval =
btif_cmgr_get_cmgrhandler_sniff_info(otherbtif_cmgr_handler_t)
->maxInterval -
20;
sniffInfo.minInterval =
btif_cmgr_get_cmgrhandler_sniff_info(otherbtif_cmgr_handler_t)
->minInterval -
20;
sniffInfo.attempt = BTIF_CMGR_SNIFF_ATTEMPT;
sniffInfo.timeout = BTIF_CMGR_SNIFF_TIMEOUT;
app_bt_CMGR_SetSniffInfoToAllHandlerByRemDev(&sniffInfo, remDev);
}
}
break;
} else {
TRACE(3,
"%s:enumerate i:%d remDev is NULL, param remDev:%p, this may "
"cause error!",
__func__, i, remDev);
}
}
}
#endif
}
void app_bt_sniff_manager_process(const btif_event_t *Event) {
static btif_remote_device_t *opRemDev = NULL;
btif_remote_device_t *remDev = NULL;
btif_cmgr_handler_t *currbtif_cmgr_handler_t = NULL;
btif_cmgr_handler_t *otherbtif_cmgr_handler_t = NULL;
btif_sniff_info_t sniffInfo;
if (!besbt_cfg.sniff)
return;
switch (btif_me_get_callback_event_type(Event)) {
case BTIF_BTEVENT_LINK_CONNECT_IND:
break;
case BTIF_BTEVENT_LINK_CONNECT_CNF:
break;
case BTIF_BTEVENT_LINK_DISCONNECT:
if (opRemDev == btif_me_get_callback_event_rem_dev(Event)) {
opRemDev = NULL;
}
sniffInfo.maxInterval = BTIF_CMGR_SNIFF_MAX_INTERVAL;
sniffInfo.minInterval = BTIF_CMGR_SNIFF_MIN_INTERVAL;
sniffInfo.attempt = BTIF_CMGR_SNIFF_ATTEMPT;
sniffInfo.timeout = BTIF_CMGR_SNIFF_TIMEOUT;
app_bt_CMGR_SetSniffInfoToAllHandlerByRemDev(
&sniffInfo, btif_me_get_callback_event_rem_dev(Event));
break;
case BTIF_BTEVENT_MODE_CHANGE:
/*
if(Event->p.modeChange.curMode == BLM_SNIFF_MODE){
currbtif_cmgr_handler_t =
btif_cmgr_get_acl_handler(btif_me_get_callback_event_rem_dev( Event)); if
(Event->p.modeChange.interval > CMGR_SNIFF_MAX_INTERVAL){ if (!opRemDev){
opRemDev = currbtif_cmgr_handler_t->remDev;
}
currbtif_cmgr_handler_t->sniffInfo.maxInterval =
CMGR_SNIFF_MAX_INTERVAL; currbtif_cmgr_handler_t->sniffInfo.minInterval =
CMGR_SNIFF_MIN_INTERVAL; currbtif_cmgr_handler_t->sniffInfo.attempt =
CMGR_SNIFF_ATTEMPT; currbtif_cmgr_handler_t->sniffInfo.timeout =
CMGR_SNIFF_TIMEOUT;
app_bt_CMGR_SetSniffInfoToAllHandlerByRemDev(&currbtif_cmgr_handler_t->sniffInfo,btif_me_get_callback_event_rem_dev(
Event)); app_bt_ME_StopSniff(currbtif_cmgr_handler_t->remDev); }else{ if
(currbtif_cmgr_handler_t){ currbtif_cmgr_handler_t->sniffInfo.maxInterval =
Event->p.modeChange.interval; currbtif_cmgr_handler_t->sniffInfo.minInterval
= Event->p.modeChange.interval; currbtif_cmgr_handler_t->sniffInfo.attempt =
CMGR_SNIFF_ATTEMPT; currbtif_cmgr_handler_t->sniffInfo.timeout =
CMGR_SNIFF_TIMEOUT;
app_bt_CMGR_SetSniffInfoToAllHandlerByRemDev(&currbtif_cmgr_handler_t->sniffInfo,btif_me_get_callback_event_rem_dev(
Event));
}
if (btif_me_get_activeCons() > 1){
for (uint8_t i=0; i<BT_DEVICE_NUM; i++){
remDev = btif_me_enumerate_remote_devices(i);
if (btif_me_get_callback_event_rem_dev( Event) != remDev){
otherbtif_cmgr_handler_t =
btif_cmgr_get_acl_handler(remDev); if (otherbtif_cmgr_handler_t){ if
(otherbtif_cmgr_handler_t->sniffInfo.maxInterval ==
currbtif_cmgr_handler_t->sniffInfo.maxInterval){ if
(btif_me_get_current_mode(remDev) == BLM_ACTIVE_MODE){
otherbtif_cmgr_handler_t->sniffInfo.maxInterval
-= 20; otherbtif_cmgr_handler_t->sniffInfo.minInterval -= 20;
otherbtif_cmgr_handler_t->sniffInfo.attempt
= CMGR_SNIFF_ATTEMPT; otherbtif_cmgr_handler_t->sniffInfo.timeout =
CMGR_SNIFF_TIMEOUT;
app_bt_CMGR_SetSniffInfoToAllHandlerByRemDev(&otherbtif_cmgr_handler_t->sniffInfo,
remDev); TRACE(1,"reconfig sniff other RemDev:%x\n", remDev); }else if
(btif_me_get_current_mode(remDev) == BLM_SNIFF_MODE){ need_reconfig = true;
}
}
}
break;
}
}
}
if (need_reconfig){
opRemDev = remDev;
if (currbtif_cmgr_handler_t){
currbtif_cmgr_handler_t->sniffInfo.maxInterval -= 20;
currbtif_cmgr_handler_t->sniffInfo.minInterval -= 20;
currbtif_cmgr_handler_t->sniffInfo.attempt =
CMGR_SNIFF_ATTEMPT; currbtif_cmgr_handler_t->sniffInfo.timeout =
CMGR_SNIFF_TIMEOUT;
app_bt_CMGR_SetSniffInfoToAllHandlerByRemDev(&currbtif_cmgr_handler_t->sniffInfo,
currbtif_cmgr_handler_t->remDev);
}
app_bt_ME_StopSniff(currbtif_cmgr_handler_t->remDev);
TRACE(1,"reconfig sniff setup op opRemDev:%x\n", opRemDev);
}
}
}
if (Event->p.modeChange.curMode == BLM_ACTIVE_MODE){
if (opRemDev ==btif_me_get_callback_event_rem_dev( Event)){
TRACE(1,"reconfig sniff op opRemDev:%x\n", opRemDev);
opRemDev = NULL;
currbtif_cmgr_handler_t =
btif_cmgr_get_acl_handler(btif_me_get_callback_event_rem_dev( Event)); if
(currbtif_cmgr_handler_t){
app_bt_CMGR_SetSniffTimer(currbtif_cmgr_handler_t, NULL,
CMGR_SNIFF_TIMER);
}
}
}
*/
break;
case BTIF_BTEVENT_ACL_DATA_ACTIVE:
btif_cmgr_handler_t *cmgrHandler;
/* Start the sniff timer */
cmgrHandler =
btif_cmgr_get_acl_handler(btif_me_get_callback_event_rem_dev(Event));
if (cmgrHandler)
app_bt_CMGR_SetSniffTimer(cmgrHandler, NULL, BTIF_CMGR_SNIFF_TIMER);
break;
case BTIF_BTEVENT_SCO_CONNECT_IND:
case BTIF_BTEVENT_SCO_CONNECT_CNF:
TRACE(1, "BTEVENT_SCO_CONNECT_IND/CNF cur_remDev = %p",
btif_me_get_callback_event_rem_dev(Event));
currbtif_cmgr_handler_t = btif_cmgr_get_conn_ind_handler(
btif_me_get_callback_event_rem_dev(Event));
app_bt_Me_SetLinkPolicy(
btif_me_get_callback_event_sco_connect_rem_dev(Event),
BTIF_BLP_DISABLE_ALL);
if (btif_me_get_activeCons() > 1) {
for (uint8_t i = 0; i < BT_DEVICE_NUM; i++) {
remDev = btif_me_enumerate_remote_devices(i);
TRACE(1, "other_remDev = %p", remDev);
if (btif_me_get_callback_event_rem_dev(Event) == remDev) {
continue;
}
otherbtif_cmgr_handler_t = btif_cmgr_get_conn_ind_handler(remDev);
if (otherbtif_cmgr_handler_t) {
if (btif_cmgr_is_link_up(otherbtif_cmgr_handler_t)) {
if (btif_me_get_current_mode(remDev) == BTIF_BLM_ACTIVE_MODE) {
TRACE(0, "other dev disable sniff");
app_bt_Me_SetLinkPolicy(remDev, BTIF_BLP_DISABLE_ALL);
} else if (btif_me_get_current_mode(remDev) ==
BTIF_BLM_SNIFF_MODE) {
TRACE(0, " ohter dev exit & disable sniff");
app_bt_ME_StopSniff(remDev);
app_bt_Me_SetLinkPolicy(remDev, BTIF_BLP_DISABLE_ALL);
}
}
}
#if defined(HFP_NO_PRERMPT)
TRACE(2, "cur_audio = %d other_audio = %d",
btif_cmgr_is_audio_up(currbtif_cmgr_handler_t),
btif_cmgr_is_audio_up(otherbtif_cmgr_handler_t));
if ((btif_cmgr_is_audio_up(otherbtif_cmgr_handler_t) == true) &&
(btif_cmgr_is_audio_up(currbtif_cmgr_handler_t) == true)
/*(btapp_hfp_get_call_active()!=0)*/) {
btif_cmgr_remove_audio_link(currbtif_cmgr_handler_t);
app_bt_Me_switch_sco(btif_cmgr_get_sco_connect_sco_Hcihandler(
otherbtif_cmgr_handler_t));
}
#endif
}
}
break;
case BTIF_BTEVENT_SCO_DISCONNECT:
app_bt_profile_reconnect_pending_process();
if (a2dp_is_music_ongoing()) {
break;
}
if (btif_me_get_activeCons() == 1) {
app_bt_Me_SetLinkPolicy(
btif_me_get_callback_event_sco_connect_rem_dev(Event),
BTIF_BLP_SNIFF_MODE);
} else {
uint8_t i;
for (i = 0; i < BT_DEVICE_NUM; i++) {
remDev = btif_me_enumerate_remote_devices(i);
if (btif_me_get_callback_event_rem_dev(Event) == remDev) {
break;
}
}
/*
if(i==0)
remDev = btif_me_enumerate_remote_devices(1);
else if(i==1)
remDev = btif_me_enumerate_remote_devices(0);
else
ASSERT(0,"error other remotedevice!!!"); */
otherbtif_cmgr_handler_t = btif_cmgr_get_conn_ind_handler(remDev);
currbtif_cmgr_handler_t = btif_cmgr_get_conn_ind_handler(
btif_me_get_callback_event_rem_dev(Event));
TRACE(4, "SCO_DISCONNECT:%d/%d %p/%p\n",
btif_cmgr_is_audio_up(currbtif_cmgr_handler_t),
btif_cmgr_is_audio_up(otherbtif_cmgr_handler_t),
btif_cmgr_get_cmgrhandler_remdev(currbtif_cmgr_handler_t),
btif_me_get_callback_event_rem_dev(Event));
if (otherbtif_cmgr_handler_t) {
if (!btif_cmgr_is_audio_up(otherbtif_cmgr_handler_t)) {
TRACE(0, "enable sniff to all\n");
app_bt_Me_SetLinkPolicy(
btif_me_get_callback_event_sco_connect_rem_dev(Event),
BTIF_BLP_SNIFF_MODE);
app_bt_Me_SetLinkPolicy(
btif_cmgr_get_cmgrhandler_remdev(otherbtif_cmgr_handler_t),
BTIF_BLP_SNIFF_MODE);
}
} else {
app_bt_Me_SetLinkPolicy(
btif_me_get_callback_event_sco_connect_rem_dev(Event),
BTIF_BLP_SNIFF_MODE);
}
}
break;
default:
break;
}
}
APP_BT_GOLBAL_HANDLE_HOOK_HANDLER
app_bt_global_handle_hook_handler[APP_BT_GOLBAL_HANDLE_HOOK_USER_QTY] = {0};
void app_bt_global_handle_hook(const btif_event_t *Event) {
uint8_t i;
for (i = 0; i < APP_BT_GOLBAL_HANDLE_HOOK_USER_QTY; i++) {
if (app_bt_global_handle_hook_handler[i])
app_bt_global_handle_hook_handler[i](Event);
}
}
int app_bt_global_handle_hook_set(enum APP_BT_GOLBAL_HANDLE_HOOK_USER_T user,
APP_BT_GOLBAL_HANDLE_HOOK_HANDLER handler) {
app_bt_global_handle_hook_handler[user] = handler;
return 0;
}
APP_BT_GOLBAL_HANDLE_HOOK_HANDLER
app_bt_global_handle_hook_get(enum APP_BT_GOLBAL_HANDLE_HOOK_USER_T user) {
return app_bt_global_handle_hook_handler[user];
}
extern uint8_t once_event_case;
extern bool IsMobileLinkLossing;
extern void startonce_event(int ms, uint8_t once_event_pram);
extern void a2dp_update_music_link(void);
/////There is a device connected, so stop PAIR_TIMER and POWEROFF_TIMER of
/// earphone.
btif_handler app_bt_handler;
void app_bt_global_handle(const btif_event_t *Event) {
switch (btif_me_get_callback_event_type(Event)) {
case BTIF_BTEVENT_HCI_INITIALIZED:
break;
#if defined(IBRT)
case BTIF_BTEVENT_HCI_COMMAND_SENT:
return;
#else
case BTIF_BTEVENT_HCI_COMMAND_SENT:
case BTIF_BTEVENT_ACL_DATA_NOT_ACTIVE:
return;
case BTIF_BTEVENT_ACL_DATA_ACTIVE:
btif_cmgr_handler_t *cmgrHandler;
/* Start the sniff timer */
cmgrHandler =
btif_cmgr_get_acl_handler(btif_me_get_callback_event_rem_dev(Event));
if (cmgrHandler)
app_bt_CMGR_SetSniffTimer(cmgrHandler, NULL, BTIF_CMGR_SNIFF_TIMER);
return;
#endif
case BTIF_BTEVENT_AUTHENTICATED:
TRACE(1, "[BTEVENT] HANDER AUTH error=%x",
btif_me_get_callback_event_err_code(Event));
// after authentication completes, re-enable sniff mode.
if (btif_me_get_callback_event_err_code(Event) == BTIF_BEC_NO_ERROR) {
app_bt_Me_SetLinkPolicy(btif_me_get_callback_event_rem_dev(Event),
BTIF_BLP_SNIFF_MODE);
} else if (btif_me_get_callback_event_err_code(Event) ==
BTIF_BEC_AUTHENTICATE_FAILURE) {
// auth failed should clear nv record link key
bt_bdaddr_t *bd_ddr = btif_me_get_callback_event_rem_dev_bd_addr(Event);
btif_device_record_t record;
if (ddbif_find_record(bd_ddr, &record) == BT_STS_SUCCESS) {
ddbif_delete_record(&record.bdAddr);
#if defined(IBRT)
ibrt_ctrl_t *p_ibrt_ctrl = app_tws_ibrt_get_bt_ctrl_ctx();
if (!p_ibrt_ctrl->mobile_pair_canceled)
#endif
{
memset(&record, 0, sizeof(record));
record.bdAddr = *bd_ddr;
ddbif_add_record(&record);
}
nv_record_flash_flush();
}
}
break;
}
// trace filter
switch (btif_me_get_callback_event_type(Event)) {
case BTIF_BTEVENT_HCI_COMMAND_SENT:
case BTIF_BTEVENT_ACL_DATA_NOT_ACTIVE:
case BTIF_BTEVENT_ACL_DATA_ACTIVE:
break;
default:
TRACE(1, "[BTEVENT] evt = %d", btif_me_get_callback_event_type(Event));
break;
}
switch (btif_me_get_callback_event_type(Event)) {
case BTIF_BTEVENT_LINK_CONNECT_IND:
hfp_reconnecting_timer_stop_callback(Event);
case BTIF_BTEVENT_LINK_CONNECT_CNF:
#ifdef __BT_ONE_BRING_TWO__
if (bt_drv_reg_op_get_reconnecting_flag()) {
bt_drv_reg_op_clear_reconnecting_flag();
if (a2dp_is_music_ongoing())
a2dp_update_music_link();
}
#endif
if (BTIF_BEC_NO_ERROR == btif_me_get_callback_event_err_code(Event)) {
connectedMobile = btif_me_get_callback_event_rem_dev(Event);
uint8_t connectedDevId = app_bt_get_devId_from_RemDev(connectedMobile);
app_bt_set_connecting_profiles_state(connectedDevId);
TRACE(1, "MEC(pendCons) is %d", btif_me_get_pendCons());
app_bt_stay_active_rem_dev(btif_me_get_callback_event_rem_dev(Event));
#ifdef __BT_ONE_BRING_TWO__
btif_remote_device_t *remote_dev =
btif_me_get_callback_event_rem_dev(Event);
uint16_t conn_handle = btif_me_get_remote_device_hci_handle(remote_dev);
btif_me_qos_set_up(conn_handle);
#endif
#if (defined(__AI_VOICE__) || defined(BISTO_ENABLED))
app_ai_if_mobile_connect_handle(
btif_me_get_callback_event_rem_dev_bd_addr(Event));
#endif
}
TRACE(4,
"[BTEVENT] CONNECT_IND/CNF evt:%d errCode:0x%0x newRole:%d "
"activeCons:%d",
btif_me_get_callback_event_type(Event),
btif_me_get_callback_event_err_code(Event),
btif_me_get_callback_event_rem_dev_role(Event),
btif_me_get_activeCons());
DUMP8("%02x ", btif_me_get_callback_event_rem_dev_bd_addr(Event),
BTIF_BD_ADDR_SIZE);
#if defined(__BTIF_EARPHONE__) && defined(__BTIF_AUTOPOWEROFF__)
if (btif_me_get_activeCons() == 0) {
app_start_10_second_timer(APP_POWEROFF_TIMER_ID);
} else {
app_stop_10_second_timer(APP_POWEROFF_TIMER_ID);
}
#endif
#if defined(__BT_ONE_BRING_TWO__) || defined(IBRT)
if (btif_me_get_activeCons() > 2) {
TRACE(1, "CONNECT_IND/CNF activeCons:%d so disconnect it",
btif_me_get_activeCons());
app_bt_MeDisconnectLink(btif_me_get_callback_event_rem_dev(Event));
}
#else
if (btif_me_get_activeCons() > 1) {
TRACE(1, "CONNECT_IND/CNF activeCons:%d so disconnect it",
btif_me_get_activeCons());
app_bt_MeDisconnectLink(btif_me_get_callback_event_rem_dev(Event));
}
#endif
break;
case BTIF_BTEVENT_LINK_DISCONNECT: {
connectedMobile = btif_me_get_callback_event_rem_dev(Event);
uint8_t disconnectedDevId = app_bt_get_devId_from_RemDev(connectedMobile);
connectedMobile = NULL;
app_bt_clear_connecting_profiles_state(disconnectedDevId);
btif_remote_device_t *remote_dev =
btif_me_get_callback_event_disconnect_rem_dev(Event);
if (remote_dev) {
uint16_t conhdl = btif_me_get_remote_device_hci_handle(remote_dev);
bt_drv_reg_op_acl_tx_silence_clear(conhdl);
bt_drv_hwspi_select(conhdl - 0x80, 0);
}
TRACE(5,
"[BTEVENT] DISCONNECT evt = %d encryptState:%d reason:0x%02x/0x%02x "
"activeCons:%d",
btif_me_get_callback_event_type(Event),
btif_me_get_remote_sevice_encrypt_state(
btif_me_get_callback_event_rem_dev(Event)),
btif_me_get_remote_device_disc_reason_saved(
btif_me_get_callback_event_rem_dev(Event)),
btif_me_get_remote_device_disc_reason(
btif_me_get_callback_event_rem_dev(Event)),
btif_me_get_activeCons());
DUMP8("%02x ", btif_me_get_callback_event_rem_dev_bd_addr(Event),
BTIF_BD_ADDR_SIZE);
#ifdef CHIP_BEST2000
bt_drv_patch_force_disconnect_ack();
#endif
// disconnect from reconnect connection, and the HF don't connect successful
// once (whitch will release the saved_reconnect_mode ). so we are reconnect
// fail with remote link key loss.
// goto pairing.
// reason 07 maybe from the controller's error .
// 05 auth error
// 16 io cap reject.
#if defined(__BTIF_EARPHONE__) && defined(__BTIF_AUTOPOWEROFF__)
if (btif_me_get_activeCons() == 0) {
app_start_10_second_timer(APP_POWEROFF_TIMER_ID);
}
#endif
#if defined(BISTO_ENABLED)
gsound_custom_bt_link_disconnected_handler(
btif_me_get_callback_event_rem_dev_bd_addr(Event)->address);
#endif
#if defined(__AI_VOICE__)
app_ai_mobile_disconnect_handle(
btif_me_get_callback_event_rem_dev_bd_addr(Event));
#endif
#ifdef __IAG_BLE_INCLUDE__
// start BLE adv
app_ble_force_switch_adv(BLE_SWITCH_USER_BT_CONNECT, true);
#endif
#ifdef BTIF_DIP_DEVICE
btif_dip_clear(remote_dev);
#endif
app_bt_active_mode_reset(disconnectedDevId);
#ifdef GFPS_ENABLED
app_gfps_handling_on_mobile_link_disconnection(
btif_me_get_callback_event_rem_dev(Event));
#endif
#if !defined(IBRT) && defined(BT_XTAL_SYNC_NO_RESET)
bt_term_xtal_sync_default();
#endif
break;
}
case BTIF_BTEVENT_ROLE_CHANGE:
TRACE(3,
"[BTEVENT] ROLE_CHANGE eType:0x%x errCode:0x%x newRole:%d "
"activeCons:%d",
btif_me_get_callback_event_type(Event),
btif_me_get_callback_event_err_code(Event),
btif_me_get_callback_event_role_change_new_role(Event),
btif_me_get_activeCons());
break;
case BTIF_BTEVENT_MODE_CHANGE:
TRACE(4,
"[BTEVENT] MODE_CHANGE evt:%d errCode:0x%0x curMode=0x%0x, "
"interval=%d ",
btif_me_get_callback_event_type(Event),
btif_me_get_callback_event_err_code(Event),
btif_me_get_callback_event_mode_change_curMode(Event),
btif_me_get_callback_event_mode_change_interval(Event));
DUMP8("%02x ", btif_me_get_callback_event_rem_dev_bd_addr(Event),
BTIF_BD_ADDR_SIZE);
break;
case BTIF_BTEVENT_ACCESSIBLE_CHANGE:
TRACE(3, "[BTEVENT] ACCESSIBLE_CHANGE evt:%d errCode:0x%0x aMode=0x%0x",
btif_me_get_callback_event_type(Event),
btif_me_get_callback_event_err_code(Event),
btif_me_get_callback_event_a_mode(Event));
#if !defined(IBRT)
if (app_is_access_mode_set_pending()) {
app_set_pending_access_mode();
} else {
if (BTIF_BEC_NO_ERROR != btif_me_get_callback_event_err_code(Event)) {
app_retry_setting_access_mode();
}
}
#endif
break;
case BTIF_BTEVENT_LINK_POLICY_CHANGED: {
BT_SET_LINKPOLICY_REQ_T *pReq = app_bt_pop_pending_set_linkpolicy();
if (NULL != pReq) {
app_bt_Me_SetLinkPolicy(pReq->remDev, pReq->policy);
}
break;
}
case BTIF_BTEVENT_DEFAULT_LINK_POLICY_CHANGED: {
TRACE(0, "[BTEVENT] DEFAULT_LINK_POLICY_CHANGED-->BT_STACK_INITIALIZED");
app_notify_stack_ready(STACK_READY_BT);
break;
}
case BTIF_BTEVENT_NAME_RESULT: {
uint8_t *ptrName;
uint8_t nameLen;
nameLen = btif_me_get_callback_event_remote_dev_name(Event, &ptrName);
TRACE(1, "[BTEVENT] NAME_RESULT name len %d", nameLen);
if (nameLen > 0) {
TRACE(1, "remote dev name: %s", ptrName);
}
// return;
}
default:
break;
}
#ifdef MULTIPOINT_DUAL_SLAVE
app_bt_role_manager_process_dual_slave(Event);
#else
app_bt_role_manager_process(Event);
#endif
app_bt_accessible_manager_process(Event);
#if !defined(IBRT)
app_bt_sniff_manager_process(Event);
#endif
app_bt_global_handle_hook(Event);
#if defined(IBRT)
app_tws_ibrt_global_callback(Event);
#endif
}
#include "app_bt_media_manager.h"
osTimerId bt_sco_recov_timer = NULL;
static void bt_sco_recov_timer_handler(void const *param);
osTimerDef(BT_SCO_RECOV_TIMER,
(void (*)(void const *))bt_sco_recov_timer_handler); // define timers
void hfp_reconnect_sco(uint8_t flag);
static void bt_sco_recov_timer_handler(void const *param) {
TRACE(1, "%s", __func__);
hfp_reconnect_sco(0);
}
static void bt_sco_recov_timer_start() {
osTimerStop(bt_sco_recov_timer);
osTimerStart(bt_sco_recov_timer, 2500);
}
enum {
SCO_DISCONNECT_RECONN_START,
SCO_DISCONNECT_RECONN_RUN,
SCO_DISCONNECT_RECONN_NONE,
};
static uint8_t sco_reconnect_status = SCO_DISCONNECT_RECONN_NONE;
void hfp_reconnect_sco(uint8_t set) {
TRACE(3, "%s cur_chl_id=%d reconnect_status =%d", __func__,
app_bt_device.curr_hf_channel_id, sco_reconnect_status);
if (set == 1) {
sco_reconnect_status = SCO_DISCONNECT_RECONN_START;
}
if (sco_reconnect_status == SCO_DISCONNECT_RECONN_START) {
app_audio_manager_sendrequest(APP_BT_STREAM_MANAGER_STOP, BT_STREAM_VOICE,
app_bt_device.curr_hf_channel_id,
MAX_RECORD_NUM);
app_bt_HF_DisconnectAudioLink(
app_bt_device.hf_channel[app_bt_device.curr_hf_channel_id]);
sco_reconnect_status = SCO_DISCONNECT_RECONN_RUN;
bt_sco_recov_timer_start();
} else if (sco_reconnect_status == SCO_DISCONNECT_RECONN_RUN) {
app_bt_HF_CreateAudioLink(
app_bt_device.hf_channel[app_bt_device.curr_hf_channel_id]);
sco_reconnect_status = SCO_DISCONNECT_RECONN_NONE;
}
}
void app_bt_global_handle_init(void) {
btif_event_mask_t mask = BTIF_BEM_NO_EVENTS;
btif_me_init_handler(&app_bt_handler);
app_bt_handler.callback = app_bt_global_handle;
btif_me_register_global_handler(&app_bt_handler);
#if defined(IBRT)
btif_me_register_accept_handler(&app_bt_handler);
#endif
#ifdef IBRT_SEARCH_UI
app_bt_global_handle_hook_set(APP_BT_GOLBAL_HANDLE_HOOK_USER_0,
app_bt_manager_ibrt_role_process);
#endif
mask |= BTIF_BEM_ROLE_CHANGE | BTIF_BEM_SCO_CONNECT_CNF |
BTIF_BEM_SCO_DISCONNECT | BTIF_BEM_SCO_CONNECT_IND;
mask |= BTIF_BEM_AUTHENTICATED;
mask |= BTIF_BEM_LINK_CONNECT_IND;
mask |= BTIF_BEM_LINK_DISCONNECT;
mask |= BTIF_BEM_LINK_CONNECT_CNF;
mask |= BTIF_BEM_ACCESSIBLE_CHANGE;
mask |= BTIF_BEM_ENCRYPTION_CHANGE;
mask |= BTIF_BEM_SIMPLE_PAIRING_COMPLETE;
#if (defined(__BT_ONE_BRING_TWO__) || defined(IBRT))
mask |= BTIF_BEM_MODE_CHANGE;
#endif
mask |= BTIF_BEM_LINK_POLICY_CHANGED;
app_bt_ME_SetConnectionRole(BTIF_BCR_ANY);
for (uint8_t i = 0; i < BT_DEVICE_NUM; i++) {
app_bt_A2DP_SetMasterRole(app_bt_device.a2dp_stream[i]->a2dp_stream, FALSE);
#if defined(A2DP_LHDC_ON)
app_bt_A2DP_SetMasterRole(app_bt_device.a2dp_lhdc_stream[i]->a2dp_stream,
FALSE);
#endif
#if defined(A2DP_AAC_ON)
app_bt_A2DP_SetMasterRole(app_bt_device.a2dp_aac_stream[i]->a2dp_stream,
FALSE);
#endif
#if defined(A2DP_SCALABLE_ON)
app_bt_A2DP_SetMasterRole(
app_bt_device.a2dp_scalable_stream[i]->a2dp_stream, FALSE);
#endif
#if defined(A2DP_LDAC_ON)
app_bt_A2DP_SetMasterRole(app_bt_device.a2dp_ldac_stream[i]->a2dp_stream,
FALSE);
#endif
app_bt_HF_SetMasterRole(app_bt_device.hf_channel[i], FALSE);
#if defined(__HSP_ENABLE__)
HS_SetMasterRole(&app_bt_device.hs_channel[i], FALSE);
#endif
}
btif_me_set_event_mask(&app_bt_handler, mask);
app_bt_sniff_manager_init();
app_bt_accessmode_timer =
osTimerCreate(osTimer(APP_BT_ACCESSMODE_TIMER), osTimerOnce,
&app_bt_accessmode_timer_argument);
bt_sco_recov_timer =
osTimerCreate(osTimer(BT_SCO_RECOV_TIMER), osTimerOnce, NULL);
}
void app_bt_send_request(uint32_t message_id, uint32_t param0, uint32_t param1,
uint32_t ptr) {
APP_MESSAGE_BLOCK msg;
msg.mod_id = APP_MODUAL_BT;
msg.msg_body.message_id = message_id;
msg.msg_body.message_Param0 = param0;
msg.msg_body.message_Param1 = param1;
msg.msg_body.message_ptr = ptr;
app_mailbox_put(&msg);
}
extern void app_start_10_second_timer(uint8_t timer_id);
static int app_bt_handle_process(APP_MESSAGE_BODY *msg_body) {
btif_accessible_mode_t old_access_mode;
switch (msg_body->message_id) {
case APP_BT_REQ_ACCESS_MODE_SET:
old_access_mode = g_bt_access_mode;
app_bt_accessmode_set(msg_body->message_Param0);
if (msg_body->message_Param0 == BTIF_BAM_GENERAL_ACCESSIBLE &&
old_access_mode != BTIF_BAM_GENERAL_ACCESSIBLE) {
// app_status_indication_set(APP_STATUS_INDICATION_BOTHSCAN);
#ifdef MEDIA_PLAYER_SUPPORT
app_voice_report(APP_STATUS_INDICATION_BOTHSCAN, 0);
#endif
app_start_10_second_timer(APP_PAIR_TIMER_ID);
} else {
// app_status_indication_set(APP_STATUS_INDICATION_PAGESCAN);
}
break;
default:
break;
}
return 0;
}
void *app_bt_profile_active_store_ptr_get(uint8_t *bdAddr) {
static btdevice_profile device_profile = {true, false, true, 0};
btdevice_profile *ptr;
nvrec_btdevicerecord *record = NULL;
if (!nv_record_btdevicerecord_find((bt_bdaddr_t *)bdAddr, &record)) {
uint32_t lock = nv_record_pre_write_operation();
ptr = &(record->device_plf);
DUMP8("0x%02x ", bdAddr, BTIF_BD_ADDR_SIZE);
TRACE(5, "%s hfp_act:%d hsp_act:%d a2dp_act:0x%x codec_type=%x", __func__,
ptr->hfp_act, ptr->hsp_act, ptr->a2dp_act, ptr->a2dp_codectype);
/* always need connect a2dp and hfp */
ptr->hfp_act = true;
ptr->a2dp_act = true;
nv_record_post_write_operation(lock);
} else {
ptr = &device_profile;
TRACE(1, "%s default", __func__);
}
return (void *)ptr;
}
static void app_bt_profile_reconnect_timehandler(void const *param);
extern void startonce_delay_event_Timer_(int ms);
osTimerDef(BT_PROFILE_CONNECT_TIMER0,
app_bt_profile_reconnect_timehandler); // define timers
#ifdef __BT_ONE_BRING_TWO__
osTimerDef(BT_PROFILE_CONNECT_TIMER1, app_bt_profile_reconnect_timehandler);
#endif
#ifdef __AUTO_CONNECT_OTHER_PROFILE__
static void app_bt_profile_connect_hf_retry_handler(void) {
struct app_bt_profile_manager *bt_profile_manager_p =
(struct app_bt_profile_manager *)param;
if (MEC(pendCons) > 0) {
TRACE(1, "Former link is not down yet, reset the timer %s.", __FUNCTION__);
osTimerStart(bt_profile_manager_p->connect_timer,
APP_BT_PROFILE_RECONNECT_RETRY_INTERVAL_MS);
} else {
app_bt_precheck_before_starting_connecting(
bt_profile_manager_p->has_connected);
if (bt_profile_manager_p->hfp_connect !=
bt_profile_connect_status_success) {
app_bt_HF_CreateServiceLink(bt_profile_manager_p->chan,
&bt_profile_manager_p->rmt_addr);
}
}
}
static void app_bt_profile_connect_hf_retry_timehandler(void const *param) {
app_bt_start_custom_function_in_bt_thread(
0, 0, (uint32_t)app_bt_profile_connect_hf_retry_handler);
}
#if defined(__HSP_ENABLE__)
static void app_bt_profile_connect_hs_retry_timehandler(void const *param) {
struct app_bt_profile_manager *bt_profile_manager_p =
(struct app_bt_profile_manager *)param;
if (MEC(pendCons) > 0) {
if (bt_profile_manager_p->reconnect_cnt <
APP_BT_PROFILE_OPENNING_RECONNECT_RETRY_LIMIT_CNT) {
bt_profile_manager_p->reconnect_cnt++;
}
TRACE(1, "Former link is not down yet, reset the timer %s.", __FUNCTION__);
osTimerStart(bt_profile_manager_p->connect_timer,
BTIF_BT_DEFAULT_PAGE_TIMEOUT_IN_MS +
APP_BT_PROFILE_RECONNECT_RETRY_INTERVAL_MS);
} else {
if (bt_profile_manager_p->hsp_connect !=
bt_profile_connect_status_success) {
app_bt_HS_CreateServiceLink(bt_profile_manager_p->hs_chan,
&bt_profile_manager_p->rmt_addr);
}
}
}
#endif
static bool app_bt_profile_manager_connect_a2dp_filter_connected_a2dp_stream(
BT_BD_ADDR bd_addr) {
uint8_t i = 0;
BtRemoteDevice *StrmRemDev;
A2dpStream *connected_stream;
for (i = 0; i < BT_DEVICE_NUM; i++) {
if ((app_bt_device.a2dp_stream[i].stream.state ==
AVDTP_STRM_STATE_STREAMING ||
app_bt_device.a2dp_stream[i].stream.state == AVDTP_STRM_STATE_OPEN)) {
connected_stream = &app_bt_device.a2dp_stream[i];
StrmRemDev = A2DP_GetRemoteDevice(connected_stream);
if (memcmp(StrmRemDev->bdAddr.addr, bd_addr.addr, BD_ADDR_SIZE) == 0) {
return true;
}
}
}
return false;
}
static void app_bt_profile_connect_a2dp_retry_handler(void) {
struct app_bt_profile_manager *bt_profile_manager_p =
(struct app_bt_profile_manager *)param;
TRACE(1, "%s reconnect_cnt = %d", __func__,
bt_profile_manager_p->reconnect_cnt);
if (MEC(pendCons) > 0) {
if (bt_profile_manager_p->reconnect_cnt <
APP_BT_PROFILE_OPENNING_RECONNECT_RETRY_LIMIT_CNT) {
bt_profile_manager_p->reconnect_cnt++;
}
TRACE(1, "Former link is not down yet, reset the timer %s.", __FUNCTION__);
osTimerStart(bt_profile_manager_p->connect_timer,
BTIF_BT_DEFAULT_PAGE_TIMEOUT_IN_MS +
APP_BT_PROFILE_RECONNECT_RETRY_INTERVAL_MS);
} else {
if (app_bt_profile_manager_connect_a2dp_filter_connected_a2dp_stream(
bt_profile_manager_p->rmt_addr) == true) {
TRACE(0, "has been connected , no need to init connect again");
return;
}
app_bt_precheck_before_starting_connecting(
bt_profile_manager_p->has_connected);
if (bt_profile_manager_p->a2dp_connect !=
bt_profile_connect_status_success) {
app_bt_A2DP_OpenStream(bt_profile_manager_p->stream,
&bt_profile_manager_p->rmt_addr);
}
}
}
static void app_bt_profile_connect_a2dp_retry_timehandler(void const *param) {
app_bt_start_custom_function_in_bt_thread(
0, 0, (uint32_t)app_bt_profile_connect_a2dp_retry_handler);
}
#endif
void app_bt_reset_reconnect_timer(bt_bdaddr_t *pBdAddr) {
uint8_t devId = 0;
for (uint8_t i = 0; i < BT_DEVICE_NUM; i++) {
if (pBdAddr == &(bt_profile_manager[i].rmt_addr)) {
devId = i;
break;
}
}
TRACE(1, "Resart the reconnecting timer of dev %d", devId);
osTimerStart(bt_profile_manager[devId].connect_timer,
BTIF_BT_DEFAULT_PAGE_TIMEOUT_IN_MS +
APP_BT_PROFILE_RECONNECT_RETRY_INTERVAL_MS);
}
static void app_bt_profile_reconnect_handler(void const *param) {
#if !defined(IBRT)
struct app_bt_profile_manager *bt_profile_manager_p =
(struct app_bt_profile_manager *)param;
TRACE(1, "%s reconnect_cnt = %d", __FUNCTION__,
bt_profile_manager_p->reconnect_cnt);
if (btif_me_get_pendCons() > 0) {
if (bt_profile_manager_p->reconnect_cnt <
APP_BT_PROFILE_OPENNING_RECONNECT_RETRY_LIMIT_CNT) {
bt_profile_manager_p->reconnect_cnt++;
}
TRACE(1, "Former link is not down yet, reset the timer %s.", __FUNCTION__);
osTimerStart(bt_profile_manager_p->connect_timer,
BTIF_BT_DEFAULT_PAGE_TIMEOUT_IN_MS +
APP_BT_PROFILE_RECONNECT_RETRY_INTERVAL_MS);
} else {
btdevice_profile *btdevice_plf_p =
(btdevice_profile *)app_bt_profile_active_store_ptr_get(
bt_profile_manager_p->rmt_addr.address);
#ifdef __BT_ONE_BRING_TWO__
if (a2dp_is_music_ongoing() &&
(bt_profile_manager_p->has_connected == false)) {
bt_drv_reg_op_set_reconnecting_flag();
a2dp_update_music_link();
}
#endif
if (bt_profile_manager_p->connect_timer_cb) {
bt_profile_manager_p->connect_timer_cb(param);
bt_profile_manager_p->connect_timer_cb = NULL;
} else {
if ((btdevice_plf_p->hfp_act) && (bt_profile_manager_p->hfp_connect !=
bt_profile_connect_status_success)) {
TRACE(0, "try connect hf");
app_bt_precheck_before_starting_connecting(
bt_profile_manager_p->has_connected);
app_bt_HF_CreateServiceLink(
bt_profile_manager_p->chan,
(bt_bdaddr_t *)&bt_profile_manager_p->rmt_addr);
}
#if defined(__HSP_ENABLE__)
else if (btdevice_plf_p->hsp_act)
&&(bt_profile_manager_p->hsp_connect != bt_profile_connect_status_success))
{
TRACE(0, "try connect hs");
app_bt_precheck_before_starting_connecting(
bt_profile_manager_p->has_connected);
app_bt_HS_CreateServiceLink(bt_profile_manager_p->hs_chan,
&bt_profile_manager_p->rmt_addr);
}
#endif
else if ((btdevice_plf_p->a2dp_act) &&
(bt_profile_manager_p->a2dp_connect !=
bt_profile_connect_status_success)) {
TRACE(0, "try connect a2dp");
app_bt_precheck_before_starting_connecting(
bt_profile_manager_p->has_connected);
app_bt_A2DP_OpenStream(bt_profile_manager_p->stream,
&bt_profile_manager_p->rmt_addr);
}
}
}
#else
TRACE(0, "ibrt_ui_log:app_bt_profile_reconnect_timehandler called");
#endif
}
static void app_bt_profile_reconnect_timehandler(void const *param) {
app_bt_start_custom_function_in_bt_thread(
(uint32_t)param, 0, (uint32_t)app_bt_profile_reconnect_handler);
}
bool app_bt_is_in_connecting_profiles_state(void) {
for (uint8_t devId = 0; devId < BT_DEVICE_NUM; devId++) {
if (APP_BT_IN_CONNECTING_PROFILES_STATE ==
bt_profile_manager[devId].connectingState) {
return true;
}
}
return false;
}
void app_bt_clear_connecting_profiles_state(uint8_t devId) {
TRACE(1, "Dev %d exists connecting profiles state", devId);
bt_profile_manager[devId].connectingState = APP_BT_IDLE_STATE;
if (!app_bt_is_in_connecting_profiles_state()) {
#ifdef __IAG_BLE_INCLUDE__
app_start_fast_connectable_ble_adv(BLE_FAST_ADVERTISING_INTERVAL);
#endif
}
}
void app_bt_set_connecting_profiles_state(uint8_t devId) {
TRACE(1, "Dev %d enters connecting profiles state", devId);
bt_profile_manager[devId].connectingState =
APP_BT_IN_CONNECTING_PROFILES_STATE;
}
void app_bt_profile_connect_manager_open(void) {
uint8_t i = 0;
for (i = 0; i < BT_DEVICE_NUM; i++) {
bt_profile_manager[i].has_connected = false;
bt_profile_manager[i].hfp_connect = bt_profile_connect_status_unknow;
bt_profile_manager[i].hsp_connect = bt_profile_connect_status_unknow;
bt_profile_manager[i].a2dp_connect = bt_profile_connect_status_unknow;
memset(bt_profile_manager[i].rmt_addr.address, 0, BTIF_BD_ADDR_SIZE);
bt_profile_manager[i].reconnect_mode = bt_profile_reconnect_null;
bt_profile_manager[i].saved_reconnect_mode = bt_profile_reconnect_null;
bt_profile_manager[i].stream = NULL;
bt_profile_manager[i].chan = NULL;
#if defined(__HSP_ENABLE__)
bt_profile_manager[i].hs_chan = NULL;
#endif
bt_profile_manager[i].reconnect_cnt = 0;
bt_profile_manager[i].connect_timer_cb = NULL;
bt_profile_manager[i].connectingState = APP_BT_IDLE_STATE;
}
bt_profile_manager[BT_DEVICE_ID_1].connect_timer =
osTimerCreate(osTimer(BT_PROFILE_CONNECT_TIMER0), osTimerOnce,
&bt_profile_manager[BT_DEVICE_ID_1]);
#ifdef __BT_ONE_BRING_TWO__
bt_profile_manager[BT_DEVICE_ID_2].connect_timer =
osTimerCreate(osTimer(BT_PROFILE_CONNECT_TIMER1), osTimerOnce,
&bt_profile_manager[BT_DEVICE_ID_2]);
#endif
}
BOOL app_bt_profile_connect_openreconnecting(void *ptr) {
bool nRet = false;
uint8_t i;
/*
* If launched from peer device,stop reconnecting and accept connection
*/
if ((ptr != NULL) && (btif_me_get_remote_device_initiator(
(btif_remote_device_t *)ptr) == FALSE)) {
// Peer device launch reconnet,then we give up reconnect procedure
TRACE(0, "give up reconnecting");
app_bt_connectable_mode_stop_reconnecting();
return false;
}
for (i = 0; i < BT_DEVICE_NUM; i++) {
nRet |= bt_profile_manager[i].reconnect_mode ==
bt_profile_reconnect_openreconnecting
? true
: false;
if (nRet) {
TRACE(2, "io cap rj [%d]: %d", i, bt_profile_manager[i].reconnect_mode);
}
}
return nRet;
}
bool app_bt_is_in_reconnecting(void) {
uint8_t devId;
for (devId = 0; devId < BT_DEVICE_NUM; devId++) {
if (bt_profile_reconnect_null != bt_profile_manager[devId].reconnect_mode) {
return true;
}
}
return false;
}
void app_bt_profile_connect_manager_opening_reconnect(void) {
int ret;
btif_device_record_t record1;
btif_device_record_t record2;
btdevice_profile *btdevice_plf_p;
int find_invalid_record_cnt;
#if defined(APP_LINEIN_A2DP_SOURCE) || defined(APP_I2S_A2DP_SOURCE)
if (app_bt_device.src_or_snk == BT_DEVICE_SRC) {
return;
}
#endif
osapi_lock_stack();
#ifndef __BT_ONE_BRING_TWO__
if (btif_me_get_activeCons() != 0) {
osapi_unlock_stack();
TRACE(0, "bt link disconnect not complete,ignore this time reconnect");
return;
}
#endif
do {
find_invalid_record_cnt = 0;
ret = nv_record_enum_latest_two_paired_dev(&record1, &record2);
if (ret == 1) {
btdevice_plf_p = (btdevice_profile *)app_bt_profile_active_store_ptr_get(
record1.bdAddr.address);
if (!(btdevice_plf_p->hfp_act) && !(btdevice_plf_p->a2dp_act)) {
nv_record_ddbrec_delete((bt_bdaddr_t *)&record1.bdAddr);
find_invalid_record_cnt++;
}
} else if (ret == 2) {
btdevice_plf_p = (btdevice_profile *)app_bt_profile_active_store_ptr_get(
record1.bdAddr.address);
if (!(btdevice_plf_p->hfp_act) && !(btdevice_plf_p->a2dp_act)) {
nv_record_ddbrec_delete((bt_bdaddr_t *)&record1.bdAddr);
find_invalid_record_cnt++;
}
btdevice_plf_p = (btdevice_profile *)app_bt_profile_active_store_ptr_get(
record2.bdAddr.address);
if (!(btdevice_plf_p->hfp_act) && !(btdevice_plf_p->a2dp_act)) {
nv_record_ddbrec_delete((bt_bdaddr_t *)&record2.bdAddr);
find_invalid_record_cnt++;
}
}
} while (find_invalid_record_cnt);
TRACE(0, "!!!app_bt_opening_reconnect:\n");
DUMP8("%02x ", &record1.bdAddr, 6);
DUMP8("%02x ", &record2.bdAddr, 6);
if (ret > 0) {
TRACE(0, "!!!start reconnect first device\n");
if (btif_me_get_pendCons() == 0) {
bt_profile_manager[BT_DEVICE_ID_1].reconnect_mode =
bt_profile_reconnect_openreconnecting;
bt_profile_manager[BT_DEVICE_ID_1].reconnect_cnt = 0;
memcpy(bt_profile_manager[BT_DEVICE_ID_1].rmt_addr.address,
record1.bdAddr.address, BTIF_BD_ADDR_SIZE);
btdevice_plf_p = (btdevice_profile *)app_bt_profile_active_store_ptr_get(
bt_profile_manager[BT_DEVICE_ID_1].rmt_addr.address);
#if defined(A2DP_LHDC_ON)
if (btdevice_plf_p->a2dp_codectype == BTIF_AVDTP_CODEC_TYPE_NON_A2DP)
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_lhdc_stream[BT_DEVICE_ID_1]->a2dp_stream;
else
#endif
#if defined(A2DP_AAC_ON)
if (btdevice_plf_p->a2dp_codectype ==
BTIF_AVDTP_CODEC_TYPE_MPEG2_4_AAC)
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_aac_stream[BT_DEVICE_ID_1]->a2dp_stream;
else
#endif
#if defined(A2DP_LDAC_ON) // workaround for mate10 no a2dp issue when link back
if (btdevice_plf_p->a2dp_codectype ==
BTIF_AVDTP_CODEC_TYPE_NON_A2DP) {
// bt_profile_manager[BT_DEVICE_ID_1].stream =
// app_bt_device.a2dp_aac_stream[BT_DEVICE_ID_1]->a2dp_stream;
// btdevice_plf_p->a2dp_codectype =
// BTIF_AVDTP_CODEC_TYPE_MPEG2_4_AAC;
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_ldac_stream[BT_DEVICE_ID_1]
->a2dp_stream;
btdevice_plf_p->a2dp_codectype =
BTIF_AVDTP_CODEC_TYPE_NON_A2DP;
} else
#endif
#if defined(A2DP_SCALABLE_ON)
if (btdevice_plf_p->a2dp_codectype ==
BTIF_AVDTP_CODEC_TYPE_NON_A2DP)
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_scalable_stream[BT_DEVICE_ID_1]
->a2dp_stream;
else
#endif
{
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_stream[BT_DEVICE_ID_1]->a2dp_stream;
}
btif_a2dp_reset_stream_state(bt_profile_manager[BT_DEVICE_ID_1].stream);
bt_profile_manager[BT_DEVICE_ID_1].chan =
app_bt_device.hf_channel[BT_DEVICE_ID_1];
#if defined(__HSP_ENABLE__)
bt_profile_manager[BT_DEVICE_ID_1].hs_chan =
&app_bt_device.hs_channel[BT_DEVICE_ID_1];
#endif
if (btdevice_plf_p->hfp_act) {
TRACE(0, "try connect hf");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[BT_DEVICE_ID_1].has_connected);
app_bt_HF_CreateServiceLink(
bt_profile_manager[BT_DEVICE_ID_1].chan,
(bt_bdaddr_t *)&bt_profile_manager[BT_DEVICE_ID_1]
.rmt_addr);
} else if (btdevice_plf_p->a2dp_act) {
TRACE(0, "try connect a2dp");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[BT_DEVICE_ID_1].has_connected);
app_bt_A2DP_OpenStream(
bt_profile_manager[BT_DEVICE_ID_1].stream,
&bt_profile_manager[BT_DEVICE_ID_1].rmt_addr);
}
#if defined(__HSP_ENABLE__)
else if (btdevice_plf_p->hsp_act) {
TRACE(0, "try connect hs");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[BT_DEVICE_ID_1].has_connected);
app_bt_HS_CreateServiceLink(
bt_profile_manager[BT_DEVICE_ID_1].hs_chan,
&bt_profile_manager[BT_DEVICE_ID_1].rmt_addr);
}
#endif
}
#ifdef __BT_ONE_BRING_TWO__
if (ret > 1) {
TRACE(0, "!!!need reconnect second device\n");
bt_profile_manager[BT_DEVICE_ID_2].reconnect_mode =
bt_profile_reconnect_openreconnecting;
bt_profile_manager[BT_DEVICE_ID_2].reconnect_cnt = 0;
memcpy(bt_profile_manager[BT_DEVICE_ID_2].rmt_addr.address,
record2.bdAddr.address, BTIF_BD_ADDR_SIZE);
btdevice_plf_p = (btdevice_profile *)app_bt_profile_active_store_ptr_get(
bt_profile_manager[BT_DEVICE_ID_2].rmt_addr.address);
#if defined(A2DP_LHDC_ON)
if (btdevice_plf_p->a2dp_codectype == BTIF_AVDTP_CODEC_TYPE_NON_A2DP)
bt_profile_manager[BT_DEVICE_ID_2].stream =
app_bt_device.a2dp_lhdc_stream[BT_DEVICE_ID_2]->a2dp_stream;
else
#endif
#if defined(A2DP_AAC_ON)
if (btdevice_plf_p->a2dp_codectype ==
BTIF_AVDTP_CODEC_TYPE_MPEG2_4_AAC)
bt_profile_manager[BT_DEVICE_ID_2].stream =
app_bt_device.a2dp_aac_stream[BT_DEVICE_ID_2]->a2dp_stream;
else
#endif
#if defined(A2DP_SCALABLE_ON)
if (btdevice_plf_p->a2dp_codectype == BTIF_AVDTP_CODEC_TYPE_NON_A2DP)
bt_profile_manager[BT_DEVICE_ID_2].stream =
app_bt_device.a2dp_scalable_stream[BT_DEVICE_ID_2]
->a2dp_stream;
else
#endif
{
bt_profile_manager[BT_DEVICE_ID_2].stream =
app_bt_device.a2dp_stream[BT_DEVICE_ID_2]->a2dp_stream;
}
btif_a2dp_reset_stream_state(bt_profile_manager[BT_DEVICE_ID_2].stream);
bt_profile_manager[BT_DEVICE_ID_2].chan =
app_bt_device.hf_channel[BT_DEVICE_ID_2];
#if defined(__HSP_ENABLE__)
bt_profile_manager[BT_DEVICE_ID_2].hs_chan =
&app_bt_device.hs_channel[BT_DEVICE_ID_2];
#endif
}
#endif
}
else {
TRACE(0, "!!!go to pairing\n");
#ifdef __EARPHONE_STAY_BOTH_SCAN__
app_bt_accessmode_set_req(BTIF_BT_DEFAULT_ACCESS_MODE_PAIR);
#else
app_bt_accessmode_set_req(BTIF_BAM_CONNECTABLE_ONLY);
#endif
}
osapi_unlock_stack();
}
void app_bt_resume_sniff_mode(uint8_t deviceId) {
if (bt_profile_connect_status_success ==
bt_profile_manager[deviceId].a2dp_connect ||
bt_profile_connect_status_success ==
bt_profile_manager[deviceId].hfp_connect ||
bt_profile_connect_status_success ==
bt_profile_manager[deviceId].hsp_connect) {
app_bt_allow_sniff(deviceId);
btif_remote_device_t *currentRemDev = app_bt_get_remoteDev(deviceId);
app_bt_sniff_config(currentRemDev);
}
}
#if !defined(IBRT)
static int8_t app_bt_profile_reconnect_pending(enum BT_DEVICE_ID_T id) {
if (btapp_hfp_is_dev_call_active(id) == true) {
bt_profile_manager[id].reconnect_mode =
bt_profile_reconnect_reconnect_pending;
return 0;
}
return -1;
}
#endif
static int8_t app_bt_profile_reconnect_pending_process(void) {
uint8_t i = BT_DEVICE_NUM;
btif_remote_device_t *remDev = NULL;
btif_cmgr_handler_t *cmgrHandler;
for (i = 0; i < BT_DEVICE_NUM; i++) {
remDev = btif_me_enumerate_remote_devices(i);
if (remDev != NULL) {
cmgrHandler = btif_cmgr_get_acl_handler(remDev);
if (btif_cmgr_is_audio_up(cmgrHandler) == 1)
return -1;
}
}
for (i = 0; i < BT_DEVICE_NUM; i++) {
if (bt_profile_manager[i].reconnect_mode ==
bt_profile_reconnect_reconnect_pending)
break;
}
if (i == BT_DEVICE_NUM)
return -1;
bt_profile_manager[i].reconnect_mode = bt_profile_reconnect_reconnecting;
#ifdef __IAG_BLE_INCLUDE__
app_ble_refresh_adv_state(BLE_ADVERTISING_INTERVAL);
#endif
osTimerStart(bt_profile_manager[i].connect_timer,
APP_BT_PROFILE_RECONNECT_RETRY_INTERVAL_MS);
return 0;
}
uint8_t app_bt_get_num_of_connected_dev(void) {
uint8_t num_of_connected_dev = 0;
uint8_t deviceId;
for (deviceId = 0; deviceId < BT_DEVICE_NUM; deviceId++) {
if (bt_profile_manager[deviceId].has_connected) {
num_of_connected_dev++;
}
}
return num_of_connected_dev;
}
static uint8_t recorded_latest_connected_service_device_id = BT_DEVICE_ID_1;
void app_bt_record_latest_connected_service_device_id(uint8_t device_id) {
recorded_latest_connected_service_device_id = device_id;
}
uint8_t app_bt_get_recorded_latest_connected_service_device_id(void) {
return recorded_latest_connected_service_device_id;
}
static void app_bt_precheck_before_starting_connecting(uint8_t isBtConnected) {
#ifdef __IAG_BLE_INCLUDE__
if (!isBtConnected) {
app_ble_force_switch_adv(BLE_SWITCH_USER_BT_CONNECT, false);
}
#endif
}
static void app_bt_restore_reconnecting_idle_mode(uint8_t deviceId) {
bt_profile_manager[deviceId].reconnect_mode = bt_profile_reconnect_null;
#ifdef __IAG_BLE_INCLUDE__
app_start_fast_connectable_ble_adv(BLE_FAST_ADVERTISING_INTERVAL);
#endif
}
#ifdef __BT_ONE_BRING_TWO__
static void app_bt_update_connectable_mode_after_connection_management(void) {
uint8_t deviceId;
bool isEnterConnetableOnlyState = true;
for (deviceId = 0; deviceId < BT_DEVICE_NUM; deviceId++) {
// assure none of the device is in reconnecting mode
if (bt_profile_manager[deviceId].reconnect_mode !=
bt_profile_reconnect_null) {
isEnterConnetableOnlyState = false;
break;
}
}
if (isEnterConnetableOnlyState) {
for (deviceId = 0; deviceId < BT_DEVICE_NUM; deviceId++) {
if (!bt_profile_manager[deviceId].has_connected) {
app_bt_accessmode_set(BTIF_BAM_CONNECTABLE_ONLY);
return;
}
}
}
}
#endif
static void app_bt_connectable_mode_stop_reconnecting_handler(void) {
uint8_t deviceId;
btif_remote_device_t *remDev;
btif_cmgr_handler_t *cmgrHandler;
for (deviceId = 0; deviceId < BT_DEVICE_NUM; deviceId++) {
if (bt_profile_manager[deviceId].reconnect_mode !=
bt_profile_reconnect_null) {
bt_profile_manager[deviceId].hfp_connect =
bt_profile_connect_status_failure;
bt_profile_manager[deviceId].reconnect_mode = bt_profile_reconnect_null;
bt_profile_manager[deviceId].saved_reconnect_mode =
bt_profile_reconnect_null;
bt_profile_manager[deviceId].reconnect_cnt = 0;
if (bt_profile_manager[deviceId].connect_timer != NULL)
osTimerStop(bt_profile_manager[deviceId].connect_timer);
remDev = btif_me_enumerate_remote_devices(deviceId);
if (remDev != NULL) {
cmgrHandler = btif_cmgr_get_acl_handler(remDev);
btif_me_cancel_create_link(
btif_cmgr_get_cmgrhandler_remdev_bthandle(cmgrHandler),
remDev);
}
}
}
}
void app_bt_connectable_mode_stop_reconnecting(void) {
app_bt_start_custom_function_in_bt_thread(
0, 0, (uint32_t)app_bt_connectable_mode_stop_reconnecting_handler);
}
#if defined(__HSP_ENABLE__)
void app_bt_profile_connect_manager_hs(enum BT_DEVICE_ID_T id, HsChannel *Chan,
HsCallbackParms *Info) {
btdevice_profile *btdevice_plf_p =
(btdevice_profile *)app_bt_profile_active_store_ptr_get(
(uint8_t *)Info->p.remDev->bdAddr.address);
osTimerStop(bt_profile_manager[id].connect_timer);
bt_profile_manager[id].connect_timer_cb = NULL;
bool profile_reconnect_enable = false;
if (Chan && Info) {
switch (Info->event) {
case HF_EVENT_SERVICE_CONNECTED:
TRACE(1, "%s HS_EVENT_SERVICE_CONNECTED", __func__);
nv_record_btdevicerecord_set_hsp_profile_active_state(btdevice_plf_p,
true);
nv_record_touch_cause_flush();
bt_profile_manager[id].hsp_connect = bt_profile_connect_status_success;
bt_profile_manager[id].reconnect_cnt = 0;
bt_profile_manager[id].hs_chan = &app_bt_device.hs_channel[id];
memcpy(bt_profile_manager[id].rmt_addr.address,
Info->p.remDev->bdAddr.address, BTIF_BD_ADDR_SIZE);
if (false == bt_profile_manager[id].has_connected) {
app_bt_resume_sniff_mode(id);
}
if (bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_openreconnecting) {
// do nothing
} else if (bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_reconnecting) {
if (btdevice_plf_p->a2dp_act &&
bt_profile_manager[id].a2dp_connect !=
bt_profile_connect_status_success) {
TRACE(0, "!!!continue connect a2dp\n");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[id].has_connected);
app_bt_A2DP_OpenStream(bt_profile_manager[id].stream,
&bt_profile_manager[id].rmt_address);
}
}
#ifdef __AUTO_CONNECT_OTHER_PROFILE__
else {
if (btdevice_plf_p->a2dp_act &&
bt_profile_manager[id].a2dp_connect !=
bt_profile_connect_status_success) {
bt_profile_manager[id].connect_timer_cb =
app_bt_profile_connect_a2dp_retry_timehandler;
app_bt_accessmode_set(BTIF_BAM_CONNECTABLE_ONLY);
osTimerStart(bt_profile_manager[id].connect_timer,
APP_BT_PROFILE_CONNECT_RETRY_MS);
}
}
#endif
break;
case HF_EVENT_SERVICE_DISCONNECTED:
TRACE(2, "%s HS_EVENT_SERVICE_DISCONNECTED discReason:%d", __func__,
Info->p.remDev->discReason);
bt_profile_manager[id].hsp_connect = bt_profile_connect_status_failure;
if (bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_openreconnecting) {
if (++bt_profile_manager[id].reconnect_cnt <
APP_BT_PROFILE_OPENNING_RECONNECT_RETRY_LIMIT_CNT) {
app_bt_accessmode_set(BTIF_BAM_CONNECTABLE_ONLY);
profile_reconnect_enable = true;
bt_profile_manager[id].hfp_connect =
bt_profile_connect_status_unknow;
}
} else if (bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_reconnecting) {
if (++bt_profile_manager[id].reconnect_cnt <
APP_BT_PROFILE_RECONNECT_RETRY_LIMIT_CNT) {
app_bt_accessmode_set(BTIF_BAM_CONNECTABLE_ONLY);
profile_reconnect_enable = true
} else {
app_bt_restore_reconnecting_idle_mode(id);
// bt_profile_manager[id].reconnect_mode =
// bt_profile_reconnect_null;
}
TRACE(2, "%s try to reconnect cnt:%d", __func__,
bt_profile_manager[id].reconnect_cnt);
#if !defined(IBRT)
} else if (Info->p.remDev->discReason == 0x8) {
bt_profile_manager[id].reconnect_mode =
bt_profile_reconnect_reconnecting;
app_bt_accessmode_set(BTIF_BAM_CONNECTABLE_ONLY);
TRACE(1, "%s try to reconnect", __func__);
if (app_bt_profile_reconnect_pending(id) != 0) {
profile_reconnect_enable = true;
}
#endif
} else {
bt_profile_manager[id].hsp_connect =
bt_profile_connect_status_unknow;
}
if (profile_reconnect_enable) {
#ifdef __IAG_BLE_INCLUDE__
app_ble_refresh_adv_state(BLE_ADVERTISING_INTERVAL);
#endif
osTimerStart(bt_profile_manager[id].connect_timer,
APP_BT_PROFILE_RECONNECT_RETRY_INTERVAL_MS);
}
break;
default:
break;
}
}
if (bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_reconnecting) {
bool reconnect_hsp_proc_final = true;
bool reconnect_a2dp_proc_final = true;
if (bt_profile_manager[id].hsp_connect ==
bt_profile_connect_status_failure) {
reconnect_hsp_proc_final = false;
}
if (bt_profile_manager[id].a2dp_connect ==
bt_profile_connect_status_failure) {
reconnect_a2dp_proc_final = false;
}
if (reconnect_hsp_proc_final && reconnect_a2dp_proc_final) {
TRACE(3, "!!!reconnect success %d/%d/%d\n",
bt_profile_manager[id].hfp_connect,
bt_profile_manager[id].hsp_connect,
bt_profile_manager[id].a2dp_connect);
app_bt_restore_reconnecting_idle_mode(id);
// bt_profile_manager[id].reconnect_mode = bt_profile_reconnect_null;
}
} else if (bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_openreconnecting) {
bool opening_hsp_proc_final = false;
bool opening_a2dp_proc_final = false;
if (btdevice_plf_p->hsp_act && bt_profile_manager[id].hsp_connect ==
bt_profile_connect_status_unknow) {
opening_hsp_proc_final = false;
} else {
opening_hsp_proc_final = true;
}
if (btdevice_plf_p->a2dp_act && bt_profile_manager[id].a2dp_connect ==
bt_profile_connect_status_unknow) {
opening_a2dp_proc_final = false;
} else {
opening_a2dp_proc_final = true;
}
if ((opening_hsp_proc_final && opening_a2dp_proc_final) ||
(bt_profile_manager[id].hsp_connect ==
bt_profile_connect_status_failure)) {
TRACE(3, "!!!reconnect success %d/%d/%d\n",
bt_profile_manager[id].hfp_connect,
bt_profile_manager[id].hsp_connect,
bt_profile_manager[id].a2dp_connect);
app_bt_restore_reconnecting_idle_mode(id);
// bt_profile_manager[id].reconnect_mode = bt_profile_reconnect_null;
}
if (btdevice_plf_p->hsp_act && bt_profile_manager[id].hsp_connect ==
bt_profile_connect_status_success) {
if (btdevice_plf_p->a2dp_act && !opening_a2dp_proc_final) {
TRACE(0, "!!!continue connect a2dp\n");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[id].has_connected);
app_bt_A2DP_OpenStream(bt_profile_manager[id].stream,
&bt_profile_manager[id].rmt_addr);
}
}
if (bt_profile_manager[id].reconnect_mode == bt_profile_reconnect_null) {
for (uint8_t i = 0; i < BT_DEVICE_NUM; i++) {
if (bt_profile_manager[i].reconnect_mode ==
bt_profile_reconnect_openreconnecting) {
TRACE(0, "!!!hs->start reconnect second device\n");
if ((btdevice_plf_p->hfp_act) &&
(!bt_profile_manager[i].hfp_connect)) {
TRACE(0, "try connect hf");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[i].has_connected);
app_bt_HF_CreateServiceLink(
bt_profile_manager[i].chan,
&bt_profile_manager[i].rmt_addr);
} else if ((btdevice_plf_p->hsp_act) &&
(!bt_profile_manager[i].hsp_connect)) {
TRACE(0, "try connect hs");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[i].has_connected);
app_bt_HS_CreateServiceLink(
bt_profile_manager[i].hs_chan,
&bt_profile_manager[i].rmt_addr);
} else if ((btdevice_plf_p->a2dp_act) &&
(!bt_profile_manager[i].a2dp_connect)) {
TRACE(0, "try connect a2dp");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[i].has_connected);
app_bt_A2DP_OpenStream(bt_profile_manager[i].stream,
&bt_profile_manager[i].rmt_addr);
}
break;
}
}
}
}
#ifdef __IAG_BLE_INCLUDE__
if (bt_profile_manager[id].hfp_connect == bt_profile_connect_status_success &&
bt_profile_manager[id].hsp_connect == bt_profile_connect_status_success &&
bt_profile_manager[id].a2dp_connect ==
bt_profile_connect_status_success) {
app_ble_force_switch_adv(BLE_SWITCH_USER_BT_CONNECT, true);
}
#endif
if (!bt_profile_manager[id].has_connected &&
(bt_profile_manager[id].hfp_connect ==
bt_profile_connect_status_success ||
bt_profile_manager[id].hsp_connect ==
bt_profile_connect_status_success ||
bt_profile_manager[id].a2dp_connect ==
bt_profile_connect_status_success)) {
bt_profile_manager[id].has_connected = true;
TRACE(0, "BT connected!!!");
#ifndef IBRT
btif_me_get_remote_device_name(&(ctx->remote_dev_bdaddr),
app_bt_global_handle);
#endif
#if defined(MEDIA_PLAYER_SUPPORT) && !defined(IBRT)
app_voice_report(APP_STATUS_INDICATION_CONNECTED, id);
#endif
#ifdef __INTERCONNECTION__
app_interconnection_start_disappear_adv(
INTERCONNECTION_BLE_ADVERTISING_INTERVAL,
APP_INTERCONNECTION_DISAPPEAR_ADV_IN_MS);
if (btif_me_get_activeCons() <= 2) {
app_interconnection_spp_open(btif_me_enumerate_remote_devices(id));
}
#endif
#ifdef __INTERACTION__
// app_interaction_spp_open();
#endif
}
if (bt_profile_manager[id].has_connected &&
(bt_profile_manager[id].hfp_connect !=
bt_profile_connect_status_success &&
bt_profile_manager[id].hsp_connect !=
bt_profile_connect_status_success &&
bt_profile_manager[id].a2dp_connect !=
bt_profile_connect_status_success)) {
bt_profile_manager[id].has_connected = false;
TRACE(0, "BT disconnected!!!");
#ifdef GFPS_ENABLED
if (app_gfps_is_last_response_pending()) {
app_gfps_enter_connectable_mode_req_handler(app_gfps_get_last_response());
}
#endif
#if defined(MEDIA_PLAYER_SUPPORT) && !defined(IBRT)
app_voice_report(APP_STATUS_INDICATION_DISCONNECTED, id);
#endif
#ifdef __INTERCONNECTION__
app_interconnection_disconnected_callback();
#endif
app_set_disconnecting_all_bt_connections(false);
}
#ifdef __BT_ONE_BRING_TWO__
app_bt_update_connectable_mode_after_connection_management();
#endif
}
#endif
void hfp_reconnecting_timer_stop_callback(const btif_event_t *event) {
uint8_t i = 0;
uint8_t id = BT_DEVICE_NUM;
bt_bdaddr_t *remote = NULL;
bt_bdaddr_t *hfp_remote = NULL;
remote = btif_me_get_callback_event_rem_dev_bd_addr(event);
if (remote != NULL) {
for (i = 0; i < BT_DEVICE_NUM; i++) {
hfp_remote = &bt_profile_manager[i].rmt_addr;
if (!strcmp((char *)hfp_remote, (char *)remote)) {
id = i;
TRACE(2, "%s: find bt device num = %d", __func__, id);
break;
}
}
}
if (i < BT_DEVICE_NUM) {
TRACE(3, "%s: hfp_connect=%d,reconnect_mode=%d,reconnect_cnt=%d", __func__,
bt_profile_manager[id].hfp_connect,
bt_profile_manager[id].reconnect_mode,
bt_profile_manager[id].reconnect_cnt);
if ((bt_profile_manager[id].reconnect_mode != bt_profile_reconnect_null) &&
bt_profile_manager[id].reconnect_cnt != 0) {
bt_profile_manager[id].reconnect_mode = bt_profile_reconnect_null;
bt_profile_manager[id].saved_reconnect_mode = bt_profile_reconnect_null;
bt_profile_manager[id].reconnect_cnt = 0;
if (bt_profile_manager[id].connect_timer != NULL)
osTimerStop(bt_profile_manager[id].connect_timer);
TRACE(1, "%s: stop success", __func__);
}
} else {
TRACE(1, "%s: not find bt device", __func__);
}
}
void app_audio_switch_flash_flush_req(void);
extern uint8_t once_event_case;
extern bool IsMobileLinkLossing;
// void app_bt_profile_connect_manager_hf(enum BT_DEVICE_ID_T id, HfChannel
// *Chan, HfCallbackParms *Info)
void app_bt_profile_connect_manager_hf(enum BT_DEVICE_ID_T id,
hf_chan_handle_t Chan,
struct hfp_context *ctx) {
static ibrt_ctrl_t *p_ibrt_ctrl = app_ibrt_if_get_bt_ctrl_ctx();
// btdevice_profile *btdevice_plf_p = (btdevice_profile
// *)app_bt_profile_active_store_ptr_get((uint8_t
// *)Info->p.remDev->bdAddr.address);
btdevice_profile *btdevice_plf_p =
(btdevice_profile *)app_bt_profile_active_store_ptr_get(
(uint8_t *)ctx->remote_dev_bdaddr.address);
bool profile_reconnect_enable = false;
osTimerStop(bt_profile_manager[id].connect_timer);
bt_profile_manager[id].connect_timer_cb = NULL;
// if (Chan&&Info){
if (Chan) {
switch (ctx->event) {
case BTIF_HF_EVENT_SERVICE_CONNECTED:
TRACE(1, "%s HF_EVENT_SERVICE_CONNECTED", __func__);
nv_record_btdevicerecord_set_hfp_profile_active_state(btdevice_plf_p,
true);
nv_record_touch_cause_flush();
bt_profile_manager[id].hfp_connect = bt_profile_connect_status_success;
bt_profile_manager[id].saved_reconnect_mode = bt_profile_reconnect_null;
bt_profile_manager[id].reconnect_cnt = 0;
bt_profile_manager[id].chan = app_bt_device.hf_channel[id];
memcpy(bt_profile_manager[id].rmt_addr.address,
ctx->remote_dev_bdaddr.address, BTIF_BD_ADDR_SIZE);
if (false == bt_profile_manager[id].has_connected) {
app_bt_resume_sniff_mode(id);
}
#ifdef BTIF_DIP_DEVICE
btif_dip_get_remote_info(app_bt_get_remoteDev(id));
#endif
if (bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_openreconnecting) {
// do nothing
}
#if defined(IBRT)
else if (app_bt_ibrt_reconnect_mobile_profile_flag_get()) {
app_bt_ibrt_reconnect_mobile_profile_flag_clear();
#else
else if (bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_reconnecting) {
#endif
TRACE(2, "app_bt: a2dp_act in NV =%d,a2dp_connect=%d",
btdevice_plf_p->a2dp_act,
bt_profile_manager[id].a2dp_connect);
if (btdevice_plf_p->a2dp_act &&
bt_profile_manager[id].a2dp_connect !=
bt_profile_connect_status_success) {
TRACE(0, "!!!continue connect a2dp\n");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[id].has_connected);
app_bt_A2DP_OpenStream(bt_profile_manager[id].stream,
&bt_profile_manager[id].rmt_addr);
}
}
#ifdef __AUTO_CONNECT_OTHER_PROFILE__
else {
// befor auto connect a2dp profile, check whether a2dp is
// supported
if (btdevice_plf_p->a2dp_act &&
bt_profile_manager[id].a2dp_connect !=
bt_profile_connect_status_success) {
bt_profile_manager[id].connect_timer_cb =
app_bt_profile_connect_a2dp_retry_timehandler;
app_bt_accessmode_set(BAM_CONNECTABLE_ONLY);
osTimerStart(bt_profile_manager[id].connect_timer,
APP_BT_PROFILE_CONNECT_RETRY_MS);
}
}
#endif
break;
case BTIF_HF_EVENT_SERVICE_DISCONNECTED:
if (((ctx->disc_reason == 0) || (ctx->disc_reason == 19)) &&
(p_ibrt_ctrl->current_role != IBRT_SLAVE)) {
once_event_case = 2;
startonce_delay_event_Timer_(1000);
}
// TRACE(3,"%s HF_EVENT_SERVICE_DISCONNECTED discReason:%d/%d",__func__,
// Info->p.remDev->discReason, Info->p.remDev->discReason_saved);
TRACE(3, "%s HF_EVENT_SERVICE_DISCONNECTED discReason:%d/%d", __func__,
ctx->disc_reason, ctx->disc_reason_saved);
bt_profile_manager[id].hfp_connect = bt_profile_connect_status_failure;
if (bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_openreconnecting) {
if (++bt_profile_manager[id].reconnect_cnt <
APP_BT_PROFILE_OPENNING_RECONNECT_RETRY_LIMIT_CNT) {
app_bt_accessmode_set(BTIF_BAM_CONNECTABLE_ONLY);
profile_reconnect_enable = true;
bt_profile_manager[id].hfp_connect =
bt_profile_connect_status_unknow;
}
} else if (bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_reconnecting) {
if (++bt_profile_manager[id].reconnect_cnt <
APP_BT_PROFILE_RECONNECT_RETRY_LIMIT_CNT) {
app_bt_accessmode_set(BTIF_BAM_CONNECTABLE_ONLY);
profile_reconnect_enable = true;
} else {
app_bt_restore_reconnecting_idle_mode(id);
// bt_profile_manager[id].reconnect_mode =
// bt_profile_reconnect_null;
}
TRACE(2, "%s try to reconnect cnt:%d", __func__,
bt_profile_manager[id].reconnect_cnt);
/*
}else if ((Info->p.remDev->discReason == 0x8)||
(Info->p.remDev->discReason_saved == 0x8)){
*/
}
#if !defined(IBRT)
#if defined(ENHANCED_STACK)
else if ((ctx->disc_reason == 0x8) || (ctx->disc_reason_saved == 0x8) ||
(ctx->disc_reason == 0x4) || (ctx->disc_reason_saved == 0x4))
#else
else if ((ctx->disc_reason == 0x8) || (ctx->disc_reason_saved == 0x8) ||
(ctx->disc_reason == 0x0) || (ctx->disc_reason_saved == 0x0))
#endif
{
bt_profile_manager[id].reconnect_mode =
bt_profile_reconnect_reconnecting;
app_bt_accessmode_set(BTIF_BAM_CONNECTABLE_ONLY);
TRACE(2, "%s try to reconnect reason =%d", __func__,
ctx->disc_reason);
if (app_bt_profile_reconnect_pending(id) != 0) {
profile_reconnect_enable = true;
}
}
#endif
else {
bt_profile_manager[id].hfp_connect =
bt_profile_connect_status_unknow;
}
if (profile_reconnect_enable) {
#ifdef __IAG_BLE_INCLUDE__
app_ble_refresh_adv_state(BLE_ADVERTISING_INTERVAL);
#endif
osTimerStart(bt_profile_manager[id].connect_timer,
APP_BT_PROFILE_RECONNECT_RETRY_INTERVAL_MS);
}
break;
default:
break;
}
}
DUMP8("%02x ", &bt_profile_manager[id].rmt_addr.address, 6);
btdevice_profile *btdevice_plf_p1 =
(btdevice_profile *)app_bt_profile_active_store_ptr_get(
(uint8_t *)&bt_profile_manager[id].rmt_addr.address);
if (bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_reconnecting) {
bool reconnect_hfp_proc_final = false;
bool reconnect_a2dp_proc_final = false;
if (bt_profile_manager[id].hfp_connect !=
bt_profile_connect_status_success) {
reconnect_hfp_proc_final = false;
} else {
reconnect_hfp_proc_final = true;
}
if (bt_profile_manager[id].a2dp_connect !=
bt_profile_connect_status_success) {
if (btdevice_plf_p1->hfp_act && btdevice_plf_p1->a2dp_act) {
reconnect_a2dp_proc_final = false;
} else {
reconnect_a2dp_proc_final = true;
}
}
if (reconnect_hfp_proc_final && reconnect_a2dp_proc_final) {
TRACE(3, "!!!reconnect success %d/%d/%d\n",
bt_profile_manager[id].hfp_connect,
bt_profile_manager[id].hsp_connect,
bt_profile_manager[id].a2dp_connect);
app_bt_restore_reconnecting_idle_mode(id);
// bt_profile_manager[id].reconnect_mode = bt_profile_reconnect_null;
}
} else if (bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_openreconnecting) {
bool opening_hfp_proc_final = false;
bool opening_a2dp_proc_final = false;
if (btdevice_plf_p1->hfp_act && bt_profile_manager[id].hfp_connect ==
bt_profile_connect_status_unknow) {
opening_hfp_proc_final = false;
} else {
opening_hfp_proc_final = true;
}
if (btdevice_plf_p1->a2dp_act && bt_profile_manager[id].a2dp_connect ==
bt_profile_connect_status_unknow) {
opening_a2dp_proc_final = false;
} else {
opening_a2dp_proc_final = true;
}
if (opening_hfp_proc_final && opening_a2dp_proc_final) {
TRACE(3, "!!!reconnect success %d/%d/%d\n",
bt_profile_manager[id].hfp_connect,
bt_profile_manager[id].hsp_connect,
bt_profile_manager[id].a2dp_connect);
bt_profile_manager[id].saved_reconnect_mode =
bt_profile_reconnect_openreconnecting;
app_bt_restore_reconnecting_idle_mode(id);
} else if (bt_profile_manager[id].hfp_connect ==
bt_profile_connect_status_failure) {
TRACE(3, "reconnect_mode888:%d", bt_profile_manager[0].reconnect_mode);
TRACE(3, "!!!reconnect success %d/%d/%d\n",
bt_profile_manager[id].hfp_connect,
bt_profile_manager[id].hsp_connect,
bt_profile_manager[id].a2dp_connect);
bt_profile_manager[id].saved_reconnect_mode =
bt_profile_reconnect_openreconnecting;
if ((bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_openreconnecting) &&
(bt_profile_manager[id].reconnect_cnt >=
APP_BT_PROFILE_OPENNING_RECONNECT_RETRY_LIMIT_CNT)) {
app_bt_restore_reconnecting_idle_mode(id);
}
}
if (btdevice_plf_p1->hfp_act && bt_profile_manager[id].hfp_connect ==
bt_profile_connect_status_success) {
if (btdevice_plf_p1->a2dp_act && !opening_a2dp_proc_final) {
TRACE(1, "!!!continue connect a2dp %p\n",
bt_profile_manager[id].stream);
app_bt_precheck_before_starting_connecting(
bt_profile_manager[id].has_connected);
app_bt_A2DP_OpenStream(bt_profile_manager[id].stream,
&bt_profile_manager[id].rmt_addr);
}
}
if (bt_profile_manager[id].reconnect_mode == bt_profile_reconnect_null) {
for (uint8_t i = 0; i < BT_DEVICE_NUM; i++) {
btdevice_profile *btdevice_plf_p_temp =
(btdevice_profile *)app_bt_profile_active_store_ptr_get(
(uint8_t *)bt_profile_manager[i].rmt_addr.address);
TRACE(3, "reconnect_mode:%d",
bt_profile_manager[i].reconnect_mode);
if (bt_profile_manager[i].reconnect_mode ==
bt_profile_reconnect_openreconnecting) {
TRACE(0, "!!!hf->start reconnect second device\n");
if ((btdevice_plf_p_temp->hfp_act) &&
(!bt_profile_manager[i].hfp_connect)) {
TRACE(0, "try connect hf");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[id].has_connected);
app_bt_HF_CreateServiceLink(
bt_profile_manager[i].chan,
(bt_bdaddr_t *)&bt_profile_manager[i].rmt_addr);
}
#if defined(__HSP_ENABLE__)
else if ((btdevice_plf_p_temp->hsp_act) &&
(!bt_profile_manager[i].hsp_connect)) {
TRACE(0, "try connect hs");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[id].has_connected);
app_bt_HS_CreateServiceLink(
bt_profile_manager[i].hs_chan,
&bt_profile_manager[i].rmt_addr);
}
#endif
else if ((btdevice_plf_p_temp->a2dp_act) &&
(!bt_profile_manager[i].a2dp_connect)) {
TRACE(0, "try connect a2dp");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[id].has_connected);
app_bt_A2DP_OpenStream(bt_profile_manager[i].stream,
&bt_profile_manager[i].rmt_addr);
}
break;
}
}
}
}
#ifdef __INTERCONNECTION__
if (bt_profile_manager[id].hfp_connect == bt_profile_connect_status_success &&
bt_profile_manager[id].a2dp_connect ==
bt_profile_connect_status_success) {
app_interconnection_start_disappear_adv(
INTERCONNECTION_BLE_ADVERTISING_INTERVAL,
APP_INTERCONNECTION_DISAPPEAR_ADV_IN_MS);
if (btif_me_get_activeCons() <= 2) {
app_interconnection_spp_open(btif_me_enumerate_remote_devices(id));
}
}
#endif
#ifdef __IAG_BLE_INCLUDE__
TRACE(3, "%s hfp %d a2dp %d", __func__, bt_profile_manager[id].hfp_connect,
bt_profile_manager[id].a2dp_connect);
if (bt_profile_manager[id].hfp_connect == bt_profile_connect_status_success &&
#ifdef __HSP_ENABLE__
bt_profile_manager[id].hsp_connect == bt_profile_connect_status_success &&
#endif
bt_profile_manager[id].a2dp_connect ==
bt_profile_connect_status_success) {
app_ble_force_switch_adv(BLE_SWITCH_USER_BT_CONNECT, true);
}
#endif
if (!bt_profile_manager[id].has_connected &&
(bt_profile_manager[id].hfp_connect ==
bt_profile_connect_status_success ||
#ifdef __HSP_ENABLE__
bt_profile_manager[id].hsp_connect ==
bt_profile_connect_status_success ||
#endif
bt_profile_manager[id].a2dp_connect ==
bt_profile_connect_status_success)) {
bt_profile_manager[id].has_connected = true;
TRACE(0, "BT connected!!!");
once_event_case = 1;
/*if(IsMobileLinkLossing){
startonce_delay_event_Timer_(3000);
}
else{
startonce_delay_event_Timer_(1500);
}*/
app_voice_report(APP_STATUS_INDICATION_CONNECTED, 0);
IsMobileLinkLossing = FALSE;
#ifndef IBRT
btif_me_get_remote_device_name(&(ctx->remote_dev_bdaddr),
app_bt_global_handle);
#endif
#if defined(MEDIA_PLAYER_SUPPORT) && !defined(IBRT)
app_voice_report(APP_STATUS_INDICATION_CONNECTED, id);
#endif
#if 0 // #ifdef __INTERCONNECTION__
app_interconnection_start_disappear_adv(BLE_ADVERTISING_INTERVAL, APP_INTERCONNECTION_DISAPPEAR_ADV_IN_MS);
app_interconnection_spp_open();
#endif
#ifdef __INTERACTION__
// app_interaction_spp_open();
#endif
}
if (bt_profile_manager[id].has_connected &&
(bt_profile_manager[id].hfp_connect !=
bt_profile_connect_status_success &&
#ifdef __HSP_ENABLE__
bt_profile_manager[id].hsp_connect !=
bt_profile_connect_status_success &&
#endif
bt_profile_manager[id].a2dp_connect !=
bt_profile_connect_status_success)) {
bt_profile_manager[id].has_connected = false;
TRACE(0, "BT disconnected!!!");
#ifdef GFPS_ENABLED
if (app_gfps_is_last_response_pending()) {
app_gfps_enter_connectable_mode_req_handler(app_gfps_get_last_response());
}
#endif
#if defined(MEDIA_PLAYER_SUPPORT) && !defined(IBRT)
app_voice_report(APP_STATUS_INDICATION_DISCONNECTED, id);
#endif
#ifdef __INTERCONNECTION__
app_interconnection_disconnected_callback();
#endif
app_set_disconnecting_all_bt_connections(false);
}
#ifdef __BT_ONE_BRING_TWO__
app_bt_update_connectable_mode_after_connection_management();
#endif
}
void app_bt_profile_connect_manager_a2dp(enum BT_DEVICE_ID_T id,
a2dp_stream_t *Stream,
const a2dp_callback_parms_t *info) {
btdevice_profile *btdevice_plf_p = NULL;
btif_remote_device_t *remDev = NULL;
btif_a2dp_callback_parms_t *Info = (btif_a2dp_callback_parms_t *)info;
osTimerStop(bt_profile_manager[id].connect_timer);
bt_profile_manager[id].connect_timer_cb = NULL;
bool profile_reconnect_enable = false;
remDev = btif_a2dp_get_stream_conn_remDev(Stream);
if (remDev) {
btdevice_plf_p = (btdevice_profile *)app_bt_profile_active_store_ptr_get(
btif_me_get_remote_device_bdaddr(remDev)->address);
} else {
btdevice_plf_p =
(btdevice_profile *)app_bt_profile_active_store_ptr_get(NULL);
}
if (Stream && Info) {
switch (Info->event) {
case BTIF_A2DP_EVENT_STREAM_OPEN:
TRACE(4, "%s A2DP_EVENT_STREAM_OPEN,codec type=%x a2dp:%d mode:%d",
__func__, Info->p.configReq->codec.codecType,
bt_profile_manager[id].a2dp_connect,
bt_profile_manager[id].reconnect_mode);
nv_record_btdevicerecord_set_a2dp_profile_active_state(btdevice_plf_p,
true);
nv_record_btdevicerecord_set_a2dp_profile_codec(
btdevice_plf_p, Info->p.configReq->codec.codecType);
nv_record_touch_cause_flush();
if (bt_profile_manager[id].a2dp_connect ==
bt_profile_connect_status_success) {
TRACE(0, "!!!a2dp has opened force return ");
return;
}
bt_profile_manager[id].a2dp_connect = bt_profile_connect_status_success;
bt_profile_manager[id].reconnect_cnt = 0;
bt_profile_manager[id].stream = app_bt_device.a2dp_connected_stream[id];
memcpy(bt_profile_manager[id].rmt_addr.address,
btif_me_get_remote_device_bdaddr(
btif_a2dp_get_stream_conn_remDev(Stream))
->address,
BTIF_BD_ADDR_SIZE);
app_bt_record_latest_connected_service_device_id(id);
if (false == bt_profile_manager[id].has_connected) {
app_bt_resume_sniff_mode(id);
}
#ifdef BTIF_DIP_DEVICE
btif_dip_get_remote_info(remDev);
#endif
if (bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_openreconnecting) {
// do nothing
}
#if defined(IBRT)
else if (app_bt_ibrt_reconnect_mobile_profile_flag_get()) {
app_bt_ibrt_reconnect_mobile_profile_flag_clear();
#else
else if (bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_reconnecting) {
#endif
TRACE(2, "app_bt: hfp_act in NV =%d,a2dp_connect=%d",
btdevice_plf_p->hfp_act,
bt_profile_manager[id].hfp_connect);
if (btdevice_plf_p->hfp_act &&
bt_profile_manager[id].hfp_connect !=
bt_profile_connect_status_success) {
if (btif_hf_check_rfcomm_l2cap_channel_is_creating(
&bt_profile_manager[id].rmt_addr)) {
TRACE(0,
"!!!remote is creating hfp after a2dp connected\n");
} else {
TRACE(0, "!!!continue connect hfp\n");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[id].has_connected);
app_bt_HF_CreateServiceLink(
bt_profile_manager[id].chan,
(bt_bdaddr_t *)&bt_profile_manager[id].rmt_addr);
}
}
#if defined(__HSP_ENABLE__)
else if (btdevice_plf_p->hsp_act &&
bt_profile_manager[id].hsp_connect !=
bt_profile_connect_status_success) {
TRACE(0, "!!!continue connect hsp\n");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[id].has_connected);
app_bt_HS_CreateServiceLink(bt_profile_manager[id].hs_chan,
&bt_profile_manager[id].rmt_addr);
}
#endif
}
#ifdef __AUTO_CONNECT_OTHER_PROFILE__
else {
if (btdevice_plf_p->hfp_act &&
bt_profile_manager[id].hfp_connect !=
bt_profile_connect_status_success) {
bt_profile_manager[id].connect_timer_cb =
app_bt_profile_connect_hf_retry_timehandler;
app_bt_accessmode_set(BAM_CONNECTABLE_ONLY);
osTimerStart(bt_profile_manager[id].connect_timer,
APP_BT_PROFILE_CONNECT_RETRY_MS);
}
#if defined(__HSP_ENABLE__)
else if (btdevice_plf_p->hsp_act &&
bt_profile_manager[id].hsp_connect !=
bt_profile_connect_status_success) {
bt_profile_manager[id].connect_timer_cb =
app_bt_profile_connect_hs_retry_timehandler;
app_bt_accessmode_set(BAM_CONNECTABLE_ONLY);
osTimerStart(bt_profile_manager[id].connect_timer,
APP_BT_PROFILE_CONNECT_RETRY_MS);
}
#endif
}
#endif
#ifdef APP_DISABLE_PAGE_SCAN_AFTER_CONN
disable_page_scan_check_timer_start();
#endif
break;
case BTIF_A2DP_EVENT_STREAM_CLOSED:
TRACE(2, "%s A2DP_EVENT_STREAM_CLOSED discReason1:%d", __func__,
Info->discReason);
if (Info->subevt != A2DP_CONN_CLOSED) {
TRACE(0, "do not need set access mode");
return;
}
if (Stream != NULL) {
if (btif_a2dp_get_remote_device(Stream) != NULL)
TRACE(2, "%s A2DP_EVENT_STREAM_CLOSED discReason2:%d",
__func__,
btif_me_get_remote_device_disc_reason_saved(
btif_a2dp_get_remote_device(Stream)));
}
#if defined(IBRT)
if (app_bt_ibrt_reconnect_mobile_profile_flag_get()) {
app_bt_HF_CreateServiceLink(
bt_profile_manager[id].chan,
(bt_bdaddr_t *)&bt_profile_manager[id].rmt_addr);
}
#endif
bt_profile_manager[id].a2dp_connect = bt_profile_connect_status_failure;
if (bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_openreconnecting) {
if (++bt_profile_manager[id].reconnect_cnt <
APP_BT_PROFILE_OPENNING_RECONNECT_RETRY_LIMIT_CNT) {
app_bt_accessmode_set(BTIF_BAM_CONNECTABLE_ONLY);
profile_reconnect_enable = true;
bt_profile_manager[id].a2dp_connect =
bt_profile_connect_status_unknow;
}
} else if (bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_reconnecting) {
if (++bt_profile_manager[id].reconnect_cnt <
APP_BT_PROFILE_RECONNECT_RETRY_LIMIT_CNT) {
app_bt_accessmode_set(BTIF_BAM_CONNECTABLE_ONLY);
profile_reconnect_enable = true;
} else {
app_bt_restore_reconnecting_idle_mode(id);
// bt_profile_manager[id].reconnect_mode =
// bt_profile_reconnect_null;
}
TRACE(2, "%s try to reconnect cnt:%d", __func__,
bt_profile_manager[id].reconnect_cnt);
}
#if !defined(IBRT)
#if defined(ENHANCED_STACK)
else if (((Info->discReason == 0x08) || (Info->discReason == 0x04)) &&
#else
else if (((Info->discReason == 0x8) || (Info->discReason_saved == 0x8) ||
(Info->discReason_saved == 0x0)) &&
#endif
(btdevice_plf_p->a2dp_act) && (!btdevice_plf_p->hfp_act) &&
(!btdevice_plf_p->hsp_act)) {
bt_profile_manager[id].reconnect_mode =
bt_profile_reconnect_reconnecting;
TRACE(2, "%s try to reconnect cnt:%d", __func__,
bt_profile_manager[id].reconnect_cnt);
app_bt_accessmode_set(BTIF_BAM_CONNECTABLE_ONLY);
if (app_bt_profile_reconnect_pending(id) != 0) {
profile_reconnect_enable = true;
}
}
#endif
else {
bt_profile_manager[id].a2dp_connect =
bt_profile_connect_status_unknow;
}
if (profile_reconnect_enable) {
#ifdef __IAG_BLE_INCLUDE__
app_ble_refresh_adv_state(BLE_ADVERTISING_INTERVAL);
#endif
osTimerStart(bt_profile_manager[id].connect_timer,
APP_BT_PROFILE_RECONNECT_RETRY_INTERVAL_MS);
}
break;
default:
break;
}
}
if (bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_reconnecting) {
bool reconnect_hfp_proc_final = true;
bool reconnect_a2dp_proc_final = true;
if (bt_profile_manager[id].hfp_connect ==
bt_profile_connect_status_failure) {
reconnect_hfp_proc_final = false;
}
#if defined(__HSP_ENABLE__)
if (btdevice_plf_p->hsp_act != 0) // has HSP
{
reconnect_hfp_proc_final = true;
if (bt_profile_manager[id].hsp_connect ==
bt_profile_connect_status_failure) {
reconnect_hfp_proc_final = false;
}
}
#endif
if (bt_profile_manager[id].a2dp_connect ==
bt_profile_connect_status_failure) {
reconnect_a2dp_proc_final = false;
}
if (reconnect_hfp_proc_final && reconnect_a2dp_proc_final) {
TRACE(3, "!!!reconnect success %d/%d/%d\n",
bt_profile_manager[id].hfp_connect,
bt_profile_manager[id].hsp_connect,
bt_profile_manager[id].a2dp_connect);
app_bt_restore_reconnecting_idle_mode(id);
// bt_profile_manager[id].reconnect_mode = bt_profile_reconnect_null;
}
} else if (bt_profile_manager[id].reconnect_mode ==
bt_profile_reconnect_openreconnecting) {
bool opening_hfp_proc_final = false;
bool opening_a2dp_proc_final = false;
if (btdevice_plf_p->hfp_act && bt_profile_manager[id].hfp_connect ==
bt_profile_connect_status_unknow) {
opening_hfp_proc_final = false;
} else {
opening_hfp_proc_final = true;
}
if (btdevice_plf_p->a2dp_act && bt_profile_manager[id].a2dp_connect ==
bt_profile_connect_status_unknow) {
opening_a2dp_proc_final = false;
} else {
opening_a2dp_proc_final = true;
}
if ((opening_hfp_proc_final && opening_a2dp_proc_final) ||
(bt_profile_manager[id].a2dp_connect ==
bt_profile_connect_status_failure)) {
TRACE(3, "!!!reconnect success %d/%d/%d\n",
bt_profile_manager[id].hfp_connect,
bt_profile_manager[id].hsp_connect,
bt_profile_manager[id].a2dp_connect);
app_bt_restore_reconnecting_idle_mode(id);
// bt_profile_manager[id].reconnect_mode = bt_profile_reconnect_null;
}
if (btdevice_plf_p->a2dp_act && bt_profile_manager[id].a2dp_connect ==
bt_profile_connect_status_success) {
if (btdevice_plf_p->hfp_act && !opening_hfp_proc_final) {
if (btif_hf_check_rfcomm_l2cap_channel_is_creating(
&bt_profile_manager[id].rmt_addr)) {
TRACE(0, "!!!remote is creating hf after a2dp connected\n");
} else {
TRACE(0, "!!!continue connect hf\n");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[id].has_connected);
app_bt_HF_CreateServiceLink(
bt_profile_manager[id].chan,
(bt_bdaddr_t *)&bt_profile_manager[id].rmt_addr);
}
}
#if defined(__HSP_ENABLE)
else if (btdevice_plf_p->hsp_act && !opening_hfp_hsp_proc_final) {
TRACE(0, "!!!continue connect hs\n");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[id].has_connected);
app_bt_HS_CreateServiceLink(bt_profile_manager[id].hs_chan,
&bt_profile_manager[id].rmt_addr);
}
#endif
}
if (bt_profile_manager[id].reconnect_mode == bt_profile_reconnect_null) {
for (uint8_t i = 0; i < BT_DEVICE_NUM; i++) {
if (bt_profile_manager[i].reconnect_mode ==
bt_profile_reconnect_openreconnecting) {
TRACE(1, "!!!a2dp->start reconnect device %d\n", i);
if ((btdevice_plf_p->hfp_act) &&
(!bt_profile_manager[i].hfp_connect)) {
TRACE(0, "try connect hf");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[i].has_connected);
app_bt_HF_CreateServiceLink(
bt_profile_manager[i].chan,
(bt_bdaddr_t *)&bt_profile_manager[i].rmt_addr);
}
#if defined(__HSP_ENABLE__)
else if ((btdevice_plf_p->hsp_act) &&
(!bt_profile_manager[i].hsp_connect)) {
TRACE(0, "try connect hs");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[i].has_connected);
app_bt_HS_CreateServiceLink(
bt_profile_manager[i].hs_chan,
&bt_profile_manager[i].rmt_addr);
}
#endif
else if ((btdevice_plf_p->a2dp_act) &&
(!bt_profile_manager[i].a2dp_connect)) {
TRACE(0, "try connect a2dp");
app_bt_precheck_before_starting_connecting(
bt_profile_manager[i].has_connected);
app_bt_A2DP_OpenStream(bt_profile_manager[i].stream,
&bt_profile_manager[i].rmt_addr);
}
break;
}
}
}
}
#ifdef __INTERCONNECTION__
if (bt_profile_manager[id].hfp_connect == bt_profile_connect_status_success &&
bt_profile_manager[id].a2dp_connect ==
bt_profile_connect_status_success) {
app_interconnection_start_disappear_adv(
INTERCONNECTION_BLE_ADVERTISING_INTERVAL,
APP_INTERCONNECTION_DISAPPEAR_ADV_IN_MS);
if (btif_me_get_activeCons() <= 2) {
app_interconnection_spp_open(remDev);
}
}
#endif
#ifdef __IAG_BLE_INCLUDE__
TRACE(3, "%s hfp %d a2dp %d", __func__, bt_profile_manager[id].hfp_connect,
bt_profile_manager[id].a2dp_connect);
if (bt_profile_manager[id].hfp_connect == bt_profile_connect_status_success &&
#ifdef __HSP_ENABLE__
bt_profile_manager[id].hsp_connect == bt_profile_connect_status_success &&
#endif
bt_profile_manager[id].a2dp_connect ==
bt_profile_connect_status_success) {
app_ble_force_switch_adv(BLE_SWITCH_USER_BT_CONNECT, true);
}
#endif
if (!bt_profile_manager[id].has_connected &&
(bt_profile_manager[id].hfp_connect ==
bt_profile_connect_status_success ||
#ifdef __HSP_ENABLE__
bt_profile_manager[id].hsp_connect ==
bt_profile_connect_status_success ||
#endif
bt_profile_manager[id].a2dp_connect ==
bt_profile_connect_status_success)) {
bt_profile_manager[id].has_connected = true;
TRACE(0, "BT connected!!!");
IsMobileLinkLossing = FALSE;
#ifndef IBRT
btif_me_get_remote_device_name(&(bt_profile_manager[id].rmt_addr),
app_bt_global_handle);
#endif
#if defined(MEDIA_PLAYER_SUPPORT) //&& !defined(IBRT)
app_voice_report(APP_STATUS_INDICATION_CONNECTED, id);
#endif
#if 0 // #ifdef __INTERCONNECTION__
app_interconnection_start_disappear_adv(BLE_ADVERTISING_INTERVAL, APP_INTERCONNECTION_DISAPPEAR_ADV_IN_MS);
app_interconnection_spp_open();
#endif
#ifdef __INTERACTION__
// app_interaction_spp_open();
#endif
}
if (bt_profile_manager[id].has_connected &&
(bt_profile_manager[id].hfp_connect !=
bt_profile_connect_status_success &&
#ifdef __HSP_ENABLE__
bt_profile_manager[id].hsp_connect !=
bt_profile_connect_status_success &&
#endif
bt_profile_manager[id].a2dp_connect !=
bt_profile_connect_status_success)) {
bt_profile_manager[id].has_connected = false;
TRACE(0, "BT disconnected!!!");
#ifdef GFPS_ENABLED
if (app_gfps_is_last_response_pending()) {
app_gfps_enter_connectable_mode_req_handler(app_gfps_get_last_response());
}
#endif
#if defined(MEDIA_PLAYER_SUPPORT) && !defined(IBRT)
app_voice_report(APP_STATUS_INDICATION_DISCONNECTED, id);
#endif
#ifdef __INTERCONNECTION__
app_interconnection_disconnected_callback();
#endif
app_set_disconnecting_all_bt_connections(false);
}
#ifdef __BT_ONE_BRING_TWO__
app_bt_update_connectable_mode_after_connection_management();
#endif
}
#ifdef BTIF_HID_DEVICE
void hid_exit_shutter_mode(void);
#endif
static bool isDisconnectAllBtConnections = false;
bool app_is_disconnecting_all_bt_connections(void) {
return isDisconnectAllBtConnections;
}
void app_set_disconnecting_all_bt_connections(bool isEnable) {
isDisconnectAllBtConnections = isEnable;
}
bt_status_t LinkDisconnectDirectly(bool PowerOffFlag) {
app_set_disconnecting_all_bt_connections(true);
// TRACE(1,"osapi_lock_is_exist:%d",osapi_lock_is_exist());
if (osapi_lock_is_exist())
osapi_lock_stack();
#ifdef __IAG_BLE_INCLUDE__
TRACE(1, "ble_connected_state:%d", app_ble_is_any_connection_exist());
#endif
#if defined(IBRT)
ibrt_ctrl_t *p_ibrt_ctrl = app_tws_ibrt_get_bt_ctrl_ctx();
if (true == PowerOffFlag)
p_ibrt_ctrl->ibrt_in_poweroff = true;
if (p_ibrt_ctrl->init_done) {
if (IBRT_MASTER == p_ibrt_ctrl->current_role) {
if (app_tws_ibrt_mobile_link_connected()) {
// should check return status
app_tws_ibrt_disconnect_connection(
btif_me_get_remote_device_by_handle(
p_ibrt_ctrl->mobile_conhandle));
}
}
if (app_tws_ibrt_tws_link_connected()) {
app_tws_ibrt_disconnect_connection(
btif_me_get_remote_device_by_handle(p_ibrt_ctrl->tws_conhandle));
}
}
if (osapi_lock_is_exist())
osapi_unlock_stack();
osDelay(500);
return BT_STS_SUCCESS;
#endif
TRACE(1, "activeCons:%d", btif_me_get_activeCons());
uint8_t Tmp_activeCons = btif_me_get_activeCons();
if (Tmp_activeCons) {
// TRACE(3,"%s id1 hf:%d a2dp:%d",__func__,
// app_bt_device.hf_channel[BT_DEVICE_ID_1].state,
// btif_a2dp_get_stream_state(app_bt_device.a2dp_stream[BT_DEVICE_ID_1]->a2dp_stream));
TRACE(3, "%s id1 hf:%d a2dp:%d", __func__,
btif_get_hf_chan_state(app_bt_device.hf_channel[BT_DEVICE_ID_1]),
btif_a2dp_get_stream_state(
app_bt_device.a2dp_stream[BT_DEVICE_ID_1]->a2dp_stream));
#ifdef BTIF_HID_DEVICE
hid_exit_shutter_mode();
#endif
if (btif_get_hf_chan_state(app_bt_device.hf_channel[BT_DEVICE_ID_1]) ==
BTIF_HF_STATE_OPEN) {
app_bt_HF_DisconnectServiceLink(app_bt_device.hf_channel[BT_DEVICE_ID_1]);
}
#if defined(__HSP_ENABLE__)
if (app_bt_device.hs_channel[BT_DEVICE_ID_1].state == HS_STATE_OPEN) {
app_bt_HS_DisconnectServiceLink(
&app_bt_device.hs_channel[BT_DEVICE_ID_1]);
}
#endif // btif_a2dp_get_stream_state(app_bt_device.a2dp_stream[device_id]->a2dp_stream)
if (btif_a2dp_get_stream_state(
app_bt_device.a2dp_stream[BT_DEVICE_ID_1]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_STREAMING ||
btif_a2dp_get_stream_state(
app_bt_device.a2dp_stream[BT_DEVICE_ID_1]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_OPEN) {
app_bt_A2DP_CloseStream(
app_bt_device.a2dp_stream[BT_DEVICE_ID_1]->a2dp_stream);
}
#if defined(A2DP_LHDC_ON)
if (btif_a2dp_get_stream_state(
app_bt_device.a2dp_lhdc_stream[BT_DEVICE_ID_1]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_STREAMING ||
btif_a2dp_get_stream_state(
app_bt_device.a2dp_lhdc_stream[BT_DEVICE_ID_1]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_OPEN) {
app_bt_A2DP_CloseStream(
app_bt_device.a2dp_lhdc_stream[BT_DEVICE_ID_1]->a2dp_stream);
}
#endif
#if defined(A2DP_LDAC_ON)
if (btif_a2dp_get_stream_state(
app_bt_device.a2dp_ldac_stream[BT_DEVICE_ID_1]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_STREAMING ||
btif_a2dp_get_stream_state(
app_bt_device.a2dp_ldac_stream[BT_DEVICE_ID_1]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_OPEN) {
app_bt_A2DP_CloseStream(
app_bt_device.a2dp_ldac_stream[BT_DEVICE_ID_1]->a2dp_stream);
}
#endif
#if defined(A2DP_AAC_ON)
if (btif_a2dp_get_stream_state(
app_bt_device.a2dp_aac_stream[BT_DEVICE_ID_1]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_STREAMING ||
btif_a2dp_get_stream_state(
app_bt_device.a2dp_aac_stream[BT_DEVICE_ID_1]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_OPEN) {
app_bt_A2DP_CloseStream(
app_bt_device.a2dp_aac_stream[BT_DEVICE_ID_1]->a2dp_stream);
}
#endif
#if defined(A2DP_SCALABLE_ON)
if (btif_a2dp_get_stream_state(
app_bt_device.a2dp_scalable_stream[BT_DEVICE_ID_1]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_STREAMING ||
btif_a2dp_get_stream_state(
app_bt_device.a2dp_scalable_stream[BT_DEVICE_ID_1]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_OPEN) {
app_bt_A2DP_CloseStream(
app_bt_device.a2dp_scalable_stream[BT_DEVICE_ID_1]->a2dp_stream);
}
#endif
if (btif_avrcp_get_remote_device(
app_bt_device.avrcp_channel[BT_DEVICE_ID_1]
->avrcp_channel_handle)) {
btif_avrcp_disconnect(
app_bt_device.avrcp_channel[BT_DEVICE_ID_1]->avrcp_channel_handle);
}
#ifdef __BT_ONE_BRING_TWO__
TRACE(3, "%s id2 hf:%d a2dp:%d", __func__,
btif_get_hf_chan_state(app_bt_device.hf_channel[BT_DEVICE_ID_2]),
btif_a2dp_get_stream_state(
app_bt_device.a2dp_stream[BT_DEVICE_ID_2]->a2dp_stream));
// if(app_bt_device.hf_channel[BT_DEVICE_ID_2].state == HF_STATE_OPEN){
if (btif_get_hf_chan_state(app_bt_device.hf_channel[BT_DEVICE_ID_2]) ==
BTIF_HF_STATE_OPEN) {
app_bt_HF_DisconnectServiceLink(app_bt_device.hf_channel[BT_DEVICE_ID_2]);
}
#if defined(__HSP_ENABLE__)
if (app_bt_device.hs_channel[BT_DEVICE_ID_2].state == HS_STATE_OPEN) {
app_bt_HS_DisconnectServiceLink(
&app_bt_device.hs_channel[BT_DEVICE_ID_2]);
}
#endif // __HSP_ENABLE__
if (btif_a2dp_get_stream_state(
app_bt_device.a2dp_stream[BT_DEVICE_ID_2]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_STREAMING ||
btif_a2dp_get_stream_state(
app_bt_device.a2dp_stream[BT_DEVICE_ID_2]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_OPEN) {
app_bt_A2DP_CloseStream(
app_bt_device.a2dp_stream[BT_DEVICE_ID_2]->a2dp_stream);
}
#if defined(A2DP_LHDC_ON)
if (btif_a2dp_get_stream_state(
app_bt_device.a2dp_lhdc_stream[BT_DEVICE_ID_2]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_STREAMING ||
btif_a2dp_get_stream_state(
app_bt_device.a2dp_lhdc_stream[BT_DEVICE_ID_2]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_OPEN) {
app_bt_A2DP_CloseStream(
app_bt_device.a2dp_lhdc_stream[BT_DEVICE_ID_2]->a2dp_stream);
}
#endif // A2DP_LHDC_ON
#if defined(A2DP_AAC_ON)
if (btif_a2dp_get_stream_state(
app_bt_device.a2dp_aac_stream[BT_DEVICE_ID_2]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_STREAMING ||
btif_a2dp_get_stream_state(
app_bt_device.a2dp_aac_stream[BT_DEVICE_ID_2]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_OPEN) {
app_bt_A2DP_CloseStream(
app_bt_device.a2dp_aac_stream[BT_DEVICE_ID_2]->a2dp_stream);
}
#endif // A2DP_AAC_ON
#if defined(A2DP_SCALABLE_ON)
if (btif_a2dp_get_stream_state(
app_bt_device.a2dp_scalable_stream[BT_DEVICE_ID_2]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_STREAMING ||
btif_a2dp_get_stream_state(
app_bt_device.a2dp_scalable_stream[BT_DEVICE_ID_2]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_OPEN) {
app_bt_A2DP_CloseStream(
app_bt_device.a2dp_scalable_stream[BT_DEVICE_ID_2]->a2dp_stream);
}
#endif // A2DP_SCALABLE_ON
if (btif_avrcp_get_remote_device(
app_bt_device.avrcp_channel[BT_DEVICE_ID_2]
->avrcp_channel_handle)) {
btif_avrcp_disconnect(
app_bt_device.avrcp_channel[BT_DEVICE_ID_2]->avrcp_channel_handle);
}
#endif //__BT_ONE_BRING_TWO__
#ifdef BISTO_ENABLED
gsound_custom_bt_disconnect_all_channel();
#endif
}
#ifdef __IAG_BLE_INCLUDE__
if (app_ble_is_any_connection_exist()) {
#ifdef GFPS_ENABLED
if (!app_gfps_is_last_response_pending())
#endif
app_ble_disconnect_all();
}
#endif
if (osapi_lock_is_exist())
osapi_unlock_stack();
osDelay(500);
if (Tmp_activeCons) {
btif_remote_device_t *remDev = app_bt_get_remoteDev(BT_DEVICE_ID_1);
if (NULL != remDev) {
app_bt_MeDisconnectLink(remDev);
}
#ifdef __BT_ONE_BRING_TWO__
remDev = app_bt_get_remoteDev(BT_DEVICE_ID_2);
if (NULL != remDev) {
osDelay(200);
app_bt_MeDisconnectLink(remDev);
}
#endif
}
return BT_STS_SUCCESS;
}
void app_disconnect_all_bt_connections(void) { LinkDisconnectDirectly(false); }
static int app_custom_function_process(APP_MESSAGE_BODY *msg_body) {
APP_APPTHREAD_REQ_CUSTOMER_CALL_FN_T customer_call =
(APP_APPTHREAD_REQ_CUSTOMER_CALL_FN_T)(msg_body->message_ptr);
TRACE(4, "func:0x%08x,param0:0x%08x, param1:0x%08x", msg_body->message_ptr,
msg_body->message_Param0, msg_body->message_Param1);
if (customer_call) {
customer_call((void *)msg_body->message_Param0,
(void *)msg_body->message_Param1);
}
return 0;
}
int app_bt_start_custom_function_in_app_thread(uint32_t param0, uint32_t param1,
uint32_t funcPtr) {
APP_MESSAGE_BLOCK msg;
msg.mod_id = APP_MODUAL_CUSTOM_FUNCTION;
msg.msg_body.message_id = 0;
msg.msg_body.message_ptr = funcPtr;
msg.msg_body.message_Param0 = param0;
msg.msg_body.message_Param1 = param1;
app_mailbox_put(&msg);
return 0;
}
void app_bt_init(void) {
app_bt_mail_init();
app_set_threadhandle(APP_MODUAL_BT, app_bt_handle_process);
btif_me_sec_set_io_cap_rsp_reject_ext(
app_bt_profile_connect_openreconnecting);
app_bt_active_mode_manager_init();
app_set_threadhandle(APP_MODUAL_CUSTOM_FUNCTION, app_custom_function_process);
}
extern "C" bool app_bt_has_connectivitys(void) {
int activeCons;
osapi_lock_stack();
activeCons = btif_me_get_activeCons();
osapi_unlock_stack();
if (activeCons > 0)
return true;
return false;
#if 0
if(app_bt_device.hf_channel[BT_DEVICE_ID_1].cmgrHandler.remDev)
return true;
if(app_bt_device.a2dp_stream[BT_DEVICE_ID_1].device->cmgrHandler.remDev)
return true;
#ifdef __BT_ONE_BRING_TWO__
if(app_bt_device.hf_channel[BT_DEVICE_ID_2].cmgrHandler.remDev)
return true;
if(app_bt_device.a2dp_stream[BT_DEVICE_ID_2].device->cmgrHandler.remDev)
return true;
#endif
return false;
#endif
}
#ifdef __TWS_CHARGER_BOX__
extern "C" {
bt_status_t ME_Ble_Clear_Whitelist(void);
bt_status_t ME_Ble_Set_Private_Address(BT_BD_ADDR *addr);
bt_status_t ME_Ble_Add_Dev_To_Whitelist(U8 addr_type, BT_BD_ADDR *addr);
bt_status_t ME_Ble_SetAdv_data(U8 len, U8 *data);
bt_status_t ME_Ble_SetScanRsp_data(U8 len, U8 *data);
bt_status_t ME_Ble_SetAdv_parameters(adv_para_struct *para);
bt_status_t ME_Ble_SetAdv_en(U8 en);
bt_status_t ME_Ble_Setscan_parameter(scan_para_struct *para);
bt_status_t ME_Ble_Setscan_en(U8 scan_en, U8 filter_duplicate);
}
int8_t power_level = 0;
#define TWS_BOX_OPEN 1
#define TWS_BOX_CLOSE 0
void app_tws_box_set_slave_adv_data(uint8_t power_level, uint8_t box_status) {
uint8_t adv_data[] = {
0x02, 0xfe, 0x00, 0x02, 0xfd, 0x00 // manufacturer data
};
adv_data[2] = power_level;
adv_data[5] = box_status;
ME_Ble_SetAdv_data(sizeof(adv_data), adv_data);
}
void app_tws_box_set_slave_adv_para(void) {
uint8_t peer_addr[BTIF_BD_ADDR_SIZE] = {0};
adv_para_struct para;
para.interval_min = 0x0040; // 20ms
para.interval_max = 0x0040; // 20ms
para.adv_type = 0x03;
para.own_addr_type = 0x01;
para.peer_addr_type = 0x01;
para.adv_chanmap = 0x07;
para.adv_filter_policy = 0x00;
memcpy(para.bd_addr.addr, peer_addr, BTIF_BD_ADDR_SIZE);
ME_Ble_SetAdv_parameters(&para);
}
extern uint8_t bt_addr[6];
void app_tws_start_chargerbox_adv(void) {
app_tws_box_set_slave_adv_data(power_level, TWS_BOX_OPEN);
ME_Ble_Set_Private_Address((BT_BD_ADDR *)bt_addr);
app_tws_box_set_slave_adv_para();
ME_Ble_SetAdv_en(1);
}
#endif
bool app_is_hfp_service_connected(void) {
return (bt_profile_manager[BT_DEVICE_ID_1].hfp_connect ==
bt_profile_connect_status_success);
}
btif_remote_device_t *app_bt_get_remoteDev(uint8_t deviceId) {
btif_remote_device_t *currentRemDev = NULL;
if (btif_a2dp_get_stream_state(
app_bt_device.a2dp_stream[deviceId]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_STREAMING ||
btif_a2dp_get_stream_state(
app_bt_device.a2dp_stream[deviceId]->a2dp_stream) ==
BTIF_AVDTP_STRM_STATE_OPEN) {
currentRemDev = btif_a2dp_get_stream_conn_remDev(
app_bt_device.a2dp_stream[deviceId]->a2dp_stream);
} else if (btif_get_hf_chan_state(app_bt_device.hf_channel[deviceId]) ==
BTIF_HF_STATE_OPEN) {
currentRemDev = (btif_remote_device_t *)btif_hf_cmgr_get_remote_device(
app_bt_device.hf_channel[deviceId]);
}
TRACE(2, "%s get current Remdev %p", __FUNCTION__, currentRemDev);
return currentRemDev;
}
void app_bt_stay_active_rem_dev(btif_remote_device_t *pRemDev) {
if (pRemDev) {
btif_cmgr_handler_t *cmgrHandler;
/* Clear the sniff timer */
cmgrHandler = btif_cmgr_get_acl_handler(pRemDev);
btif_cmgr_clear_sniff_timer(cmgrHandler);
btif_cmgr_disable_sniff_timer(cmgrHandler);
app_bt_Me_SetLinkPolicy(pRemDev, BTIF_BLP_MASTER_SLAVE_SWITCH);
}
}
void app_bt_stay_active(uint8_t deviceId) {
btif_remote_device_t *currentRemDev = app_bt_get_remoteDev(deviceId);
app_bt_stay_active_rem_dev(currentRemDev);
}
void app_bt_allow_sniff_rem_dev(btif_remote_device_t *pRemDev) {
if (pRemDev &&
(BTIF_BDS_CONNECTED == btif_me_get_remote_device_state(pRemDev))) {
btif_cmgr_handler_t *cmgrHandler;
/* Enable the sniff timer */
cmgrHandler = btif_cmgr_get_acl_handler(pRemDev);
/* Start the sniff timer */
btif_sniff_info_t sniffInfo;
sniffInfo.minInterval = BTIF_CMGR_SNIFF_MIN_INTERVAL;
sniffInfo.maxInterval = BTIF_CMGR_SNIFF_MAX_INTERVAL;
sniffInfo.attempt = BTIF_CMGR_SNIFF_ATTEMPT;
sniffInfo.timeout = BTIF_CMGR_SNIFF_TIMEOUT;
if (cmgrHandler) {
btif_cmgr_set_sniff_timer(cmgrHandler, &sniffInfo, BTIF_CMGR_SNIFF_TIMER);
}
app_bt_Me_SetLinkPolicy(pRemDev,
BTIF_BLP_MASTER_SLAVE_SWITCH | BTIF_BLP_SNIFF_MODE);
}
}
extern "C" uint8_t is_sco_mode(void);
void app_bt_allow_sniff(uint8_t deviceId) {
if (a2dp_is_music_ongoing() || is_sco_mode()) {
return;
}
btif_remote_device_t *currentRemDev = app_bt_get_remoteDev(deviceId);
app_bt_allow_sniff_rem_dev(currentRemDev);
}
void app_bt_stop_sniff(uint8_t deviceId) {
btif_remote_device_t *currentRemDev = app_bt_get_remoteDev(deviceId);
if (currentRemDev &&
(btif_me_get_remote_device_state(currentRemDev) == BTIF_BDS_CONNECTED)) {
if (btif_me_get_current_mode(currentRemDev) == BTIF_BLM_SNIFF_MODE) {
TRACE(1, "!!! stop sniff currmode:%d\n",
btif_me_get_current_mode(currentRemDev));
app_bt_ME_StopSniff(currentRemDev);
}
}
}
bool app_bt_is_device_connected(uint8_t deviceId) {
if (deviceId < BT_DEVICE_NUM) {
return bt_profile_manager[deviceId].has_connected;
} else {
// Indicate no connection is user passes invalid deviceId
return false;
}
}
#if defined(__BT_SELECT_PROF_DEVICE_ID__)
int8_t app_bt_a2dp_is_same_stream(a2dp_stream_t *src_Stream,
a2dp_stream_t *dst_Stream) {
return btif_a2dp_is_register_codec_same(src_Stream, dst_Stream);
}
void app_bt_a2dp_find_same_unused_stream(a2dp_stream_t *in_Stream,
a2dp_stream_t **out_Stream,
uint32_t device_id) {
*out_Stream = NULL;
if (app_bt_a2dp_is_same_stream(
app_bt_device.a2dp_stream[device_id]->a2dp_stream, in_Stream))
*out_Stream = app_bt_device.a2dp_stream[device_id]->a2dp_stream;
#if defined(A2DP_LHDC_ON)
else if (app_bt_a2dp_is_same_stream(
app_bt_device.a2dp_lhdc_stream[device_id]->a2dp_stream,
in_Stream))
*out_Stream = app_bt_device.a2dp_lhdc_stream[device_id]->a2dp_stream;
#endif
#if defined(A2DP_LDAC_ON)
else if (app_bt_a2dp_is_same_stream(
app_bt_device.a2dp_ldac_stream[device_id]->a2dp_stream,
in_Stream))
*out_Stream = app_bt_device.a2dp_ldac_stream[device_id]->a2dp_stream;
#endif
#if defined(A2DP_AAC_ON)
else if (app_bt_a2dp_is_same_stream(
app_bt_device.a2dp_aac_stream[device_id]->a2dp_stream,
in_Stream))
*out_Stream = app_bt_device.a2dp_aac_stream[device_id]->a2dp_stream;
#endif
#if defined(A2DP_SCALABLE_ON)
else if (app_bt_a2dp_is_same_stream(
app_bt_device.a2dp_scalable_stream[device_id]->a2dp_stream,
in_Stream))
*out_Stream = app_bt_device.a2dp_scalable_stream[device_id]->a2dp_stream;
#endif
}
int8_t app_bt_a2dp_is_stream_on_device_id(a2dp_stream_t *in_Stream,
uint32_t device_id) {
if (app_bt_device.a2dp_stream[device_id]->a2dp_stream == in_Stream)
return 1;
#if defined(A2DP_LHDC_ON)
else if (app_bt_device.a2dp_lhdc_stream[device_id]->a2dp_stream == in_Stream)
return 1;
#endif
#if defined(A2DP_LDAC_ON)
else if (app_bt_device.a2dp_ldac_stream[device_id]->a2dp_stream == in_Stream)
return 1;
#endif
#if defined(A2DP_AAC_ON)
else if (app_bt_device.a2dp_aac_stream[device_id]->a2dp_stream == in_Stream)
return 1;
#endif
#if defined(A2DP_SCALABLE_ON)
else if (app_bt_device.a2dp_scalable_stream[device_id]->a2dp_stream ==
in_Stream)
return 1;
#endif
return 0;
}
int8_t app_bt_hfp_is_chan_on_device_id(hf_chan_handle_t chan,
uint32_t device_id) {
if (app_bt_device.hf_channel[device_id] == chan)
return 1;
return 0;
}
int8_t app_bt_is_any_profile_connected(uint32_t device_id) {
// TODO avrcp?spp?hid?bisto?ama?dma?rfcomm?
if ((bt_profile_manager[device_id].hfp_connect ==
bt_profile_connect_status_success) ||
(bt_profile_manager[device_id].hsp_connect ==
bt_profile_connect_status_success) ||
(bt_profile_manager[device_id].a2dp_connect ==
bt_profile_connect_status_success)) {
return 1;
}
return 0;
}
int8_t app_bt_is_a2dp_connected(uint32_t device_id) {
if (bt_profile_manager[device_id].a2dp_connect ==
bt_profile_connect_status_success) {
return 1;
}
return 0;
}
btif_remote_device_t *app_bt_get_connected_profile_remdev(uint32_t device_id) {
if (bt_profile_manager[device_id].a2dp_connect ==
bt_profile_connect_status_success) {
return (btif_remote_device_t *)btif_a2dp_get_remote_device(
app_bt_device.a2dp_connected_stream[device_id]);
} else if (bt_profile_manager[device_id].hfp_connect ==
bt_profile_connect_status_success) {
return (btif_remote_device_t *)btif_hf_cmgr_get_remote_device(
app_bt_device.hf_channel[device_id]);
}
#if defined(__HSP_ENABLE__)
else if (bt_profile_manager[device_id].hsp_connect ==
bt_profile_connect_status_success) {
// TODO hsp support
// return (btif_remote_device_t
// *)btif_hs_cmgr_get_remote_device(app_bt_device.hs_channel[i]);
}
#endif
return NULL;
}
#endif
bool app_bt_get_device_bdaddr(uint8_t deviceId, uint8_t *btAddr) {
bool ret = false;
if (app_bt_is_device_connected(deviceId)) {
btif_remote_device_t *currentRemDev = app_bt_get_remoteDev(deviceId);
if (currentRemDev) {
memcpy(btAddr, btif_me_get_remote_device_bdaddr(currentRemDev)->address,
BTIF_BD_ADDR_SIZE);
ret = true;
}
}
return ret;
}
void fast_pair_enter_pairing_mode_handler(void) {
#if defined(IBRT)
app_ibrt_ui_judge_scan_type(IBRT_FASTPAIR_TRIGGER, MOBILE_LINK, 0);
#else
app_bt_accessmode_set(BTIF_BAM_GENERAL_ACCESSIBLE);
#endif
#ifdef __INTERCONNECTION__
clear_discoverable_adv_timeout_flag();
app_interceonnection_start_discoverable_adv(
INTERCONNECTION_BLE_FAST_ADVERTISING_INTERVAL,
APP_INTERCONNECTION_FAST_ADV_TIMEOUT_IN_MS);
#endif
}
bool app_bt_is_hfp_audio_on(void) {
bool hfp_audio_is_on = false;
for (uint8_t i = 0; i < BT_DEVICE_NUM; i++) {
if (BTIF_HF_AUDIO_CON == app_bt_device.hf_audio_state[i]) {
hfp_audio_is_on = true;
break;
}
}
return hfp_audio_is_on;
}
btif_remote_device_t *app_bt_get_connected_mobile_device_ptr(void) {
return connectedMobile;
}
void app_bt_set_spp_device_ptr(btif_remote_device_t *device) {
TRACE(2, "%s set sppOpenMobile is %p", __func__, device);
sppOpenMobile = device;
return;
}
btif_remote_device_t *app_bt_get_spp_device_ptr(void) {
TRACE(2, "%s sppOpenMobile %p", __func__, sppOpenMobile);
ASSERT((sppOpenMobile != NULL), "sppOpenMobile is NULL!!!!!!!!!!!!");
return sppOpenMobile;
}
#ifdef BT_USB_AUDIO_DUAL_MODE
#include "a2dp_api.h"
extern "C" a2dp_stream_t *app_bt_get_steam(enum BT_DEVICE_ID_T id) {
a2dp_stream_t *stream;
stream = (a2dp_stream_t *)bt_profile_manager[id].stream;
return stream;
}
extern "C" int app_bt_get_bt_addr(enum BT_DEVICE_ID_T id, bt_bdaddr_t *bdaddr) {
memcpy(bdaddr, &bt_profile_manager[id].rmt_addr, sizeof(bt_bdaddr_t));
return 0;
}
extern "C" bool app_bt_a2dp_service_is_connected(void) {
return (bt_profile_manager[BT_DEVICE_ID_1].a2dp_connect ==
bt_profile_connect_status_success);
}
#endif
struct app_bt_search_t {
bool search_start;
bool inquiry_pending;
bool device_searched;
bt_bdaddr_t address;
};
static bool app_bt_search_device_match(const bt_bdaddr_t *addr,
const char *name) {
TRACE(7,
"app_bt_search_callback found device %02x:%02x:%02x:%02x:%02x:%02x "
"'%s'\n",
addr->address[0], addr->address[1], addr->address[2], addr->address[3],
addr->address[4], addr->address[5], name);
#if defined(HFP_MOBILE_AG_ROLE)
bt_bdaddr_t test_device1 = {{0xd2, 0x53, 0x86, 0x42, 0x71, 0x31}};
bt_bdaddr_t test_device2 = {{0xd3, 0x53, 0x86, 0x42, 0x71, 0x31}};
return (memcmp(addr, test_device1.address, sizeof(bt_bdaddr_t)) == 0 ||
memcmp(addr, test_device2.address, sizeof(bt_bdaddr_t)) == 0);
#else
return false;
#endif
}
static struct app_bt_search_t g_bt_search;
static void app_bt_search_callback(const btif_event_t *event) {
TRACE(2, "%s event %d\n", __func__, btif_me_get_callback_event_type(event));
switch (btif_me_get_callback_event_type(event)) {
case BTIF_BTEVENT_INQUIRY_RESULT: {
bt_bdaddr_t *addr = btif_me_get_callback_event_inq_result_bd_addr(event);
uint8_t mode = btif_me_get_callback_event_inq_result_inq_mode(event);
const int NAME_MAX_LEN = 255;
char device_name[NAME_MAX_LEN + 1] = {0};
int device_name_len = 0;
uint8_t *eir = NULL;
if ((mode == BTIF_INQ_MODE_EXTENDED) &&
(eir = btif_me_get_callback_event_inq_result_ext_inq_resp(event))) {
device_name_len = btif_me_get_ext_inq_data(
eir, 0x09, (uint8_t *)device_name, NAME_MAX_LEN);
}
if (app_bt_search_device_match(addr,
device_name_len > 0 ? device_name : "")) {
g_bt_search.address = *addr;
g_bt_search.device_searched = true;
btif_me_cancel_inquiry();
}
} break;
case BTIF_BTEVENT_INQUIRY_COMPLETE:
case BTIF_BTEVENT_INQUIRY_CANCELED:
btif_me_unregister_globa_handler((btif_handler *)btif_me_get_bt_handler());
g_bt_search.search_start = false;
g_bt_search.inquiry_pending = false;
if (g_bt_search.device_searched) {
#if defined(HFP_MOBILE_AG_ROLE)
bt_profile_manager[BT_DEVICE_ID_1].reconnect_mode =
bt_profile_reconnect_null;
bt_profile_manager[BT_DEVICE_ID_1].rmt_addr = g_bt_search.address;
bt_profile_manager[BT_DEVICE_ID_1].chan =
app_bt_device.hf_channel[BT_DEVICE_ID_1];
app_bt_precheck_before_starting_connecting(
bt_profile_manager[BT_DEVICE_ID_1].has_connected);
app_bt_HF_CreateServiceLink(bt_profile_manager[BT_DEVICE_ID_1].chan,
&bt_profile_manager[BT_DEVICE_ID_1].rmt_addr);
#endif
} else {
TRACE(1, "%s no device matched\n", __func__);
#if 0
/* continue to search ??? */
app_bt_start_search();
#endif
}
break;
default:
break;
}
}
void app_bt_start_search(void) {
uint8_t max_search_time = 10; /* 12.8s */
if (g_bt_search.search_start) {
TRACE(1, "%s already started\n", __func__);
return;
}
btif_me_set_handler(btif_me_get_bt_handler(), app_bt_search_callback);
btif_me_set_event_mask(
btif_me_get_bt_handler(),
BTIF_BEM_INQUIRY_RESULT | BTIF_BEM_INQUIRY_COMPLETE |
BTIF_BEM_INQUIRY_CANCELED | BTIF_BEM_LINK_CONNECT_IND |
BTIF_BEM_LINK_CONNECT_CNF | BTIF_BEM_LINK_DISCONNECT |
BTIF_BEM_ROLE_CHANGE | BTIF_BEM_MODE_CHANGE);
btif_me_register_global_handler(btif_me_get_bt_handler());
g_bt_search.search_start = true;
g_bt_search.device_searched = false;
g_bt_search.inquiry_pending = false;
if (BT_STS_PENDING != btif_me_inquiry(BTIF_BT_IAC_GIAC, max_search_time, 0)) {
TRACE(1, "%s start inquiry failed\n", __func__);
g_bt_search.inquiry_pending = true;
}
}
uint8_t app_bt_avrcp_get_notify_trans_id(void) {
return btif_a2dp_get_avrcpadvancedpdu_trans_id(
app_bt_device.avrcp_notify_rsp[BT_DEVICE_ID_1]);
}
void app_bt_avrcp_set_notify_trans_id(uint8_t trans_id) {
TRACE(3, "%s %d %p\n", __func__, trans_id,
app_bt_device.avrcp_notify_rsp[BT_DEVICE_ID_1]);
btif_a2dp_set_avrcpadvancedpdu_trans_id(
app_bt_device.avrcp_notify_rsp[BT_DEVICE_ID_1], trans_id);
}
uint8_t app_bt_avrcp_get_ctl_trans_id(void) {
return btif_avrcp_get_ctl_trans_id(
app_bt_device.avrcp_channel[BT_DEVICE_ID_1]);
}
void app_bt_avrcp_set_ctl_trans_id(uint8_t trans_id) {
TRACE(3, "%s %d %p\n", __func__, trans_id,
app_bt_device.avrcp_channel[BT_DEVICE_ID_1]);
btif_avrcp_set_ctl_trans_id(app_bt_device.avrcp_channel[BT_DEVICE_ID_1],
trans_id);
}
void app_bt_set_mobile_a2dp_stream(uint32_t deviceId, a2dp_stream_t *stream) {
bt_profile_manager[deviceId].stream = stream;
}
#if defined(IBRT)
#if defined(ENHANCED_STACK)
uint32_t app_bt_save_spp_app_ctx(uint32_t app_id, btif_remote_device_t *rem_dev,
uint8_t *buf, uint32_t buf_len) {
bt_bdaddr_t *remote = NULL;
uint32_t offset = 0;
struct spp_device *device = (struct spp_device *)btif_spp_get_device(app_id);
ASSERT(device, "%s NULL spp device app_id=0x%x", __func__, app_id);
// save app_id
buf[offset++] = app_id & 0xFF;
buf[offset++] = (app_id >> 8) & 0xFF;
buf[offset++] = (app_id >> 16) & 0xFF;
buf[offset++] = (app_id >> 24) & 0xFF;
// save port type
buf[offset++] = device->portType;
// save remote address
remote = btif_me_get_remote_device_bdaddr(rem_dev);
memcpy(buf + offset, remote, sizeof(bt_bdaddr_t));
offset += sizeof(bt_bdaddr_t);
// TRACE(7,"%s:%02x:%02x:%02x:%02x:%02x:%02x\r\n",
// __func__, remote->addr[5], remote->addr[4], remote->addr[3],
// remote->addr[2], remote->addr[1], remote->addr[0]);
// spp device
buf[offset++] = device->spp_connected_flag;
return offset;
}
uint32_t app_bt_restore_spp_app_ctx(uint8_t *buf, uint32_t buf_len,
uint32_t app_id) {
bt_bdaddr_t remote;
uint32_t offset = 0;
struct spp_device *device = NULL;
uint8_t i = 0;
uint8_t port_type = 0;
uint32_t app_id_restore = 0;
// restore app_id
for (i = 0; i < 4; i++) {
app_id_restore += (buf[offset + i] << (8 * i));
}
offset += 4;
port_type = buf[offset++];
ASSERT(app_id_restore == app_id, "%s,spp app id mismatch=%x,%x", __func__,
app_id_restore, app_id);
// restore remote address
memcpy(&remote, buf + offset, sizeof(remote));
offset += sizeof(remote);
device = (struct spp_device *)btif_spp_get_device(app_id);
#ifdef __INTERCONNECTION__
TRACE(1, "%s,%x,%x,%x", __func__, app_id, BTIF_APP_SPP_CLIENT_CCMP_ID,
BTIF_APP_SPP_CLIENT_RED_ID);
#endif
if (device == NULL) {
/*
* SPP client device may not be created in bt host initialized stage,so IBRT
* SLAVE will restore it
*/
if (port_type == BTIF_SPP_CLIENT_PORT) {
TRACE(1, "%s,spp client device null", __func__);
switch (app_id) {
#ifdef __INTERCONNECTION__
case BTIF_APP_SPP_CLIENT_CCMP_ID:
app_ccmp_client_open((uint8_t *)SppServiceSearchReq,
app_interconnection_get_length(), 0, 1);
device = (struct spp_device *)btif_spp_get_device(
BTIF_APP_SPP_CLIENT_CCMP_ID);
device->spp_callback = ccmp_callback;
// device->_channel = chnl; //restore in
// btif_spp_profile_restore_ctx
device->sppUsedFlag = 1;
break;
case BTIF_APP_SPP_CLIENT_RED_ID:
app_spp_client_open((uint8_t *)SppServiceSearchReq,
app_interconnection_get_length(), 1);
device = (struct spp_device *)btif_spp_get_device(
BTIF_APP_SPP_CLIENT_RED_ID);
device->spp_callback = spp_client_callback;
// device->_channel = chnl; //restore in
// btif_spp_profile_restore_ctx
device->sppUsedFlag = 1;
break;
#endif
default:
ASSERT(device, "%s NULL spp client device app_id=0x%x",
__func__, app_id);
break;
}
} else {
ASSERT(device, "%s NULL spp server device app_id=0x%x", __func__, app_id);
}
}
#ifdef __INTERCONNECTION__
if (app_id == BTIF_APP_SPP_CLIENT_RED_ID) {
// btm_conn will NULL if only SPP profile since btm_conn restore later
app_bt_set_spp_device_ptr(
(btif_remote_device_t *)btif_me_get_remote_device_by_bdaddr(&remote));
}
#endif
// restore spp device
device->portType = port_type;
device->spp_connected_flag = buf[offset++];
return offset;
}
uint32_t app_bt_save_hfp_app_ctx(btif_remote_device_t *rem_dev, uint8_t *buf,
uint32_t buf_len) {
BTIF_CTX_INIT(buf);
BTIF_CTX_STR_BUF(btif_me_get_remote_device_bdaddr(rem_dev),
BTIF_BD_ADDR_SIZE);
BTIF_CTX_STR_VAL8(app_bt_device.hfchan_call[BT_DEVICE_ID_1]);
BTIF_CTX_STR_VAL8(app_bt_device.hfchan_callSetup[BT_DEVICE_ID_1]);
BTIF_CTX_STR_VAL8(app_bt_device.hf_callheld[BT_DEVICE_ID_1]);
BTIF_CTX_SAVE_UPDATE_DATA_LEN();
return BTIF_CTX_GET_TOTAL_LEN();
}
uint32_t app_bt_restore_hfp_app_ctx(uint8_t *buf, uint32_t buf_len) {
bt_bdaddr_t remote;
uint8_t call, callsetup, callheld;
BTIF_CTX_INIT(buf);
BTIF_CTX_LDR_BUF(&remote, BTIF_BD_ADDR_SIZE);
BTIF_CTX_LDR_VAL8(call);
BTIF_CTX_LDR_VAL8(callsetup);
BTIF_CTX_LDR_VAL8(callheld);
app_bt_device.hfchan_call[BT_DEVICE_ID_1] = call;
app_bt_device.hfchan_callSetup[BT_DEVICE_ID_1] = callsetup;
app_bt_device.hf_callheld[BT_DEVICE_ID_1] = callheld;
TRACE(4, "%s call %d callsetup %d callheld %d", __func__, call, callsetup,
callheld);
return BTIF_CTX_GET_TOTAL_LEN();
}
uint32_t app_bt_save_a2dp_app_ctx(btif_remote_device_t *rem_dev, uint8_t *buf,
uint32_t buf_len) {
uint32_t offset = 0;
unsigned char stream_enc = 0;
uint32_t factor = 0;
// TODO
// more codecs, BT_DEVICE_ID_2
if (bt_profile_manager[BT_DEVICE_ID_1].stream ==
app_bt_device.a2dp_stream[BT_DEVICE_ID_1]->a2dp_stream) {
stream_enc = 0;
}
#if defined(A2DP_AAC_ON)
else if (bt_profile_manager[BT_DEVICE_ID_1].stream ==
app_bt_device.a2dp_aac_stream[BT_DEVICE_ID_1]->a2dp_stream) {
stream_enc = 1;
}
#endif
#if defined(A2DP_LHDC_ON)
else if (bt_profile_manager[BT_DEVICE_ID_1].stream ==
app_bt_device.a2dp_lhdc_stream[BT_DEVICE_ID_1]->a2dp_stream) {
stream_enc = 2;
}
#endif
#if defined(A2DP_LDAC_ON)
else if (bt_profile_manager[BT_DEVICE_ID_1].stream ==
app_bt_device.a2dp_ldac_stream[BT_DEVICE_ID_1]->a2dp_stream) {
stream_enc = 2;
}
#endif
buf[offset++] = stream_enc;
memcpy(buf + offset, btif_me_get_remote_device_bdaddr(rem_dev),
BTIF_BD_ADDR_SIZE);
offset += BTIF_BD_ADDR_SIZE;
buf[offset++] = app_bt_device.a2dp_state[BT_DEVICE_ID_1];
buf[offset++] = app_bt_device.a2dp_play_pause_flag;
buf[offset++] = avrcp_get_media_status();
// codec
buf[offset++] = app_bt_device.codec_type[BT_DEVICE_ID_1];
buf[offset++] = app_bt_device.sample_rate[BT_DEVICE_ID_1];
buf[offset++] = app_bt_device.sample_bit[BT_DEVICE_ID_1];
#if defined(A2DP_LHDC_ON)
buf[offset++] = app_bt_device.a2dp_lhdc_llc[BT_DEVICE_ID_1];
#endif
#if defined(__A2DP_AVDTP_CP__)
buf[offset++] = app_bt_device.avdtp_cp[BT_DEVICE_ID_1];
#endif
// volume
buf[offset++] = (uint8_t)a2dp_volume_get(BT_DEVICE_ID_1);
// latency factor
factor = (uint32_t)a2dp_audio_latency_factor_get();
buf[offset++] = factor & 0xFF;
buf[offset++] = (factor >> 8) & 0xFF;
buf[offset++] = (factor >> 16) & 0xFF;
buf[offset++] = (factor >> 24) & 0xFF;
// a2dp session
buf[offset++] = a2dp_ibrt_session_get() & 0xFF;
buf[offset++] = (a2dp_ibrt_session_get() >> 8) & 0xFF;
buf[offset++] = (a2dp_ibrt_session_get() >> 16) & 0xFF;
buf[offset++] = (a2dp_ibrt_session_get() >> 24) & 0xFF;
return offset;
}
uint32_t app_bt_restore_a2dp_app_ctx(uint8_t *buf, uint32_t buf_len) {
uint32_t offset = 0;
bt_bdaddr_t remote;
unsigned char stream_enc = 0;
stream_enc = buf[offset++];
memcpy(&remote, buf + offset, BTIF_BD_ADDR_SIZE);
offset += BTIF_BD_ADDR_SIZE;
app_bt_device.a2dp_state[BT_DEVICE_ID_1] = buf[offset++];
app_bt_device.a2dp_play_pause_flag = buf[offset++];
avrcp_set_media_status(buf[offset++]);
// codec info
app_bt_device.codec_type[BT_DEVICE_ID_1] = buf[offset++];
app_bt_device.sample_rate[BT_DEVICE_ID_1] = buf[offset++];
app_bt_device.sample_bit[BT_DEVICE_ID_1] = buf[offset++];
#if defined(A2DP_LHDC_ON)
app_bt_device.a2dp_lhdc_llc[BT_DEVICE_ID_1] = buf[offset++];
#endif
#if defined(__A2DP_AVDTP_CP__)
app_bt_device.avdtp_cp[BT_DEVICE_ID_1] = buf[offset++];
#endif
// volume
a2dp_volume_set(BT_DEVICE_ID_1, buf[offset++]);
// latency factor
a2dp_audio_latency_factor_set((float)(buf[offset] + (buf[offset + 1] << 8) +
(buf[offset + 2] << 16) +
(buf[offset + 3] << 24)));
offset += 4;
// a2dp session
a2dp_ibrt_session_set((buf[offset] + (buf[offset + 1] << 8) +
(buf[offset + 2] << 16) + (buf[offset + 3] << 24)));
offset += 4;
// TODO
// more codecs, BT_DEVICE_ID_2
if (stream_enc == 0) {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_stream[BT_DEVICE_ID_1]->a2dp_stream;
}
#if defined(A2DP_AAC_ON)
else if (stream_enc == 1) {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_aac_stream[BT_DEVICE_ID_1]->a2dp_stream;
}
#endif
#if defined(A2DP_LHDC_ON)
else if (stream_enc == 2) {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_lhdc_stream[BT_DEVICE_ID_1]->a2dp_stream;
}
#endif
#if defined(A2DP_LDAC_ON)
else if (stream_enc == 2) {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_ldac_stream[BT_DEVICE_ID_1]->a2dp_stream;
}
#endif
memcpy(bt_profile_manager[BT_DEVICE_ID_1].rmt_addr.address, &remote,
BTIF_BD_ADDR_SIZE);
bt_profile_manager[BT_DEVICE_ID_1].a2dp_connect =
bt_profile_connect_status_success;
bt_profile_manager[BT_DEVICE_ID_1].hfp_connect =
bt_profile_connect_status_success;
bt_profile_manager[BT_DEVICE_ID_1].has_connected = true;
return offset;
}
uint32_t app_bt_save_avrcp_app_ctx(btif_remote_device_t *rem_dev, uint8_t *buf,
uint32_t buf_len) {
uint32_t offset = 0;
buf[offset++] = app_bt_device.avrcp_state[BT_DEVICE_ID_1];
buf[offset++] = app_bt_device.volume_report[BT_DEVICE_ID_1];
if (app_bt_device.avrcp_notify_rsp[BT_DEVICE_ID_1]) {
buf[offset++] = true;
buf[offset++] = app_bt_avrcp_get_notify_trans_id();
} else {
buf[offset++] = false;
buf[offset++] = 0;
}
return offset;
}
uint32_t app_bt_restore_avrcp_app_ctx(uint8_t *buf, uint32_t buf_len) {
uint32_t offset = 0;
uint8_t notify_rsp_exist = 0;
uint8_t trans_id = 0;
app_bt_device.avrcp_state[BT_DEVICE_ID_1] = buf[offset++];
app_bt_device.volume_report[BT_DEVICE_ID_1] = buf[offset++];
notify_rsp_exist = buf[offset++];
trans_id = buf[offset++];
TRACE(4, "app_bt_restore_avrcp_app_ctx state %d report %d notify %d %d\n",
app_bt_device.avrcp_state[BT_DEVICE_ID_1],
app_bt_device.volume_report[BT_DEVICE_ID_1], notify_rsp_exist,
trans_id);
if (notify_rsp_exist &&
app_bt_device.avrcp_notify_rsp[BT_DEVICE_ID_1] == NULL) {
btif_app_a2dp_avrcpadvancedpdu_mempool_calloc(
&app_bt_device.avrcp_notify_rsp[BT_DEVICE_ID_1]);
}
app_bt_avrcp_set_notify_trans_id(trans_id);
return offset;
}
#ifdef __BTMAP_ENABLE__
uint32_t app_bt_save_map_app_ctx(btif_remote_device_t *rem_dev, uint8_t *buf,
uint32_t buf_len) {
// struct bdaddr_t *remote = NULL;
uint32_t offset = 0;
memcpy((void *)buf, (void *)app_bt_device.map_session_handle,
sizeof(app_bt_device.map_session_handle));
offset += sizeof(app_bt_device.map_session_handle);
return offset;
}
uint32_t app_bt_restore_map_app_ctx(uint8_t *buf, uint32_t buf_len) {
uint32_t offset = 0;
memcpy((void *)app_bt_device.map_session_handle, (void *)buf,
sizeof(btif_map_session_handle_t));
offset += sizeof(btif_map_session_handle_t);
return offset;
}
#endif
#if BTIF_HID_DEVICE
uint32_t app_bt_save_hid_app_ctx(uint8_t *buf) {
uint32_t offset = 0;
if (app_bt_device.hid_channel == NULL) {
TRACE(0, "app_bt_save_hid_app_ctx app_bt_device.hid_channel is NULL");
return offset;
}
memcpy((void *)buf, (void *)app_bt_device.hid_channel,
sizeof(app_bt_device.hid_channel));
offset += sizeof(app_bt_device.hid_channel);
return offset;
}
uint32_t app_bt_restore_hid_app_ctx(uint8_t *buf) {
uint32_t offset = 0;
memcpy((void *)app_bt_device.hid_channel, (void *)buf, sizeof(hid_channel_t));
offset += sizeof(hid_channel_t);
return offset;
}
#else
uint32_t app_bt_restore_hid_app_ctx(uint8_t *buf) { return 4; }
uint32_t app_bt_save_hid_app_ctx(uint8_t *buf) {
buf[0] = 0;
buf[1] = 0;
buf[2] = 0;
buf[3] = 0;
return 4;
}
#endif
#endif /* ENHANCED_STACK */
a2dp_stream_t *app_bt_get_mobile_a2dp_stream(uint32_t deviceId) {
return bt_profile_manager[deviceId].stream;
}
void app_bt_update_bt_profile_manager(void) {
ibrt_ctrl_t *p_ibrt_ctrl = app_tws_ibrt_get_bt_ctrl_ctx();
memcpy(bt_profile_manager[BT_DEVICE_ID_1].rmt_addr.address,
p_ibrt_ctrl->mobile_addr.address, BTIF_BD_ADDR_SIZE);
#if defined(A2DP_AAC_ON)
if (p_ibrt_ctrl->a2dp_codec.codec_type == BTIF_AVDTP_CODEC_TYPE_MPEG2_4_AAC) {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_aac_stream[BT_DEVICE_ID_1]->a2dp_stream;
} else
#endif
#if defined(A2DP_SCALABLE_ON)
if (p_ibrt_ctrl->a2dp_codec.codec_type ==
BTIF_AVDTP_CODEC_TYPE_NON_A2DP) {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_scalable_stream[BT_DEVICE_ID_1]->a2dp_stream;
} else
#endif
#if defined(A2DP_LHDC_ON)
if (p_ibrt_ctrl->a2dp_codec.codec_type ==
BTIF_AVDTP_CODEC_TYPE_NON_A2DP) {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_lhdc_stream[BT_DEVICE_ID_1]->a2dp_stream;
} else
#endif
#if defined(A2DP_LDAC_ON)
if (p_ibrt_ctrl->a2dp_codec.codec_type ==
BTIF_AVDTP_CODEC_TYPE_NON_A2DP) {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_ldac_stream[BT_DEVICE_ID_1]->a2dp_stream;
} else
#endif
if (p_ibrt_ctrl->a2dp_codec.codec_type == BTIF_AVDTP_CODEC_TYPE_SBC) {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_stream[BT_DEVICE_ID_1]->a2dp_stream;
} else {
ASSERT(0, "%s err codec_type:%d ", __func__,
p_ibrt_ctrl->a2dp_codec.codec_type);
}
bt_profile_manager[BT_DEVICE_ID_1].a2dp_connect =
bt_profile_connect_status_success;
bt_profile_manager[BT_DEVICE_ID_1].hfp_connect =
bt_profile_connect_status_success;
bt_profile_manager[BT_DEVICE_ID_1].has_connected = true;
TRACE(3, "%s codec_type:%x if_a2dp_stream:%p", __func__,
p_ibrt_ctrl->a2dp_codec.codec_type,
bt_profile_manager[BT_DEVICE_ID_1].stream);
DUMP8("%02x ", bt_profile_manager[BT_DEVICE_ID_1].rmt_addr.address,
BTIF_BD_ADDR_SIZE);
}
void app_bt_update_bt_profile_manager_codec_type(uint8_t codec_type) {
ibrt_ctrl_t *p_ibrt_ctrl = app_tws_ibrt_get_bt_ctrl_ctx();
p_ibrt_ctrl->a2dp_codec.codec_type = codec_type;
#if defined(A2DP_AAC_ON)
if (p_ibrt_ctrl->a2dp_codec.codec_type == BTIF_AVDTP_CODEC_TYPE_MPEG2_4_AAC) {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_aac_stream[BT_DEVICE_ID_1]->a2dp_stream;
} else
#endif
#if defined(A2DP_SCALABLE_ON)
if (p_ibrt_ctrl->a2dp_codec.codec_type ==
BTIF_AVDTP_CODEC_TYPE_NON_A2DP) {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_scalable_stream[BT_DEVICE_ID_1]->a2dp_stream;
} else
#endif
#if defined(A2DP_LHDC_ON)
if (p_ibrt_ctrl->a2dp_codec.codec_type ==
BTIF_AVDTP_CODEC_TYPE_NON_A2DP) {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_lhdc_stream[BT_DEVICE_ID_1]->a2dp_stream;
} else
#endif
#if defined(A2DP_LDAC_ON)
if (p_ibrt_ctrl->a2dp_codec.codec_type ==
BTIF_AVDTP_CODEC_TYPE_NON_A2DP) {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_ldac_stream[BT_DEVICE_ID_1]->a2dp_stream;
} else
#endif
if (p_ibrt_ctrl->a2dp_codec.codec_type == BTIF_AVDTP_CODEC_TYPE_SBC) {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_stream[BT_DEVICE_ID_1]->a2dp_stream;
} else {
ASSERT(0, "%s err codec_type:%d ", __func__,
p_ibrt_ctrl->a2dp_codec.codec_type);
}
bt_profile_manager[BT_DEVICE_ID_1].a2dp_connect =
bt_profile_connect_status_success;
bt_profile_manager[BT_DEVICE_ID_1].hfp_connect =
bt_profile_connect_status_success;
bt_profile_manager[BT_DEVICE_ID_1].has_connected = true;
TRACE(3, "%s codec_type:%x if_a2dp_stream:%p", __func__,
p_ibrt_ctrl->a2dp_codec.codec_type,
bt_profile_manager[BT_DEVICE_ID_1].stream);
}
static bool ibrt_reconnect_mobile_profile_flag = false;
void app_bt_ibrt_reconnect_mobile_profile_flag_set(void) {
ibrt_reconnect_mobile_profile_flag = true;
}
void app_bt_ibrt_reconnect_mobile_profile_flag_clear(void) {
ibrt_reconnect_mobile_profile_flag = false;
}
bool app_bt_ibrt_reconnect_mobile_profile_flag_get(void) {
return ibrt_reconnect_mobile_profile_flag;
}
void app_bt_ibrt_reconnect_mobile_profile(bt_bdaddr_t mobile_addr) {
nvrec_btdevicerecord *mobile_record = NULL;
bt_profile_manager[BT_DEVICE_ID_1].reconnect_mode = bt_profile_reconnect_null;
bt_profile_manager[BT_DEVICE_ID_1].rmt_addr = mobile_addr;
bt_profile_manager[BT_DEVICE_ID_1].chan =
app_bt_device.hf_channel[BT_DEVICE_ID_1];
if (!nv_record_btdevicerecord_find(&mobile_addr, &mobile_record)) {
#if defined(A2DP_AAC_ON)
if (mobile_record->device_plf.a2dp_codectype ==
BTIF_AVDTP_CODEC_TYPE_MPEG2_4_AAC) {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_aac_stream[BT_DEVICE_ID_1]->a2dp_stream;
} else
#endif
#if defined(A2DP_SCALABLE_ON)
if (mobile_record->device_plf.a2dp_codectype ==
BTIF_AVDTP_CODEC_TYPE_NON_A2DP) {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_scalable_stream[BT_DEVICE_ID_1]->a2dp_stream;
} else
#endif
#if defined(A2DP_LHDC_ON)
if (mobile_record->device_plf.a2dp_codectype ==
BTIF_AVDTP_CODEC_TYPE_NON_A2DP) {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_lhdc_stream[BT_DEVICE_ID_1]->a2dp_stream;
} else
#endif
#if defined(A2DP_LDAC_ON)
if (mobile_record->device_plf.a2dp_codectype ==
BTIF_AVDTP_CODEC_TYPE_NON_A2DP) {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_ldac_stream[BT_DEVICE_ID_1]->a2dp_stream;
} else
#endif
{
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_stream[BT_DEVICE_ID_1]->a2dp_stream;
}
} else {
bt_profile_manager[BT_DEVICE_ID_1].stream =
app_bt_device.a2dp_stream[BT_DEVICE_ID_1]
->a2dp_stream; // default using SBC
}
btif_a2dp_reset_stream_state(bt_profile_manager[BT_DEVICE_ID_1].stream);
TRACE(0, "ibrt_ui_log:start reconnect mobile, addr below:");
DUMP8("0x%02x ", &(mobile_addr.address[0]), BTIF_BD_ADDR_SIZE);
app_bt_ibrt_reconnect_mobile_profile_flag_set();
app_bt_precheck_before_starting_connecting(
bt_profile_manager[BT_DEVICE_ID_1].has_connected);
app_ibrt_ui_t *p_ibrt_ui = app_ibrt_ui_get_ctx();
if (p_ibrt_ui->config.profile_concurrency_supported) {
app_bt_A2DP_OpenStream(bt_profile_manager[BT_DEVICE_ID_1].stream,
&(bt_profile_manager[BT_DEVICE_ID_1].rmt_addr));
app_bt_HF_CreateServiceLink(bt_profile_manager[BT_DEVICE_ID_1].chan,
&(bt_profile_manager[BT_DEVICE_ID_1].rmt_addr));
} else {
app_bt_A2DP_OpenStream(bt_profile_manager[BT_DEVICE_ID_1].stream,
&(bt_profile_manager[BT_DEVICE_ID_1].rmt_addr));
// app_bt_HF_CreateServiceLink(bt_profile_manager[BT_DEVICE_ID_1].chan,
// &(bt_profile_manager[BT_DEVICE_ID_1].rmt_addr));
}
// osTimerStart(bt_profile_manager[BT_DEVICE_ID_1].connect_timer,
// APP_IBRT_RECONNECT_TIMEOUT_MS);
}
#endif
#ifdef __IAG_BLE_INCLUDE__
static void app_start_fast_connectable_ble_adv(uint16_t advInterval) {
bool ret = FALSE;
if (NULL == app_fast_ble_adv_timeout_timer) {
app_fast_ble_adv_timeout_timer = osTimerCreate(
osTimer(APP_FAST_BLE_ADV_TIMEOUT_TIMER), osTimerOnce, NULL);
}
osTimerStart(app_fast_ble_adv_timeout_timer, APP_FAST_BLE_ADV_TIMEOUT_IN_MS);
#ifdef IBRT
ret = app_ibrt_ui_get_snoop_via_ble_enable();
#endif
if (FALSE == ret) {
app_ble_start_connectable_adv(advInterval);
}
}
static int app_fast_ble_adv_timeout_timehandler(void const *param) {
bool ret = FALSE;
#ifdef IBRT
ret = app_ibrt_ui_get_snoop_via_ble_enable();
#endif
if (FALSE == ret) {
app_ble_refresh_adv_state(BLE_ADVERTISING_INTERVAL);
}
return 0;
}
void app_stop_fast_connectable_ble_adv_timer(void) {
if (NULL != app_fast_ble_adv_timeout_timer) {
osTimerStop(app_fast_ble_adv_timeout_timer);
}
}
#endif
static uint32_t bt_link_active_mode_bits[MAX_ACTIVE_MODE_MANAGED_LINKS];
void app_bt_active_mode_manager_init(void) {
memset(bt_link_active_mode_bits, 0, sizeof(bt_link_active_mode_bits));
}
void app_bt_active_mode_reset(uint32_t linkIndex) {
bt_link_active_mode_bits[linkIndex] = 0;
}
void app_bt_active_mode_set(BT_LINK_ACTIVE_MODE_KEEPER_USER_E user,
uint32_t linkIndex) {
#if defined(IBRT)
return;
#endif
bool isAlreadyInActiveMode = false;
if (linkIndex < MAX_ACTIVE_MODE_MANAGED_LINKS) {
uint32_t lock = int_lock_global();
if (bt_link_active_mode_bits[linkIndex] > 0) {
isAlreadyInActiveMode = true;
} else {
isAlreadyInActiveMode = false;
}
bt_link_active_mode_bits[linkIndex] |= (1 << user);
int_unlock_global(lock);
if (!isAlreadyInActiveMode) {
app_bt_stop_sniff(linkIndex);
app_bt_stay_active(linkIndex);
}
} else if (MAX_ACTIVE_MODE_MANAGED_LINKS == linkIndex) {
for (uint8_t devId = 0; devId < BT_DEVICE_NUM; devId++) {
uint32_t lock = int_lock_global();
if (bt_link_active_mode_bits[devId] > 0) {
isAlreadyInActiveMode = true;
} else {
isAlreadyInActiveMode = false;
}
bt_link_active_mode_bits[devId] |= (1 << user);
int_unlock_global(lock);
if (!isAlreadyInActiveMode) {
app_bt_stop_sniff(devId);
app_bt_stay_active(devId);
}
}
}
TRACE(2, "set active mode for user %d, link %d, now state:", user, linkIndex);
DUMP32("%08x", bt_link_active_mode_bits, MAX_ACTIVE_MODE_MANAGED_LINKS);
}
void app_bt_active_mode_clear(BT_LINK_ACTIVE_MODE_KEEPER_USER_E user,
uint32_t linkIndex) {
#if defined(IBRT)
return;
#endif
bool isAlreadyAllowSniff = false;
if (linkIndex < MAX_ACTIVE_MODE_MANAGED_LINKS) {
uint32_t lock = int_lock_global();
if (0 == bt_link_active_mode_bits[linkIndex]) {
isAlreadyAllowSniff = true;
} else {
isAlreadyAllowSniff = false;
}
bt_link_active_mode_bits[linkIndex] &= (~(1 << user));
int_unlock_global(lock);
if (!isAlreadyAllowSniff) {
app_bt_allow_sniff(linkIndex);
}
} else if (MAX_ACTIVE_MODE_MANAGED_LINKS == linkIndex) {
for (uint8_t devId = 0; devId < BT_DEVICE_NUM; devId++) {
uint32_t lock = int_lock_global();
if (0 == bt_link_active_mode_bits[devId]) {
isAlreadyAllowSniff = true;
} else {
isAlreadyAllowSniff = false;
}
bt_link_active_mode_bits[devId] &= (~(1 << user));
int_unlock_global(lock);
if (!isAlreadyAllowSniff) {
app_bt_allow_sniff(devId);
}
}
}
TRACE(2, "clear active mode for user %d, link %d, now state:", user,
linkIndex);
DUMP32("%08x ", bt_link_active_mode_bits, MAX_ACTIVE_MODE_MANAGED_LINKS);
}
int8_t app_bt_get_rssi(void) {
int8_t rssi = 127;
uint8_t i;
btif_remote_device_t *remDev = NULL;
rx_agc_t tws_agc = {0};
for (i = 0; i < BT_DEVICE_NUM; i++) {
remDev = btif_me_enumerate_remote_devices(i);
if (remDev) {
if (btif_me_get_remote_device_hci_handle(remDev)) {
rssi = bt_drv_reg_op_read_rssi_in_dbm(
btif_me_get_remote_device_hci_handle(remDev), &tws_agc);
rssi = bt_drv_reg_op_rssi_correction(rssi);
TRACE(1, " headset to mobile RSSI:%d dBm", rssi);
}
}
}
return rssi;
}
#ifdef TILE_DATAPATH
int8_t app_tile_get_ble_rssi(void) {
int8_t rssi = 127;
uint8_t i;
btif_remote_device_t *remDev = NULL;
rx_agc_t tws_agc = {0};
for (i = 0; i < BT_DEVICE_NUM; i++) {
remDev = btif_me_enumerate_remote_devices(i);
if (remDev) {
if (app_tile_ble_get_connection_index() != BLE_INVALID_CONNECTION_INDEX) {
rssi = bt_drv_reg_op_read_ble_rssi_in_dbm(
app_tile_ble_get_connection_index(), &tws_agc);
rssi = bt_drv_reg_op_rssi_correction(rssi);
TRACE(1, " headset to mobile RSSI:%d dBm", rssi);
}
}
}
return rssi;
}
#endif
#ifdef __GMA_VOICE__
void app_bt_prepare_for_ota(void) {
app_ibrt_ui_t *p_ui_ctrl = app_ibrt_ui_get_ctx();
p_ui_ctrl->config.disable_tws_switch = true; // disable role switch
app_key_close();
if (IBRT_MASTER == app_tws_ibrt_role_get_callback(NULL)) {
btif_hf_disconnect_service_link(app_bt_device.hf_channel[BT_DEVICE_ID_1]);
btif_a2dp_suspend_stream(
app_bt_device.a2dp_stream[BT_DEVICE_ID_1]->a2dp_stream);
}
TRACE(0, "app_bt_prepare_for_ota");
}
#endif