5278 lines
192 KiB
C++
5278 lines
192 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 "hal_aud.h"
|
|
#include "hal_chipid.h"
|
|
#include "hal_trace.h"
|
|
#include "apps.h"
|
|
#include "app_thread.h"
|
|
#include "app_status_ind.h"
|
|
#include "bluetooth.h"
|
|
#include "nvrecord.h"
|
|
#include "besbt.h"
|
|
#include "besbt_cfg.h"
|
|
#include "me_api.h"
|
|
#include "mei_api.h"
|
|
#include "a2dp_api.h"
|
|
#include "hci_api.h"
|
|
#include "l2cap_api.h"
|
|
#include "hfp_api.h"
|
|
#include "dip_api.h"
|
|
#include "btapp.h"
|
|
#include "app_bt.h"
|
|
#include "app_hfp.h"
|
|
#include "app_bt_func.h"
|
|
#include "bt_drv_interface.h"
|
|
#include "os_api.h"
|
|
#include "bt_drv_reg_op.h"
|
|
#include "app_a2dp.h"
|
|
#include "app_dip.h"
|
|
#include "app_ai_manager_api.h"
|
|
#include "app_bt.h"
|
|
|
|
#if defined(IBRT)
|
|
#include "app_tws_ibrt.h"
|
|
#include "app_ibrt_ui.h"
|
|
#include "app_ibrt_if.h"
|
|
#include "app_ibrt_a2dp.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_spp.h"
|
|
#include "ai_thread.h"
|
|
#include "ai_manager.h"
|
|
#endif
|
|
|
|
#ifdef __INTERCONNECTION__
|
|
#include "app_interconnection.h"
|
|
#include "app_interconnection_ble.h"
|
|
#include "spp_api.h"
|
|
#include "app_interconnection_ccmp.h"
|
|
#include "app_spp.h"
|
|
#include "app_interconnection_spp.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 "tile_target_ble.h"
|
|
#include "rwip_config.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();
|
|
}
|
|
|
|
|
|
#ifdef FPGA
|
|
void app_bt_accessmode_set_for_test(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
|
|
};
|
|
|
|
TRACE(2,"%s %d",__func__, mode);
|
|
app_bt_ME_SetAccessibleMode_Fortest(mode, &info);
|
|
}
|
|
|
|
void app_bt_adv_mode_set_for_test(uint8_t en)
|
|
{
|
|
|
|
TRACE(2,"%s %d",__func__, en);
|
|
app_bt_ME_Set_Advmode_Fortest(en);
|
|
}
|
|
|
|
void app_start_ble_adv_for_test(void)
|
|
{
|
|
TRACE(1,"%s",__func__);
|
|
|
|
U8 adv_data[31];
|
|
U8 adv_data_len = 0;
|
|
U8 scan_rsp_data[31];
|
|
U8 scan_rsp_data_len = 0;
|
|
|
|
adv_data[adv_data_len++] = 2;
|
|
adv_data[adv_data_len++] = 0x01;
|
|
adv_data[adv_data_len++] = 0x1A;
|
|
|
|
adv_data[adv_data_len++] = 17;
|
|
adv_data[adv_data_len++] = 0xFF;
|
|
|
|
adv_data[adv_data_len++] = 0x9A;
|
|
adv_data[adv_data_len++] = 0x07;
|
|
|
|
adv_data[adv_data_len++] = 0x10;
|
|
adv_data[adv_data_len++] = 0x04;
|
|
adv_data[adv_data_len++] = 0x06;
|
|
|
|
adv_data[adv_data_len++] = 0x07;
|
|
adv_data[adv_data_len++] = 0x00;
|
|
|
|
adv_data[adv_data_len++] = 0x01;
|
|
adv_data[adv_data_len++] = 0x98;
|
|
|
|
adv_data[adv_data_len++] = 1;
|
|
|
|
adv_data[adv_data_len++] = 0xFF;
|
|
adv_data[adv_data_len++] = 0xFF;
|
|
adv_data[adv_data_len++] = 0xFF;
|
|
adv_data[adv_data_len++] = 0xFF;
|
|
adv_data[adv_data_len++] = 0xFF;
|
|
adv_data[adv_data_len++] = 0xFF;
|
|
|
|
// Get default Device Name (No name if not enough space)
|
|
const char* ble_name_in_nv = BLE_DEFAULT_NAME;
|
|
uint32_t nameLen = strlen(ble_name_in_nv);
|
|
// Get remaining space in the Advertising Data - 2 bytes are used for name length/flag
|
|
int8_t avail_space = 31-adv_data_len;
|
|
if(avail_space - 2 >= (int8_t)nameLen)
|
|
{
|
|
// Check if data can be added to the adv Data
|
|
adv_data[adv_data_len++] = nameLen + 1;
|
|
// Fill Device Name Flag
|
|
adv_data[adv_data_len++] = '\x09';
|
|
// Copy device name
|
|
memcpy(&adv_data[adv_data_len], ble_name_in_nv, nameLen);
|
|
// Update adv Data Length
|
|
adv_data_len += nameLen;
|
|
btif_me_ble_set_adv_data(adv_data_len, adv_data);
|
|
|
|
btif_adv_para_struct_t adv_para;
|
|
adv_para.interval_min = 32;
|
|
adv_para.interval_max = 32;
|
|
adv_para.adv_type = 0;
|
|
adv_para.own_addr_type = 0;
|
|
adv_para.peer_addr_type = 0;
|
|
adv_para.adv_chanmap = 0x07;
|
|
adv_para.adv_filter_policy = 0;
|
|
btif_me_ble_set_adv_parameters(&adv_para);
|
|
btif_me_set_ble_bd_address(ble_addr);
|
|
btif_me_ble_set_adv_en(1);
|
|
}
|
|
else
|
|
{
|
|
nameLen = avail_space - 2;
|
|
// Check if data can be added to the adv Data
|
|
adv_data[adv_data_len++] = nameLen + 1;
|
|
// Fill Device Name Flag
|
|
adv_data[adv_data_len++] = '\x08';
|
|
// Copy device name
|
|
memcpy(&adv_data[adv_data_len], ble_name_in_nv, nameLen);
|
|
// Update adv Data Length
|
|
adv_data_len += nameLen;
|
|
btif_me_ble_set_adv_data(adv_data_len, adv_data);
|
|
|
|
btif_adv_para_struct_t adv_para;
|
|
adv_para.interval_min = 256;
|
|
adv_para.interval_max = 256;
|
|
adv_para.adv_type = 0;
|
|
adv_para.own_addr_type = 0;
|
|
adv_para.peer_addr_type = 0;
|
|
adv_para.adv_chanmap = 0x07;
|
|
adv_para.adv_filter_policy = 0;
|
|
btif_me_ble_set_adv_parameters(&adv_para);
|
|
btif_me_set_ble_bd_address(ble_addr);
|
|
|
|
avail_space = 31;
|
|
nameLen = strlen(ble_name_in_nv);
|
|
if(avail_space - 2 < (int8_t)nameLen)
|
|
nameLen = avail_space - 2;
|
|
|
|
scan_rsp_data[scan_rsp_data_len++] = nameLen + 1;
|
|
// Fill Device Name Flag
|
|
scan_rsp_data[scan_rsp_data_len++] = '\x09';
|
|
// Copy device name
|
|
memcpy(&scan_rsp_data[scan_rsp_data_len], ble_name_in_nv, nameLen);
|
|
// Update Scan response Data Length
|
|
scan_rsp_data_len += nameLen;
|
|
btif_me_ble_set_scan_rsp_data(scan_rsp_data_len, scan_rsp_data);
|
|
btif_me_ble_set_adv_en(1);
|
|
}
|
|
}
|
|
|
|
void app_bt_write_controller_memory_for_test(uint32_t addr,uint32_t val,uint8_t type)
|
|
{
|
|
TRACE(2,"%s addr=0x%x val=0x%x type=%d",__func__, addr,val,type);
|
|
app_bt_ME_Write_Controller_Memory_Fortest(addr,val,type);
|
|
}
|
|
|
|
void app_bt_read_controller_memory_for_test(uint32_t addr,uint32_t len,uint8_t type)
|
|
{
|
|
TRACE(2,"%s addr=0x%x len=%x type=%d",__func__, addr,len,type);
|
|
app_bt_ME_Read_Controller_Memory_Fortest(addr,len,type);
|
|
}
|
|
#endif
|
|
|
|
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__) && !defined(FPGA)
|
|
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__) && !defined(FPGA)
|
|
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__) && !defined(FPGA)
|
|
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)
|
|
{
|
|
#ifndef FPGA
|
|
//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);
|
|
#endif
|
|
}
|
|
else
|
|
{
|
|
#ifndef FPGA
|
|
//app_status_indication_set(APP_STATUS_INDICATION_PAGESCAN);
|
|
#endif
|
|
}
|
|
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;
|
|
|
|
#ifndef FPGA
|
|
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
|
|
#endif
|
|
{
|
|
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);
|
|
#ifndef FPGA
|
|
nv_record_touch_cause_flush();
|
|
#endif
|
|
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);
|
|
#ifndef FPGA
|
|
nv_record_touch_cause_flush();
|
|
#endif
|
|
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);
|
|
#ifndef FPGA
|
|
nv_record_touch_cause_flush();
|
|
#endif
|
|
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(¶);
|
|
|
|
}
|
|
|
|
|
|
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;
|
|
}
|
|
#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
|