| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837 |
- /******************************************************************************
- *
- * Copyright (C) 1999-2012 Broadcom Corporation
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at:
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- *
- ******************************************************************************/
- /******************************************************************************
- *
- * This file contains functions for BLE whitelist operation.
- *
- ******************************************************************************/
- #include <string.h>
- #include "common/bt_trace.h"
- #include "device/controller.h"
- #include "osi/allocator.h"
- #include "osi/hash_map.h"
- #include "stack/bt_types.h"
- #include "stack/btu.h"
- #include "btm_int.h"
- #include "l2c_int.h"
- #include "stack/hcimsgs.h"
- //#include "bt_utils.h"
- #ifndef BTM_BLE_SCAN_PARAM_TOUT
- #define BTM_BLE_SCAN_PARAM_TOUT 50 /* 50 seconds */
- #endif
- #if (BLE_INCLUDED == TRUE)
- static void btm_suspend_wl_activity(tBTM_BLE_WL_STATE wl_state);
- static void btm_wl_update_to_controller(void);
- // Unfortunately (for now?) we have to maintain a copy of the device whitelist
- // on the host to determine if a device is pending to be connected or not. This
- // controls whether the host should keep trying to scan for whitelisted
- // peripherals or not.
- // TODO: Move all of this to controller/le/background_list or similar?
- static const size_t background_connection_buckets = 42;
- static hash_map_t *background_connections = NULL;
- typedef struct background_connection_t {
- bt_bdaddr_t address;
- } background_connection_t;
- static bool bdaddr_equality_fn(const void *x, const void *y)
- {
- return bdaddr_equals((bt_bdaddr_t *)x, (bt_bdaddr_t *)y);
- }
- static void background_connections_lazy_init(void)
- {
- if (!background_connections) {
- background_connections = hash_map_new(background_connection_buckets,
- hash_function_bdaddr, NULL, osi_free_func, bdaddr_equality_fn);
- assert(background_connections);
- }
- }
- static BOOLEAN background_connection_add(bt_bdaddr_t *address)
- {
- assert(address);
- background_connections_lazy_init();
- background_connection_t *connection = hash_map_get(background_connections, address);
- if (!connection) {
- connection = osi_calloc(sizeof(background_connection_t));
- connection->address = *address;
- hash_map_set(background_connections, &(connection->address), connection);
- return TRUE;
- }
- return FALSE;
- }
- static BOOLEAN background_connection_remove(bt_bdaddr_t *address)
- {
- if (address && background_connections) {
- return hash_map_erase(background_connections, address);
- }
- return FALSE;
- }
- static void background_connections_clear(void)
- {
- if (background_connections) {
- hash_map_clear(background_connections);
- }
- }
- static bool background_connections_pending_cb(hash_map_entry_t *hash_entry, void *context)
- {
- bool *pending_connections = context;
- background_connection_t *connection = hash_entry->data;
- const bool connected = BTM_IsAclConnectionUp(connection->address.address, BT_TRANSPORT_LE);
- if (!connected) {
- *pending_connections = true;
- return false;
- }
- return true;
- }
- static bool background_connections_pending(void)
- {
- bool pending_connections = false;
- if (background_connections) {
- hash_map_foreach(background_connections, background_connections_pending_cb, &pending_connections);
- }
- return pending_connections;
- }
- /*******************************************************************************
- **
- ** Function btm_update_scanner_filter_policy
- **
- ** Description This function updates the filter policy of scanner
- *******************************************************************************/
- void btm_update_scanner_filter_policy(tBTM_BLE_SFP scan_policy)
- {
- tBTM_BLE_INQ_CB *p_inq = &btm_cb.ble_ctr_cb.inq_var;
- UINT32 scan_interval = !p_inq->scan_interval ? BTM_BLE_GAP_DISC_SCAN_INT : p_inq->scan_interval;
- UINT32 scan_window = !p_inq->scan_window ? BTM_BLE_GAP_DISC_SCAN_WIN : p_inq->scan_window;
- BTM_TRACE_EVENT ("%s\n", __func__);
- p_inq->sfp = scan_policy;
- p_inq->scan_type = p_inq->scan_type == BTM_BLE_SCAN_MODE_NONE ? BTM_BLE_SCAN_MODE_ACTI : p_inq->scan_type;
- if (btm_cb.cmn_ble_vsc_cb.extended_scan_support == 0) {
- btsnd_hcic_ble_set_scan_params(p_inq->scan_type, (UINT16)scan_interval,
- (UINT16)scan_window,
- btm_cb.ble_ctr_cb.addr_mgnt_cb.own_addr_type,
- scan_policy);
- } else {
- btm_ble_send_extended_scan_params(p_inq->scan_type, scan_interval, scan_window,
- btm_cb.ble_ctr_cb.addr_mgnt_cb.own_addr_type,
- scan_policy);
- }
- }
- /*******************************************************************************
- **
- ** Function btm_add_dev_to_controller
- **
- ** Description This function load the device into controller white list
- *******************************************************************************/
- BOOLEAN btm_add_dev_to_controller (BOOLEAN to_add, BD_ADDR bd_addr, tBLE_ADDR_TYPE wl_addr_type)
- {
- /*
- tBTM_SEC_DEV_REC *p_dev_rec = btm_find_dev (bd_addr);
- tBLE_ADDR_TYPE addr_type = BLE_ADDR_PUBLIC;
- BOOLEAN started = FALSE;
- BD_ADDR dummy_bda = {0};
- tBT_DEVICE_TYPE dev_type;
- if (p_dev_rec != NULL &&
- p_dev_rec->device_type & BT_DEVICE_TYPE_BLE) {
- if (to_add) {
- if (p_dev_rec->ble.ble_addr_type == BLE_ADDR_PUBLIC || !BTM_BLE_IS_RESOLVE_BDA(bd_addr)) {
- started = btsnd_hcic_ble_add_white_list (p_dev_rec->ble.ble_addr_type, bd_addr);
- p_dev_rec->ble.in_controller_list |= BTM_WHITE_LIST_BIT;
- } else if (memcmp(p_dev_rec->ble.static_addr, bd_addr, BD_ADDR_LEN) != 0 &&
- memcmp(p_dev_rec->ble.static_addr, dummy_bda, BD_ADDR_LEN) != 0) {
- started = btsnd_hcic_ble_add_white_list (p_dev_rec->ble.static_addr_type,
- p_dev_rec->ble.static_addr);
- p_dev_rec->ble.in_controller_list |= BTM_WHITE_LIST_BIT;
- }
- } else {
- if (p_dev_rec->ble.ble_addr_type == BLE_ADDR_PUBLIC || !BTM_BLE_IS_RESOLVE_BDA(bd_addr)) {
- started = btsnd_hcic_ble_remove_from_white_list (p_dev_rec->ble.ble_addr_type, bd_addr);
- }
- if (memcmp(p_dev_rec->ble.static_addr, dummy_bda, BD_ADDR_LEN) != 0 &&
- memcmp(p_dev_rec->ble.static_addr, bd_addr, BD_ADDR_LEN) != 0) {
- started = btsnd_hcic_ble_remove_from_white_list (p_dev_rec->ble.static_addr_type, p_dev_rec->ble.static_addr);
- }
- p_dev_rec->ble.in_controller_list &= ~BTM_WHITE_LIST_BIT;
- }
- } // if not a known device, shall we add it?
- else {
- BTM_ReadDevInfo(bd_addr, &dev_type, &addr_type);
- if (to_add) {
- started = btsnd_hcic_ble_add_white_list (addr_type, bd_addr);
- }else{
- started = btsnd_hcic_ble_remove_from_white_list (addr_type, bd_addr);
- }
- }
- return started;
- */
- /* Controller do not support resolvable address now, only support public address and static random address */
- BOOLEAN started = FALSE;
- if(wl_addr_type > BLE_ADDR_RANDOM) {
- BTM_TRACE_ERROR("wl_addr_type is error\n");
- return started;
- }
- if (to_add) {
- started = btsnd_hcic_ble_add_white_list (wl_addr_type, bd_addr);
- }else{
- started = btsnd_hcic_ble_remove_from_white_list (wl_addr_type, bd_addr);
- }
- return started;
- }
- /*******************************************************************************
- **
- ** Function btm_execute_wl_dev_operation
- **
- ** Description execute the pending whitelist device operation(loading or removing)
- *******************************************************************************/
- BOOLEAN btm_execute_wl_dev_operation(void)
- {
- tBTM_BLE_WL_OP *p_dev_op = btm_cb.ble_ctr_cb.wl_op_q;
- UINT8 i = 0;
- BOOLEAN rt = TRUE;
- for (i = 0; i < BTM_BLE_MAX_BG_CONN_DEV_NUM && rt; i ++, p_dev_op ++) {
- if (p_dev_op->in_use) {
- rt = btm_add_dev_to_controller(p_dev_op->to_add, p_dev_op->bd_addr, p_dev_op->addr_type);
- memset(p_dev_op, 0, sizeof(tBTM_BLE_WL_OP));
- } else {
- break;
- }
- }
- return rt;
- }
- /*******************************************************************************
- **
- ** Function btm_enq_wl_dev_operation
- **
- ** Description enqueue the pending whitelist device operation(loading or removing).
- *******************************************************************************/
- void btm_enq_wl_dev_operation(BOOLEAN to_add, BD_ADDR bd_addr, tBLE_ADDR_TYPE addr_type)
- {
- tBTM_BLE_WL_OP *p_dev_op = btm_cb.ble_ctr_cb.wl_op_q;
- UINT8 i = 0;
- for (i = 0; i < BTM_BLE_MAX_BG_CONN_DEV_NUM; i ++, p_dev_op ++) {
- if (p_dev_op->in_use && p_dev_op->addr_type == addr_type && !memcmp(p_dev_op->bd_addr, bd_addr, BD_ADDR_LEN)) {
- p_dev_op->to_add = to_add;
- return;
- } else if (!p_dev_op->in_use) {
- break;
- }
- }
- if (i != BTM_BLE_MAX_BG_CONN_DEV_NUM) {
- p_dev_op->in_use = TRUE;
- p_dev_op->to_add = to_add;
- p_dev_op->addr_type = addr_type;
- memcpy(p_dev_op->bd_addr, bd_addr, BD_ADDR_LEN);
- } else {
- BTM_TRACE_ERROR("max pending WL operation reached, discard");
- }
- return;
- }
- /*******************************************************************************
- **
- ** Function btm_update_dev_to_white_list
- **
- ** Description This function adds or removes a device into/from
- ** the white list.
- **
- *******************************************************************************/
- BOOLEAN btm_update_dev_to_white_list(BOOLEAN to_add, BD_ADDR bd_addr, tBLE_ADDR_TYPE addr_type, tBTM_UPDATE_WHITELIST_CBACK *update_wl_cb)
- {
- if(addr_type > BLE_ADDR_RANDOM) {
- BTM_TRACE_ERROR("%s address type is error, unable to add device", __func__);
- if (update_wl_cb){
- update_wl_cb(HCI_ERR_ILLEGAL_PARAMETER_FMT,to_add);
- }
- return FALSE;
- }
- BD_ADDR invalid_rand_addr_a, invalid_rand_addr_b;
- memset(invalid_rand_addr_a, 0xff, sizeof(BD_ADDR));
- memset(invalid_rand_addr_b, 0x00, sizeof(BD_ADDR));
- // look for public address information
- tBTM_SEC_DEV_REC *p_dev_rec = btm_find_dev(bd_addr);
- // p_dev_rec is created at bluetooth initialization, p_dev_rec->ble.static_addr maybe be all 0 before pairing
- if(p_dev_rec && memcmp(invalid_rand_addr_b, p_dev_rec->ble.static_addr, BD_ADDR_LEN) != 0) {
- memcpy(bd_addr, p_dev_rec->ble.static_addr, BD_ADDR_LEN);
- addr_type = p_dev_rec->ble.static_addr_type;
- }
- // white list must be public address or static random address
- if(addr_type == BLE_ADDR_RANDOM) {
- /*
- A static address is a 48-bit randomly generated address and shall meet the following requirements:
- • The two most significant bits of the address shall be equal to 1
- • All bits of the random part of the address shall not be equal to 1
- • All bits of the random part of the address shall not be equal to 0
- */
- invalid_rand_addr_b[0] = invalid_rand_addr_b[0] | BT_STATIC_RAND_ADDR_MASK;
- if((bd_addr[0] & BT_STATIC_RAND_ADDR_MASK) == BT_STATIC_RAND_ADDR_MASK
- && memcmp(invalid_rand_addr_a, bd_addr, BD_ADDR_LEN) != 0
- && memcmp(invalid_rand_addr_b, bd_addr, BD_ADDR_LEN) != 0){
- // do nothing
- } else {
- BTC_TRACE_ERROR(" controller not support resolvable address");
- if (update_wl_cb){
- update_wl_cb(HCI_ERR_ILLEGAL_PARAMETER_FMT,to_add);
- }
- return FALSE;
- }
- }
- tBTM_BLE_CB *p_cb = &btm_cb.ble_ctr_cb;
- if (to_add && p_cb->white_list_avail_size == 0) {
- BTM_TRACE_ERROR("%s Whitelist full, unable to add device", __func__);
- if (update_wl_cb){
- update_wl_cb(HCI_ERR_MEMORY_FULL,to_add);
- }
- return FALSE;
- }
- if (to_add) {
- /* added the bd_addr to the connection hash map queue */
- if(!background_connection_add((bt_bdaddr_t *)bd_addr)) {
- /* if the bd_addr already exist in whitelist, just callback return TRUE */
- if (update_wl_cb){
- update_wl_cb(HCI_SUCCESS,to_add);
- }
- return TRUE;
- }
- } else {
- /* remove the bd_addr to the connection hash map queue */
- if(!background_connection_remove((bt_bdaddr_t *)bd_addr)){
- /* if the bd_addr don't exist in whitelist, just callback return TRUE */
- if (update_wl_cb){
- update_wl_cb(HCI_SUCCESS,to_add);
- }
- return TRUE;
- }
- }
- if (update_wl_cb){
- //save add whitelist complete callback
- p_cb->update_wl_cb = update_wl_cb;
- }
- /* stop the auto connect */
- btm_suspend_wl_activity(p_cb->wl_state);
- /* save the bd_addr to the btm_cb env */
- btm_enq_wl_dev_operation(to_add, bd_addr, addr_type);
- /* save the ba_addr to the controller white list */
- btm_wl_update_to_controller();
- return TRUE;
- }
- /*******************************************************************************
- **
- ** Function btm_ble_clear_white_list
- **
- ** Description This function clears the white list.
- **
- *******************************************************************************/
- void btm_ble_clear_white_list (tBTM_UPDATE_WHITELIST_CBACK *update_wl_cb)
- {
- tBTM_BLE_CB *p_cb = &btm_cb.ble_ctr_cb;
- BTM_TRACE_EVENT ("btm_ble_clear_white_list");
- btsnd_hcic_ble_clear_white_list();
- background_connections_clear();
- if (update_wl_cb) {
- p_cb->update_wl_cb = update_wl_cb;
- }
- }
- /*******************************************************************************
- **
- ** Function btm_ble_clear_white_list_complete
- **
- ** Description Indicates white list cleared.
- **
- *******************************************************************************/
- void btm_ble_clear_white_list_complete(UINT8 *p_data, UINT16 evt_len)
- {
- tBTM_BLE_CB *p_cb = &btm_cb.ble_ctr_cb;
- UINT8 status;
- UNUSED(evt_len);
- BTM_TRACE_EVENT ("btm_ble_clear_white_list_complete");
- STREAM_TO_UINT8 (status, p_data);
- if (status == HCI_SUCCESS) {
- p_cb->white_list_avail_size = controller_get_interface()->get_ble_white_list_size();
- } else {
- BTM_TRACE_ERROR ("%s failed, status 0x%x\n", __func__, status);
- }
- if (p_cb->update_wl_cb) {
- (*p_cb->update_wl_cb)(status, BTM_WHITELIST_CLEAR);
- }
- }
- /*******************************************************************************
- **
- ** Function btm_ble_white_list_init
- **
- ** Description Initialize white list size
- **
- *******************************************************************************/
- void btm_ble_white_list_init(UINT8 white_list_size)
- {
- BTM_TRACE_DEBUG("%s white_list_size = %d", __func__, white_list_size);
- btm_cb.ble_ctr_cb.white_list_avail_size = white_list_size;
- }
- /*******************************************************************************
- **
- ** Function btm_ble_add_2_white_list_complete
- **
- ** Description White list element added
- **
- *******************************************************************************/
- void btm_ble_add_2_white_list_complete(UINT8 status)
- {
- BTM_TRACE_EVENT("%s status=%d", __func__, status);
- tBTM_BLE_CB *p_cb = &btm_cb.ble_ctr_cb;
- if (status == HCI_SUCCESS) {
- --btm_cb.ble_ctr_cb.white_list_avail_size;
- }
- // add whitelist complete callback
- if (p_cb->update_wl_cb)
- {
- (*p_cb->update_wl_cb)(status, BTM_WHITELIST_ADD);
- }
- }
- /*******************************************************************************
- **
- ** Function btm_ble_remove_from_white_list_complete
- **
- ** Description White list element removal complete
- **
- *******************************************************************************/
- void btm_ble_remove_from_white_list_complete(UINT8 *p, UINT16 evt_len)
- {
- tBTM_BLE_CB *p_cb = &btm_cb.ble_ctr_cb;
- UNUSED(evt_len);
- BTM_TRACE_EVENT ("%s status=%d", __func__, *p);
- if (*p == HCI_SUCCESS) {
- ++btm_cb.ble_ctr_cb.white_list_avail_size;
- }
- if (p_cb->update_wl_cb)
- {
- (*p_cb->update_wl_cb)(*p, BTM_WHITELIST_REMOVE);
- }
- }
- /*******************************************************************************
- **
- ** Function btm_ble_start_auto_conn
- **
- ** Description This function is to start/stop auto connection procedure.
- **
- ** Parameters start: TRUE to start; FALSE to stop.
- **
- ** Returns void
- **
- *******************************************************************************/
- BOOLEAN btm_ble_start_auto_conn(BOOLEAN start)
- {
- tBTM_BLE_CB *p_cb = &btm_cb.ble_ctr_cb;
- BD_ADDR dummy_bda = {0};
- BOOLEAN exec = TRUE;
- UINT16 scan_int;
- UINT16 scan_win;
- UINT8 own_addr_type = p_cb->addr_mgnt_cb.own_addr_type;
- UINT8 peer_addr_type = BLE_ADDR_PUBLIC;
- if (start) {
- if (p_cb->conn_state == BLE_CONN_IDLE && background_connections_pending()
- && btm_ble_topology_check(BTM_BLE_STATE_INIT)) {
- p_cb->wl_state |= BTM_BLE_WL_INIT;
- btm_execute_wl_dev_operation();
- #if BLE_PRIVACY_SPT == TRUE
- btm_ble_enable_resolving_list_for_platform(BTM_BLE_RL_INIT);
- #endif
- scan_int = (p_cb->scan_int == BTM_BLE_SCAN_PARAM_UNDEF) ?
- BTM_BLE_SCAN_SLOW_INT_1 : p_cb->scan_int;
- scan_win = (p_cb->scan_win == BTM_BLE_SCAN_PARAM_UNDEF) ?
- BTM_BLE_SCAN_SLOW_WIN_1 : p_cb->scan_win;
- #if BLE_PRIVACY_SPT == TRUE
- if (btm_cb.ble_ctr_cb.rl_state != BTM_BLE_RL_IDLE
- && controller_get_interface()->supports_ble_privacy()) {
- own_addr_type |= BLE_ADDR_TYPE_ID_BIT;
- peer_addr_type |= BLE_ADDR_TYPE_ID_BIT;
- }
- #endif
- if (!btsnd_hcic_ble_create_ll_conn (scan_int, /* UINT16 scan_int */
- scan_win, /* UINT16 scan_win */
- 0x01, /* UINT8 white_list */
- peer_addr_type, /* UINT8 addr_type_peer */
- dummy_bda, /* BD_ADDR bda_peer */
- own_addr_type, /* UINT8 addr_type_own */
- BTM_BLE_CONN_INT_MIN_DEF, /* UINT16 conn_int_min */
- BTM_BLE_CONN_INT_MAX_DEF, /* UINT16 conn_int_max */
- BTM_BLE_CONN_SLAVE_LATENCY_DEF, /* UINT16 conn_latency */
- BTM_BLE_CONN_TIMEOUT_DEF, /* UINT16 conn_timeout */
- 0, /* UINT16 min_len */
- 0)) { /* UINT16 max_len */
- /* start auto connection failed */
- exec = FALSE;
- p_cb->wl_state &= ~BTM_BLE_WL_INIT;
- } else {
- btm_ble_set_conn_st (BLE_BG_CONN);
- }
- } else {
- exec = FALSE;
- }
- } else {
- if (p_cb->conn_state == BLE_BG_CONN) {
- btsnd_hcic_ble_create_conn_cancel();
- btm_ble_set_conn_st (BLE_CONN_CANCEL);
- p_cb->wl_state &= ~BTM_BLE_WL_INIT;
- } else {
- BTM_TRACE_DEBUG("conn_st = %d, not in auto conn state, cannot stop", p_cb->conn_state);
- exec = FALSE;
- }
- }
- return exec;
- }
- /*******************************************************************************
- **
- ** Function btm_ble_start_select_conn
- **
- ** Description This function is to start/stop selective connection procedure.
- **
- ** Parameters start: TRUE to start; FALSE to stop.
- ** p_select_cback: callback function to return application
- ** selection.
- **
- ** Returns BOOLEAN: selective connection procedure is started.
- **
- *******************************************************************************/
- BOOLEAN btm_ble_start_select_conn(BOOLEAN start, tBTM_BLE_SEL_CBACK *p_select_cback)
- {
- tBTM_BLE_CB *p_cb = &btm_cb.ble_ctr_cb;
- UINT32 scan_int = p_cb->scan_int == BTM_BLE_SCAN_PARAM_UNDEF ? BTM_BLE_SCAN_FAST_INT : p_cb->scan_int;
- UINT32 scan_win = p_cb->scan_win == BTM_BLE_SCAN_PARAM_UNDEF ? BTM_BLE_SCAN_FAST_WIN : p_cb->scan_win;
- BTM_TRACE_EVENT ("%s", __func__);
- if (start) {
- if (!BTM_BLE_IS_SCAN_ACTIVE(p_cb->scan_activity)) {
- if (p_select_cback != NULL) {
- btm_cb.ble_ctr_cb.p_select_cback = p_select_cback;
- }
- btm_execute_wl_dev_operation();
- btm_update_scanner_filter_policy(SP_ADV_WL);
- btm_cb.ble_ctr_cb.inq_var.scan_type = BTM_BLE_SCAN_MODE_PASS;
- /* Process advertising packets only from devices in the white list */
- if (btm_cb.cmn_ble_vsc_cb.extended_scan_support == 0) {
- /* use passive scan by default */
- if (!btsnd_hcic_ble_set_scan_params(BTM_BLE_SCAN_MODE_PASS,
- scan_int,
- scan_win,
- p_cb->addr_mgnt_cb.own_addr_type,
- SP_ADV_WL)) {
- return FALSE;
- }
- } else {
- if (!btm_ble_send_extended_scan_params(BTM_BLE_SCAN_MODE_PASS,
- scan_int,
- scan_win,
- p_cb->addr_mgnt_cb.own_addr_type,
- SP_ADV_WL)) {
- return FALSE;
- }
- }
- if (!btm_ble_topology_check(BTM_BLE_STATE_PASSIVE_SCAN)) {
- BTM_TRACE_ERROR("peripheral device cannot initiate passive scan for a selective connection");
- return FALSE;
- } else if (background_connections_pending()) {
- #if BLE_PRIVACY_SPT == TRUE
- btm_ble_enable_resolving_list_for_platform(BTM_BLE_RL_SCAN);
- #endif
- if (!btsnd_hcic_ble_set_scan_enable(TRUE, TRUE)) { /* duplicate filtering enabled */
- return FALSE;
- }
- /* mark up inquiry status flag */
- p_cb->scan_activity |= BTM_LE_SELECT_CONN_ACTIVE;
- p_cb->wl_state |= BTM_BLE_WL_SCAN;
- }
- } else {
- BTM_TRACE_ERROR("scan active, can not start selective connection procedure");
- return FALSE;
- }
- } else { /* disable selective connection mode */
- p_cb->scan_activity &= ~BTM_LE_SELECT_CONN_ACTIVE;
- p_cb->p_select_cback = NULL;
- p_cb->wl_state &= ~BTM_BLE_WL_SCAN;
- /* stop scanning */
- if (!BTM_BLE_IS_SCAN_ACTIVE(p_cb->scan_activity)) {
- btm_ble_stop_scan(); /* duplicate filtering enabled */
- }
- }
- return TRUE;
- }
- /*******************************************************************************
- **
- ** Function btm_ble_initiate_select_conn
- **
- ** Description This function is to start/stop selective connection procedure.
- **
- ** Parameters start: TRUE to start; FALSE to stop.
- ** p_select_cback: callback function to return application
- ** selection.
- **
- ** Returns BOOLEAN: selective connection procedure is started.
- **
- *******************************************************************************/
- void btm_ble_initiate_select_conn(BD_ADDR bda)
- {
- BTM_TRACE_EVENT ("btm_ble_initiate_select_conn");
- /* use direct connection procedure to initiate connection */
- if (!L2CA_ConnectFixedChnl(L2CAP_ATT_CID, bda, BLE_ADDR_UNKNOWN_TYPE, FALSE)) {
- BTM_TRACE_ERROR("btm_ble_initiate_select_conn failed");
- }
- }
- /*******************************************************************************
- **
- ** Function btm_ble_suspend_bg_conn
- **
- ** Description This function is to suspend an active background connection
- ** procedure.
- **
- ** Parameters none.
- **
- ** Returns none.
- **
- *******************************************************************************/
- BOOLEAN btm_ble_suspend_bg_conn(void)
- {
- BTM_TRACE_EVENT ("%s\n", __func__);
- if (btm_cb.ble_ctr_cb.bg_conn_type == BTM_BLE_CONN_AUTO) {
- return btm_ble_start_auto_conn(FALSE);
- } else if (btm_cb.ble_ctr_cb.bg_conn_type == BTM_BLE_CONN_SELECTIVE) {
- return btm_ble_start_select_conn(FALSE, NULL);
- }
- return FALSE;
- }
- /*******************************************************************************
- **
- ** Function btm_suspend_wl_activity
- **
- ** Description This function is to suspend white list related activity
- **
- ** Returns none.
- **
- *******************************************************************************/
- static void btm_suspend_wl_activity(tBTM_BLE_WL_STATE wl_state)
- {
- if (wl_state & BTM_BLE_WL_INIT) {
- btm_ble_start_auto_conn(FALSE);
- }
- if (wl_state & BTM_BLE_WL_SCAN) {
- btm_ble_start_select_conn(FALSE, NULL);
- }
- if (wl_state & BTM_BLE_WL_ADV) {
- btm_ble_stop_adv();
- }
- }
- /*******************************************************************************
- **
- ** Function btm_resume_wl_activity
- **
- ** Description This function is to resume white list related activity
- **
- ** Returns none.
- **
- *******************************************************************************/
- void btm_resume_wl_activity(tBTM_BLE_WL_STATE wl_state)
- {
- btm_ble_resume_bg_conn();
- if (wl_state & BTM_BLE_WL_ADV) {
- btm_ble_start_adv();
- }
- }
- /*******************************************************************************
- **
- ** Function btm_wl_update_to_controller
- **
- ** Description This function is to update white list to controller
- **
- ** Returns none.
- **
- *******************************************************************************/
- static void btm_wl_update_to_controller(void)
- {
- /* whitelist will be added in the btm_ble_resume_bg_conn(), we do not
- support background connection now, so we nedd to use btm_execute_wl_dev_operation
- to add whitelist directly ,if we support background connection in the future,
- please delete btm_execute_wl_dev_operation(). */
- btm_execute_wl_dev_operation();
- }
- /*******************************************************************************
- **
- ** Function btm_ble_resume_bg_conn
- **
- ** Description This function is to resume a background auto connection
- ** procedure.
- **
- ** Parameters none.
- **
- ** Returns none.
- **
- *******************************************************************************/
- BOOLEAN btm_ble_resume_bg_conn(void)
- {
- tBTM_BLE_CB *p_cb = &btm_cb.ble_ctr_cb;
- BOOLEAN ret = FALSE;
- if (p_cb->bg_conn_type != BTM_BLE_CONN_NONE) {
- if (p_cb->bg_conn_type == BTM_BLE_CONN_AUTO) {
- ret = btm_ble_start_auto_conn(TRUE);
- }
- if (p_cb->bg_conn_type == BTM_BLE_CONN_SELECTIVE) {
- ret = btm_ble_start_select_conn(TRUE, btm_cb.ble_ctr_cb.p_select_cback);
- }
- }
- return ret;
- }
- /*******************************************************************************
- **
- ** Function btm_ble_get_conn_st
- **
- ** Description This function get BLE connection state
- **
- ** Returns connection state
- **
- *******************************************************************************/
- tBTM_BLE_CONN_ST btm_ble_get_conn_st(void)
- {
- return btm_cb.ble_ctr_cb.conn_state;
- }
- /*******************************************************************************
- **
- ** Function btm_ble_set_conn_st
- **
- ** Description This function set BLE connection state
- **
- ** Returns None.
- **
- *******************************************************************************/
- void btm_ble_set_conn_st(tBTM_BLE_CONN_ST new_st)
- {
- btm_cb.ble_ctr_cb.conn_state = new_st;
- if (new_st == BLE_BG_CONN || new_st == BLE_DIR_CONN) {
- btm_ble_set_topology_mask(BTM_BLE_STATE_INIT_BIT);
- } else {
- btm_ble_clear_topology_mask(BTM_BLE_STATE_INIT_BIT);
- }
- }
- /*******************************************************************************
- **
- ** Function btm_ble_enqueue_direct_conn_req
- **
- ** Description This function enqueue the direct connection request
- **
- ** Returns None.
- **
- *******************************************************************************/
- void btm_ble_enqueue_direct_conn_req(void *p_param)
- {
- tBTM_BLE_CONN_REQ *p = (tBTM_BLE_CONN_REQ *)osi_malloc(sizeof(tBTM_BLE_CONN_REQ));
- p->p_param = p_param;
- fixed_queue_enqueue(btm_cb.ble_ctr_cb.conn_pending_q, p, FIXED_QUEUE_MAX_TIMEOUT);
- }
- /*******************************************************************************
- **
- ** Function btm_send_pending_direct_conn
- **
- ** Description This function send the pending direct connection request in queue
- **
- ** Returns TRUE if started, FALSE otherwise
- **
- *******************************************************************************/
- BOOLEAN btm_send_pending_direct_conn(void)
- {
- tBTM_BLE_CONN_REQ *p_req;
- BOOLEAN rt = FALSE;
- p_req = (tBTM_BLE_CONN_REQ*)fixed_queue_dequeue(btm_cb.ble_ctr_cb.conn_pending_q, 0);
- if (p_req != NULL) {
- rt = l2cble_init_direct_conn((tL2C_LCB *)(p_req->p_param));
- osi_free((void *)p_req);
- }
- return rt;
- }
- #endif
|