/****************************************************************************** * * Copyright 2006-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 action functions for BTA JV APIs. * ******************************************************************************/ #include #include #include #include #include #include #include "avct_api.h" #include "avdt_api.h" #include "bt_common.h" #include "bt_types.h" #include "bta_api.h" #include "bta_jv_api.h" #include "bta_jv_co.h" #include "bta_jv_int.h" #include "bta_sys.h" #include "btm_api.h" #include "btm_int.h" #include "device/include/controller.h" #include "gap_api.h" #include "l2c_api.h" #include "osi/include/allocator.h" #include "port_api.h" #include "rfcdefs.h" #include "sdp_api.h" #include "stack/l2cap/l2c_int.h" #include "utl.h" #include "osi/include/osi.h" using bluetooth::Uuid; tBTA_JV_CB bta_jv_cb; /* one of these exists for each client */ struct fc_client { struct fc_client* next_all_list; struct fc_client* next_chan_list; RawAddress remote_addr; uint32_t id; tBTA_JV_L2CAP_CBACK* p_cback; uint32_t l2cap_socket_id; uint16_t handle; uint16_t chan; uint8_t sec_id; unsigned server : 1; unsigned init_called : 1; }; /* one of these exists for each channel we're dealing with */ struct fc_channel { struct fc_channel* next; struct fc_client* clients; uint8_t has_server : 1; uint16_t chan; }; static struct fc_client* fc_clients; static struct fc_channel* fc_channels; static uint32_t fc_next_id; static void fcchan_conn_chng_cbk(uint16_t chan, const RawAddress& bd_addr, bool connected, uint16_t reason, tBT_TRANSPORT); static void fcchan_data_cbk(uint16_t chan, const RawAddress& bd_addr, BT_HDR* p_buf); static tBTA_JV_PCB* bta_jv_add_rfc_port(tBTA_JV_RFC_CB* p_cb, tBTA_JV_PCB* p_pcb_open); static tBTA_JV_STATUS bta_jv_free_set_pm_profile_cb(uint32_t jv_handle); static void bta_jv_pm_conn_busy(tBTA_JV_PM_CB* p_cb); static void bta_jv_pm_conn_idle(tBTA_JV_PM_CB* p_cb); static void bta_jv_pm_state_change(tBTA_JV_PM_CB* p_cb, const tBTA_JV_CONN_STATE state); /******************************************************************************* * * Function bta_jv_alloc_sec_id * * Description allocate a security id * * Returns * ******************************************************************************/ uint8_t bta_jv_alloc_sec_id(void) { uint8_t ret = 0; int i; for (i = 0; i < BTA_JV_NUM_SERVICE_ID; i++) { if (0 == bta_jv_cb.sec_id[i]) { bta_jv_cb.sec_id[i] = BTA_JV_FIRST_SERVICE_ID + i; ret = bta_jv_cb.sec_id[i]; break; } } return ret; } static int get_sec_id_used(void) { int i; int used = 0; for (i = 0; i < BTA_JV_NUM_SERVICE_ID; i++) { if (bta_jv_cb.sec_id[i]) used++; } if (used == BTA_JV_NUM_SERVICE_ID) LOG(ERROR) << __func__ << ": sec id exceeds the limit=" << BTA_JV_NUM_SERVICE_ID; return used; } static int get_rfc_cb_used(void) { int i; int used = 0; for (i = 0; i < BTA_JV_MAX_RFC_CONN; i++) { if (bta_jv_cb.rfc_cb[i].handle) used++; } if (used == BTA_JV_MAX_RFC_CONN) LOG(ERROR) << __func__ << ": rfc ctrl block exceeds the limit=" << BTA_JV_MAX_RFC_CONN; return used; } /******************************************************************************* * * Function bta_jv_free_sec_id * * Description free the given security id * * Returns * ******************************************************************************/ static void bta_jv_free_sec_id(uint8_t* p_sec_id) { uint8_t sec_id = *p_sec_id; *p_sec_id = 0; if (sec_id >= BTA_JV_FIRST_SERVICE_ID && sec_id <= BTA_JV_LAST_SERVICE_ID) { BTM_SecClrService(sec_id); bta_jv_cb.sec_id[sec_id - BTA_JV_FIRST_SERVICE_ID] = 0; } } /******************************************************************************* * * Function bta_jv_alloc_rfc_cb * * Description allocate a control block for the given port handle * * Returns * ******************************************************************************/ tBTA_JV_RFC_CB* bta_jv_alloc_rfc_cb(uint16_t port_handle, tBTA_JV_PCB** pp_pcb) { tBTA_JV_RFC_CB* p_cb = NULL; tBTA_JV_PCB* p_pcb; int i, j; for (i = 0; i < BTA_JV_MAX_RFC_CONN; i++) { if (0 == bta_jv_cb.rfc_cb[i].handle) { p_cb = &bta_jv_cb.rfc_cb[i]; /* mask handle to distinguish it with L2CAP handle */ p_cb->handle = (i + 1) | BTA_JV_RFCOMM_MASK; p_cb->max_sess = 1; p_cb->curr_sess = 1; for (j = 0; j < BTA_JV_MAX_RFC_SR_SESSION; j++) p_cb->rfc_hdl[j] = 0; p_cb->rfc_hdl[0] = port_handle; VLOG(2) << __func__ << "port_handle=" << +port_handle << ", handle=" << loghex(p_cb->handle); p_pcb = &bta_jv_cb.port_cb[port_handle - 1]; p_pcb->handle = p_cb->handle; p_pcb->port_handle = port_handle; p_pcb->p_pm_cb = NULL; *pp_pcb = p_pcb; break; } } if (p_cb == NULL) { LOG(ERROR) << __func__ << "port_handle=" << port_handle << " ctrl block exceeds limit:" << BTA_JV_MAX_RFC_CONN; } return p_cb; } /******************************************************************************* * * Function bta_jv_rfc_port_to_pcb * * Description find the port control block associated with the given port * handle * * Returns * ******************************************************************************/ tBTA_JV_PCB* bta_jv_rfc_port_to_pcb(uint16_t port_handle) { tBTA_JV_PCB* p_pcb = NULL; if ((port_handle > 0) && (port_handle <= MAX_RFC_PORTS) && bta_jv_cb.port_cb[port_handle - 1].handle) { p_pcb = &bta_jv_cb.port_cb[port_handle - 1]; } return p_pcb; } /******************************************************************************* * * Function bta_jv_rfc_port_to_cb * * Description find the RFCOMM control block associated with the given port * handle * * Returns * ******************************************************************************/ tBTA_JV_RFC_CB* bta_jv_rfc_port_to_cb(uint16_t port_handle) { tBTA_JV_RFC_CB* p_cb = NULL; uint32_t handle; if ((port_handle > 0) && (port_handle <= MAX_RFC_PORTS) && bta_jv_cb.port_cb[port_handle - 1].handle) { handle = bta_jv_cb.port_cb[port_handle - 1].handle; handle &= BTA_JV_RFC_HDL_MASK; handle &= ~BTA_JV_RFCOMM_MASK; if (handle) p_cb = &bta_jv_cb.rfc_cb[handle - 1]; } else { LOG(WARNING) << __func__ << ": jv handle not found port_handle:" << port_handle; } return p_cb; } static tBTA_JV_STATUS bta_jv_free_rfc_cb(tBTA_JV_RFC_CB* p_cb, tBTA_JV_PCB* p_pcb) { tBTA_JV_STATUS status = BTA_JV_SUCCESS; bool remove_server = false; int close_pending = 0; if (!p_cb || !p_pcb) { LOG(ERROR) << __func__ << " p_cb or p_pcb cannot be null"; return BTA_JV_FAILURE; } VLOG(2) << __func__ << ": max_sess=" << p_cb->max_sess << ", curr_sess=" << p_cb->curr_sess << ", p_pcb=" << p_pcb << ", user=" << p_pcb->rfcomm_slot_id << ", state=" << p_pcb->state << ", jv handle=" << loghex(p_pcb->handle); if (p_cb->curr_sess <= 0) return BTA_JV_SUCCESS; switch (p_pcb->state) { case BTA_JV_ST_CL_CLOSING: case BTA_JV_ST_SR_CLOSING: LOG(WARNING) << __func__ << ": return on closing, port state=" << p_pcb->state << ", scn=" << p_cb->scn << ", p_pcb=" << p_pcb << ", user_data=" << p_pcb->rfcomm_slot_id; status = BTA_JV_FAILURE; return status; case BTA_JV_ST_CL_OPEN: case BTA_JV_ST_CL_OPENING: VLOG(2) << __func__ << ": state=" << p_pcb->state << ", scn=" << p_cb->scn << ", user_data=" << p_pcb->rfcomm_slot_id; p_pcb->state = BTA_JV_ST_CL_CLOSING; break; case BTA_JV_ST_SR_LISTEN: p_pcb->state = BTA_JV_ST_SR_CLOSING; remove_server = true; VLOG(2) << __func__ << ": state: BTA_JV_ST_SR_LISTEN, scn=" << p_cb->scn << ", user_data=" << p_pcb->rfcomm_slot_id; break; case BTA_JV_ST_SR_OPEN: p_pcb->state = BTA_JV_ST_SR_CLOSING; VLOG(2) << ": state: BTA_JV_ST_SR_OPEN, scn=" << p_cb->scn << " user_data=" << p_pcb->rfcomm_slot_id; break; default: LOG(WARNING) << __func__ << ":failed, ignore port state= " << p_pcb->state << ", scn=" << p_cb->scn << ", p_pcb= " << p_pcb << ", jv handle=" << loghex(p_pcb->handle) << ", port_handle=" << p_pcb->port_handle << ", user_data=" << p_pcb->rfcomm_slot_id; status = BTA_JV_FAILURE; break; } if (BTA_JV_SUCCESS == status) { int port_status; if (!remove_server) port_status = RFCOMM_RemoveConnection(p_pcb->port_handle); else port_status = RFCOMM_RemoveServer(p_pcb->port_handle); if (port_status != PORT_SUCCESS) { status = BTA_JV_FAILURE; LOG(WARNING) << __func__ << ": Remove jv handle=" << loghex(p_pcb->handle) << ", state=" << p_pcb->state << ", port_status=" << port_status << ", port_handle=" << p_pcb->port_handle << ", close_pending=" << close_pending; } } if (!close_pending) { p_pcb->port_handle = 0; p_pcb->state = BTA_JV_ST_NONE; bta_jv_free_set_pm_profile_cb(p_pcb->handle); // Initialize congestion flags p_pcb->cong = false; p_pcb->rfcomm_slot_id = 0; int si = BTA_JV_RFC_HDL_TO_SIDX(p_pcb->handle); if (0 <= si && si < BTA_JV_MAX_RFC_SR_SESSION) p_cb->rfc_hdl[si] = 0; p_pcb->handle = 0; p_cb->curr_sess--; if (p_cb->curr_sess == 0) { p_cb->scn = 0; bta_jv_free_sec_id(&p_cb->sec_id); p_cb->p_cback = NULL; p_cb->handle = 0; p_cb->curr_sess = -1; } if (remove_server) { bta_jv_free_sec_id(&p_cb->sec_id); } } return status; } /******************************************************************************* * * Function bta_jv_free_l2c_cb * * Description free the given L2CAP control block * * Returns * ******************************************************************************/ tBTA_JV_STATUS bta_jv_free_l2c_cb(tBTA_JV_L2C_CB* p_cb) { tBTA_JV_STATUS status = BTA_JV_SUCCESS; if (BTA_JV_ST_NONE != p_cb->state) { bta_jv_free_set_pm_profile_cb((uint32_t)p_cb->handle); if (GAP_ConnClose(p_cb->handle) != BT_PASS) status = BTA_JV_FAILURE; } p_cb->psm = 0; p_cb->state = BTA_JV_ST_NONE; p_cb->cong = false; bta_jv_free_sec_id(&p_cb->sec_id); p_cb->p_cback = NULL; p_cb->handle = 0; p_cb->l2cap_socket_id = 0; return status; } /******************************************************************************* * * * Function bta_jv_clear_pm_cb * * Description clears jv pm control block and optionally calls * bta_sys_conn_close() * In general close_conn should be set to true to remove registering * with dm pm! * * WARNING: Make sure to clear pointer form port or l2c to this control block * too! * ******************************************************************************/ static void bta_jv_clear_pm_cb(tBTA_JV_PM_CB* p_pm_cb, bool close_conn) { /* needs to be called if registered with bta pm, otherwise we may run out of * dm pm slots! */ if (close_conn) bta_sys_conn_close(BTA_ID_JV, p_pm_cb->app_id, p_pm_cb->peer_bd_addr); p_pm_cb->state = BTA_JV_PM_FREE_ST; p_pm_cb->app_id = BTA_JV_PM_ALL; p_pm_cb->handle = BTA_JV_PM_HANDLE_CLEAR; p_pm_cb->peer_bd_addr = RawAddress::kEmpty; } /******************************************************************************* * * Function bta_jv_free_set_pm_profile_cb * * Description free pm profile control block * * Returns BTA_JV_SUCCESS if cb has been freed correctly, * BTA_JV_FAILURE in case of no profile has been registered or * already freed * ******************************************************************************/ static tBTA_JV_STATUS bta_jv_free_set_pm_profile_cb(uint32_t jv_handle) { tBTA_JV_STATUS status = BTA_JV_FAILURE; tBTA_JV_PM_CB** p_cb; int i, j, bd_counter = 0, appid_counter = 0; for (i = 0; i < BTA_JV_PM_MAX_NUM; i++) { p_cb = NULL; if ((bta_jv_cb.pm_cb[i].state != BTA_JV_PM_FREE_ST) && (jv_handle == bta_jv_cb.pm_cb[i].handle)) { for (j = 0; j < BTA_JV_PM_MAX_NUM; j++) { if (bta_jv_cb.pm_cb[j].peer_bd_addr == bta_jv_cb.pm_cb[i].peer_bd_addr) bd_counter++; if (bta_jv_cb.pm_cb[j].app_id == bta_jv_cb.pm_cb[i].app_id) appid_counter++; } VLOG(2) << __func__ << ": jv_handle=" << loghex(jv_handle) << ", idx=" << i << "app_id=" << bta_jv_cb.pm_cb[i].app_id << ", bd_counter=" << bd_counter << ", appid_counter=" << appid_counter; if (bd_counter > 1) { bta_jv_pm_conn_idle(&bta_jv_cb.pm_cb[i]); } if (bd_counter <= 1 || (appid_counter <= 1)) { bta_jv_clear_pm_cb(&bta_jv_cb.pm_cb[i], true); } else { bta_jv_clear_pm_cb(&bta_jv_cb.pm_cb[i], false); } if (BTA_JV_RFCOMM_MASK & jv_handle) { uint32_t hi = ((jv_handle & BTA_JV_RFC_HDL_MASK) & ~BTA_JV_RFCOMM_MASK) - 1; uint32_t si = BTA_JV_RFC_HDL_TO_SIDX(jv_handle); if (hi < BTA_JV_MAX_RFC_CONN && bta_jv_cb.rfc_cb[hi].p_cback && si < BTA_JV_MAX_RFC_SR_SESSION && bta_jv_cb.rfc_cb[hi].rfc_hdl[si]) { tBTA_JV_PCB* p_pcb = bta_jv_rfc_port_to_pcb(bta_jv_cb.rfc_cb[hi].rfc_hdl[si]); if (p_pcb) { if (NULL == p_pcb->p_pm_cb) LOG(WARNING) << __func__ << ": jv_handle=" << loghex(jv_handle) << ", port_handle=" << p_pcb->port_handle << ", i=" << i << ", no link to pm_cb?"; p_cb = &p_pcb->p_pm_cb; } } } else { if (jv_handle < BTA_JV_MAX_L2C_CONN) { tBTA_JV_L2C_CB* p_l2c_cb = &bta_jv_cb.l2c_cb[jv_handle]; if (NULL == p_l2c_cb->p_pm_cb) LOG(WARNING) << __func__ << ": jv_handle=" << loghex(jv_handle) << ", i=" << i << " no link to pm_cb?"; p_cb = &p_l2c_cb->p_pm_cb; } } if (p_cb) { *p_cb = NULL; status = BTA_JV_SUCCESS; } } } return status; } /******************************************************************************* * * Function bta_jv_alloc_set_pm_profile_cb * * Description set PM profile control block * * Returns pointer to allocated cb or NULL in case of failure * ******************************************************************************/ static tBTA_JV_PM_CB* bta_jv_alloc_set_pm_profile_cb(uint32_t jv_handle, tBTA_JV_PM_ID app_id) { bool bRfcHandle = (jv_handle & BTA_JV_RFCOMM_MASK) != 0; RawAddress peer_bd_addr = RawAddress::kEmpty; int i, j; tBTA_JV_PM_CB** pp_cb; for (i = 0; i < BTA_JV_PM_MAX_NUM; i++) { pp_cb = NULL; if (bta_jv_cb.pm_cb[i].state == BTA_JV_PM_FREE_ST) { /* rfc handle bd addr retrieval requires core stack handle */ if (bRfcHandle) { for (j = 0; j < BTA_JV_MAX_RFC_CONN; j++) { if (jv_handle == bta_jv_cb.port_cb[j].handle) { pp_cb = &bta_jv_cb.port_cb[j].p_pm_cb; if (PORT_SUCCESS != PORT_CheckConnection(bta_jv_cb.port_cb[j].port_handle, &peer_bd_addr, NULL)) { i = BTA_JV_PM_MAX_NUM; } break; } } } else { /* use jv handle for l2cap bd address retrieval */ for (j = 0; j < BTA_JV_MAX_L2C_CONN; j++) { if (jv_handle == bta_jv_cb.l2c_cb[j].handle) { pp_cb = &bta_jv_cb.l2c_cb[j].p_pm_cb; const RawAddress* p_bd_addr = GAP_ConnGetRemoteAddr((uint16_t)jv_handle); if (p_bd_addr) peer_bd_addr = *p_bd_addr; else i = BTA_JV_PM_MAX_NUM; break; } } } VLOG(2) << __func__ << ": handle=" << loghex(jv_handle) << ", app_id=" << app_id << ", idx=" << i << ", BTA_JV_PM_MAX_NUM=" << BTA_JV_PM_MAX_NUM << ", pp_cb=" << pp_cb; break; } } if ((i != BTA_JV_PM_MAX_NUM) && (NULL != pp_cb)) { *pp_cb = &bta_jv_cb.pm_cb[i]; bta_jv_cb.pm_cb[i].handle = jv_handle; bta_jv_cb.pm_cb[i].app_id = app_id; bta_jv_cb.pm_cb[i].peer_bd_addr = peer_bd_addr; bta_jv_cb.pm_cb[i].state = BTA_JV_PM_IDLE_ST; return &bta_jv_cb.pm_cb[i]; } LOG(WARNING) << __func__ << ": handle=" << loghex(jv_handle) << ", app_id=" << app_id << ", return NULL"; return NULL; } /******************************************************************************* * * Function bta_jv_check_psm * * Description for now use only the legal PSM per JSR82 spec * * Returns true, if allowed * ******************************************************************************/ bool bta_jv_check_psm(uint16_t psm) { bool ret = false; if (L2C_IS_VALID_PSM(psm)) { if (psm < 0x1001) { /* see if this is defined by spec */ switch (psm) { case SDP_PSM: /* 1 */ case BT_PSM_RFCOMM: /* 3 */ /* do not allow java app to use these 2 PSMs */ break; case TCS_PSM_INTERCOM: /* 5 */ case TCS_PSM_CORDLESS: /* 7 */ if (!bta_sys_is_register(BTA_ID_CT) && !bta_sys_is_register(BTA_ID_CG)) ret = true; break; case BT_PSM_BNEP: /* F */ if (!bta_sys_is_register(BTA_ID_PAN)) ret = true; break; case HID_PSM_CONTROL: /* 0x11 */ case HID_PSM_INTERRUPT: /* 0x13 */ // FIX: allow HID Device and HID Host to coexist if (!bta_sys_is_register(BTA_ID_HD) || !bta_sys_is_register(BTA_ID_HH)) ret = true; break; case AVCT_PSM: /* 0x17 */ case AVDT_PSM: /* 0x19 */ if ((!bta_sys_is_register(BTA_ID_AV)) && (!bta_sys_is_register(BTA_ID_AVK))) ret = true; break; default: ret = true; break; } } else { ret = true; } } return ret; } /* Initialises the JAVA I/F */ void bta_jv_enable(tBTA_JV_DM_CBACK* p_cback) { tBTA_JV_STATUS status = BTA_JV_SUCCESS; bta_jv_cb.p_dm_cback = p_cback; tBTA_JV bta_jv; bta_jv.status = status; bta_jv_cb.p_dm_cback(BTA_JV_ENABLE_EVT, &bta_jv, 0); memset(bta_jv_cb.free_psm_list, 0, sizeof(bta_jv_cb.free_psm_list)); } /** Disables the BT device manager free the resources used by java */ void bta_jv_disable() { LOG(INFO) << __func__; } /** * We keep a list of PSM's that have been freed from JAVA, for reuse. * This function will return a free PSM, and delete it from the free * list. * If no free PSMs exist, 0 will be returned. */ static uint16_t bta_jv_get_free_psm() { const int cnt = sizeof(bta_jv_cb.free_psm_list) / sizeof(bta_jv_cb.free_psm_list[0]); for (int i = 0; i < cnt; i++) { uint16_t psm = bta_jv_cb.free_psm_list[i]; if (psm != 0) { VLOG(2) << __func__ << ": Reusing PSM=" << loghex(psm); bta_jv_cb.free_psm_list[i] = 0; return psm; } } return 0; } static void bta_jv_set_free_psm(uint16_t psm) { int free_index = -1; const int cnt = sizeof(bta_jv_cb.free_psm_list) / sizeof(bta_jv_cb.free_psm_list[0]); for (int i = 0; i < cnt; i++) { if (bta_jv_cb.free_psm_list[i] == 0) { free_index = i; } else if (psm == bta_jv_cb.free_psm_list[i]) { return; // PSM already freed? } } if (free_index != -1) { bta_jv_cb.free_psm_list[free_index] = psm; VLOG(2) << __func__ << ": Recycling PSM=" << loghex(psm); } else { LOG(ERROR) << __func__ << ": unable to free psm=" << loghex(psm) << " no more free slots"; } } /** Obtain a free SCN (Server Channel Number) (RFCOMM channel or L2CAP PSM) */ void bta_jv_get_channel_id( int32_t type /* One of BTA_JV_CONN_TYPE_ */, int32_t channel /* optionally request a specific channel */, uint32_t l2cap_socket_id, uint32_t rfcomm_slot_id) { uint16_t psm = 0; switch (type) { case BTA_JV_CONN_TYPE_RFCOMM: { uint8_t scn = 0; if (channel > 0) { if (!BTM_TryAllocateSCN(channel)) { LOG(ERROR) << "rfc channel=" << channel << " already in use or invalid"; channel = 0; } } else { channel = BTM_AllocateSCN(); if (channel == 0) { LOG(ERROR) << "run out of rfc channels"; channel = 0; } } if (channel != 0) { bta_jv_cb.scn[channel - 1] = true; scn = (uint8_t)channel; } if (bta_jv_cb.p_dm_cback) { tBTA_JV bta_jv; bta_jv.scn = scn; bta_jv_cb.p_dm_cback(BTA_JV_GET_SCN_EVT, &bta_jv, rfcomm_slot_id); } return; } case BTA_JV_CONN_TYPE_L2CAP: psm = bta_jv_get_free_psm(); if (psm == 0) { psm = L2CA_AllocatePSM(); VLOG(2) << __func__ << ": returned PSM=" << loghex(psm); } break; case BTA_JV_CONN_TYPE_L2CAP_LE: psm = L2CA_AllocateLePSM(); if (psm == 0) { LOG(ERROR) << __func__ << ": Error: No free LE PSM available"; } break; default: break; } if (bta_jv_cb.p_dm_cback) { tBTA_JV bta_jv; bta_jv.psm = psm; bta_jv_cb.p_dm_cback(BTA_JV_GET_PSM_EVT, &bta_jv, l2cap_socket_id); } } /** free a SCN */ void bta_jv_free_scn(int32_t type /* One of BTA_JV_CONN_TYPE_ */, uint16_t scn) { switch (type) { case BTA_JV_CONN_TYPE_RFCOMM: { if (scn > 0 && scn <= BTA_JV_MAX_SCN && bta_jv_cb.scn[scn - 1]) { /* this scn is used by JV */ bta_jv_cb.scn[scn - 1] = false; BTM_FreeSCN(scn); } break; } case BTA_JV_CONN_TYPE_L2CAP: bta_jv_set_free_psm(scn); break; case BTA_JV_CONN_TYPE_L2CAP_LE: VLOG(2) << __func__ << ": type=BTA_JV_CONN_TYPE_L2CAP_LE. psm=" << scn; L2CA_FreeLePSM(scn); break; default: break; } } /******************************************************************************* * * Function bta_jv_start_discovery_cback * * Description Callback for Start Discovery * * Returns void * ******************************************************************************/ static void bta_jv_start_discovery_cback(uint16_t result, void* user_data) { tBTA_JV_STATUS status; uint32_t* p_rfcomm_slot_id = static_cast(user_data); VLOG(2) << __func__ << ": res=" << loghex(result); bta_jv_cb.sdp_active = BTA_JV_SDP_ACT_NONE; if (bta_jv_cb.p_dm_cback) { tBTA_JV_DISCOVERY_COMP dcomp; dcomp.scn = 0; status = BTA_JV_FAILURE; if (result == SDP_SUCCESS || result == SDP_DB_FULL) { tSDP_DISC_REC* p_sdp_rec = NULL; tSDP_PROTOCOL_ELEM pe; VLOG(2) << __func__ << ": bta_jv_cb.uuid=" << bta_jv_cb.uuid; p_sdp_rec = SDP_FindServiceUUIDInDb(p_bta_jv_cfg->p_sdp_db, bta_jv_cb.uuid, p_sdp_rec); VLOG(2) << __func__ << ": p_sdp_rec=" << p_sdp_rec; if (p_sdp_rec && SDP_FindProtocolListElemInRec(p_sdp_rec, UUID_PROTOCOL_RFCOMM, &pe)) { dcomp.scn = (uint8_t)pe.params[0]; status = BTA_JV_SUCCESS; } } dcomp.status = status; tBTA_JV bta_jv; bta_jv.disc_comp = dcomp; bta_jv_cb.p_dm_cback(BTA_JV_DISCOVERY_COMP_EVT, &bta_jv, *p_rfcomm_slot_id); osi_free(p_rfcomm_slot_id); } } /* Discovers services on a remote device */ void bta_jv_start_discovery(const RawAddress& bd_addr, uint16_t num_uuid, bluetooth::Uuid* uuid_list, uint32_t rfcomm_slot_id) { tBTA_JV_STATUS status = BTA_JV_FAILURE; VLOG(2) << __func__ << ": in, sdp_active=" << bta_jv_cb.sdp_active; if (bta_jv_cb.sdp_active != BTA_JV_SDP_ACT_NONE) { /* SDP is still in progress */ status = BTA_JV_BUSY; if (bta_jv_cb.p_dm_cback) { tBTA_JV bta_jv; bta_jv.status = status; bta_jv_cb.p_dm_cback(BTA_JV_DISCOVERY_COMP_EVT, &bta_jv, rfcomm_slot_id); } return; } /* init the database/set up the filter */ VLOG(2) << __func__ << ": call SDP_InitDiscoveryDb, num_uuid=" << num_uuid; SDP_InitDiscoveryDb(p_bta_jv_cfg->p_sdp_db, p_bta_jv_cfg->sdp_db_size, num_uuid, uuid_list, 0, NULL); /* tell SDP to keep the raw data */ p_bta_jv_cfg->p_sdp_db->raw_data = p_bta_jv_cfg->p_sdp_raw_data; p_bta_jv_cfg->p_sdp_db->raw_size = p_bta_jv_cfg->sdp_raw_size; bta_jv_cb.p_sel_raw_data = 0; bta_jv_cb.uuid = uuid_list[0]; bta_jv_cb.sdp_active = BTA_JV_SDP_ACT_YES; uint32_t* rfcomm_slot_id_copy = (uint32_t*)osi_malloc(sizeof(uint32_t)); *rfcomm_slot_id_copy = rfcomm_slot_id; if (!SDP_ServiceSearchAttributeRequest2(bd_addr, p_bta_jv_cfg->p_sdp_db, bta_jv_start_discovery_cback, (void*)rfcomm_slot_id_copy)) { bta_jv_cb.sdp_active = BTA_JV_SDP_ACT_NONE; /* failed to start SDP. report the failure right away */ if (bta_jv_cb.p_dm_cback) { tBTA_JV bta_jv; bta_jv.status = status; bta_jv_cb.p_dm_cback(BTA_JV_DISCOVERY_COMP_EVT, &bta_jv, rfcomm_slot_id); } } /* else report the result when the cback is called */ } /* Create an SDP record with the given attributes */ void bta_jv_create_record(uint32_t rfcomm_slot_id) { tBTA_JV_CREATE_RECORD evt_data; evt_data.status = BTA_JV_SUCCESS; if (bta_jv_cb.p_dm_cback) { // callback immediately to create the sdp record in stack thread context tBTA_JV bta_jv; bta_jv.create_rec = evt_data; bta_jv_cb.p_dm_cback(BTA_JV_CREATE_RECORD_EVT, &bta_jv, rfcomm_slot_id); } } /* Delete an SDP record */ void bta_jv_delete_record(uint32_t handle) { if (handle) { /* this is a record created by btif layer*/ SDP_DeleteRecord(handle); } } /******************************************************************************* * * Function bta_jv_l2cap_client_cback * * Description handles the l2cap client events * * Returns void * ******************************************************************************/ static void bta_jv_l2cap_client_cback(uint16_t gap_handle, uint16_t event, tGAP_CB_DATA* data) { tBTA_JV_L2C_CB* p_cb = &bta_jv_cb.l2c_cb[gap_handle]; tBTA_JV evt_data; if (gap_handle >= BTA_JV_MAX_L2C_CONN && !p_cb->p_cback) return; VLOG(2) << __func__ << ": gap_handle=" << gap_handle << ", evt=" << loghex(event); evt_data.l2c_open.status = BTA_JV_SUCCESS; evt_data.l2c_open.handle = gap_handle; switch (event) { case GAP_EVT_CONN_OPENED: evt_data.l2c_open.rem_bda = *GAP_ConnGetRemoteAddr(gap_handle); evt_data.l2c_open.tx_mtu = GAP_ConnGetRemMtuSize(gap_handle); p_cb->state = BTA_JV_ST_CL_OPEN; p_cb->p_cback(BTA_JV_L2CAP_OPEN_EVT, &evt_data, p_cb->l2cap_socket_id); break; case GAP_EVT_CONN_CLOSED: p_cb->state = BTA_JV_ST_NONE; bta_jv_free_sec_id(&p_cb->sec_id); evt_data.l2c_close.async = true; p_cb->p_cback(BTA_JV_L2CAP_CLOSE_EVT, &evt_data, p_cb->l2cap_socket_id); p_cb->p_cback = NULL; break; case GAP_EVT_CONN_DATA_AVAIL: evt_data.data_ind.handle = gap_handle; /* Reset idle timer to avoid requesting sniff mode while receiving data */ bta_jv_pm_conn_busy(p_cb->p_pm_cb); p_cb->p_cback(BTA_JV_L2CAP_DATA_IND_EVT, &evt_data, p_cb->l2cap_socket_id); bta_jv_pm_conn_idle(p_cb->p_pm_cb); break; case GAP_EVT_TX_EMPTY: bta_jv_pm_conn_idle(p_cb->p_pm_cb); break; case GAP_EVT_CONN_CONGESTED: case GAP_EVT_CONN_UNCONGESTED: p_cb->cong = (event == GAP_EVT_CONN_CONGESTED) ? true : false; evt_data.l2c_cong.cong = p_cb->cong; p_cb->p_cback(BTA_JV_L2CAP_CONG_EVT, &evt_data, p_cb->l2cap_socket_id); break; default: break; } } /* makes an l2cap client connection */ void bta_jv_l2cap_connect(int32_t type, tBTA_SEC sec_mask, tBTA_JV_ROLE role, uint16_t remote_psm, uint16_t rx_mtu, const RawAddress& peer_bd_addr, std::unique_ptr cfg_param, std::unique_ptr ertm_info, tBTA_JV_L2CAP_CBACK* p_cback, uint32_t l2cap_socket_id) { uint16_t handle = GAP_INVALID_HANDLE; uint8_t chan_mode_mask = GAP_FCR_CHAN_OPT_BASIC; tL2CAP_CFG_INFO cfg; memset(&cfg, 0, sizeof(tL2CAP_CFG_INFO)); if (cfg_param) { cfg = *cfg_param; if (cfg.fcr_present && cfg.fcr.mode == L2CAP_FCR_ERTM_MODE) { chan_mode_mask = GAP_FCR_CHAN_OPT_ERTM; } } /* We need to use this value for MTU to be able to handle cases where cfg is * not set in req. */ cfg.mtu_present = true; cfg.mtu = rx_mtu; /* TODO: DM role manager L2CA_SetDesireRole(role); */ uint8_t sec_id = bta_jv_alloc_sec_id(); tBTA_JV_L2CAP_CL_INIT evt_data; evt_data.sec_id = sec_id; evt_data.status = BTA_JV_FAILURE; if (sec_id) { /* PSM checking is not required for LE COC */ if ((type != BTA_JV_CONN_TYPE_L2CAP) || (bta_jv_check_psm(remote_psm))) /* allowed */ { uint16_t max_mps = 0xffff; // Let GAP_ConnOpen set the max_mps. handle = GAP_ConnOpen("", sec_id, 0, &peer_bd_addr, remote_psm, max_mps, &cfg, ertm_info.get(), sec_mask, chan_mode_mask, bta_jv_l2cap_client_cback, type); if (handle != GAP_INVALID_HANDLE) { evt_data.status = BTA_JV_SUCCESS; } } } if (evt_data.status == BTA_JV_SUCCESS) { tBTA_JV_L2C_CB* p_cb; p_cb = &bta_jv_cb.l2c_cb[handle]; p_cb->handle = handle; p_cb->p_cback = p_cback; p_cb->l2cap_socket_id = l2cap_socket_id; p_cb->psm = 0; /* not a server */ p_cb->sec_id = sec_id; p_cb->state = BTA_JV_ST_CL_OPENING; } else { bta_jv_free_sec_id(&sec_id); } evt_data.handle = handle; if (p_cback) { tBTA_JV bta_jv; bta_jv.l2c_cl_init = evt_data; p_cback(BTA_JV_L2CAP_CL_INIT_EVT, &bta_jv, l2cap_socket_id); } } /** Close an L2CAP client connection */ void bta_jv_l2cap_close(uint32_t handle, tBTA_JV_L2C_CB* p_cb) { tBTA_JV_L2CAP_CLOSE evt_data; tBTA_JV_L2CAP_CBACK* p_cback = p_cb->p_cback; uint32_t l2cap_socket_id = p_cb->l2cap_socket_id; evt_data.handle = handle; evt_data.status = bta_jv_free_l2c_cb(p_cb); evt_data.async = false; if (p_cback) { tBTA_JV bta_jv; bta_jv.l2c_close = evt_data; p_cback(BTA_JV_L2CAP_CLOSE_EVT, &bta_jv, l2cap_socket_id); } } /******************************************************************************* * * Function bta_jv_l2cap_server_cback * * Description handles the l2cap server callback * * Returns void * ******************************************************************************/ static void bta_jv_l2cap_server_cback(uint16_t gap_handle, uint16_t event, tGAP_CB_DATA* data) { tBTA_JV_L2C_CB* p_cb = &bta_jv_cb.l2c_cb[gap_handle]; tBTA_JV evt_data; tBTA_JV_L2CAP_CBACK* p_cback; uint32_t socket_id; if (gap_handle >= BTA_JV_MAX_L2C_CONN && !p_cb->p_cback) return; VLOG(2) << __func__ << ": gap_handle=" << gap_handle << ", evt=" << loghex(event); evt_data.l2c_open.status = BTA_JV_SUCCESS; evt_data.l2c_open.handle = gap_handle; switch (event) { case GAP_EVT_CONN_OPENED: evt_data.l2c_open.rem_bda = *GAP_ConnGetRemoteAddr(gap_handle); evt_data.l2c_open.tx_mtu = GAP_ConnGetRemMtuSize(gap_handle); p_cb->state = BTA_JV_ST_SR_OPEN; p_cb->p_cback(BTA_JV_L2CAP_OPEN_EVT, &evt_data, p_cb->l2cap_socket_id); break; case GAP_EVT_CONN_CLOSED: evt_data.l2c_close.async = true; evt_data.l2c_close.handle = p_cb->handle; p_cback = p_cb->p_cback; socket_id = p_cb->l2cap_socket_id; evt_data.l2c_close.status = bta_jv_free_l2c_cb(p_cb); p_cback(BTA_JV_L2CAP_CLOSE_EVT, &evt_data, socket_id); break; case GAP_EVT_CONN_DATA_AVAIL: evt_data.data_ind.handle = gap_handle; /* Reset idle timer to avoid requesting sniff mode while receiving data */ bta_jv_pm_conn_busy(p_cb->p_pm_cb); p_cb->p_cback(BTA_JV_L2CAP_DATA_IND_EVT, &evt_data, p_cb->l2cap_socket_id); bta_jv_pm_conn_idle(p_cb->p_pm_cb); break; case GAP_EVT_TX_EMPTY: bta_jv_pm_conn_idle(p_cb->p_pm_cb); break; case GAP_EVT_CONN_CONGESTED: case GAP_EVT_CONN_UNCONGESTED: p_cb->cong = (event == GAP_EVT_CONN_CONGESTED) ? true : false; evt_data.l2c_cong.cong = p_cb->cong; p_cb->p_cback(BTA_JV_L2CAP_CONG_EVT, &evt_data, p_cb->l2cap_socket_id); break; default: break; } } /** starts an L2CAP server */ void bta_jv_l2cap_start_server(int32_t type, tBTA_SEC sec_mask, tBTA_JV_ROLE role, uint16_t local_psm, uint16_t rx_mtu, std::unique_ptr cfg_param, std::unique_ptr ertm_info, tBTA_JV_L2CAP_CBACK* p_cback, uint32_t l2cap_socket_id) { uint16_t handle; tBTA_JV_L2CAP_START evt_data; uint8_t chan_mode_mask = GAP_FCR_CHAN_OPT_BASIC; tL2CAP_CFG_INFO cfg; memset(&cfg, 0, sizeof(tL2CAP_CFG_INFO)); if (cfg_param) { cfg = *cfg_param; if (cfg.fcr_present && cfg.fcr.mode == L2CAP_FCR_ERTM_MODE) { chan_mode_mask = GAP_FCR_CHAN_OPT_ERTM; } } // FIX: MTU=0 means not present if (rx_mtu > 0) { cfg.mtu_present = true; cfg.mtu = rx_mtu; } else { cfg.mtu_present = false; cfg.mtu = 0; } /* TODO DM role manager L2CA_SetDesireRole(role); */ uint8_t sec_id = bta_jv_alloc_sec_id(); uint16_t max_mps = 0xffff; // Let GAP_ConnOpen set the max_mps. /* PSM checking is not required for LE COC */ if (0 == sec_id || ((type == BTA_JV_CONN_TYPE_L2CAP) && (!bta_jv_check_psm(local_psm))) || (handle = GAP_ConnOpen("JV L2CAP", sec_id, 1, nullptr, local_psm, max_mps, &cfg, ertm_info.get(), sec_mask, chan_mode_mask, bta_jv_l2cap_server_cback, type)) == GAP_INVALID_HANDLE) { bta_jv_free_sec_id(&sec_id); evt_data.status = BTA_JV_FAILURE; } else { tBTA_JV_L2C_CB* p_cb = &bta_jv_cb.l2c_cb[handle]; evt_data.status = BTA_JV_SUCCESS; evt_data.handle = handle; evt_data.sec_id = sec_id; p_cb->p_cback = p_cback; p_cb->l2cap_socket_id = l2cap_socket_id; p_cb->handle = handle; p_cb->sec_id = sec_id; p_cb->state = BTA_JV_ST_SR_LISTEN; p_cb->psm = local_psm; } if (p_cback) { tBTA_JV bta_jv; bta_jv.l2c_start = evt_data; p_cback(BTA_JV_L2CAP_START_EVT, &bta_jv, l2cap_socket_id); } } /* stops an L2CAP server */ void bta_jv_l2cap_stop_server(uint16_t local_psm, uint32_t l2cap_socket_id) { for (int i = 0; i < BTA_JV_MAX_L2C_CONN; i++) { if (bta_jv_cb.l2c_cb[i].l2cap_socket_id == l2cap_socket_id) { tBTA_JV_L2C_CB* p_cb = &bta_jv_cb.l2c_cb[i]; tBTA_JV_L2CAP_CBACK* p_cback = p_cb->p_cback; tBTA_JV_L2CAP_CLOSE evt_data; evt_data.handle = p_cb->handle; evt_data.status = bta_jv_free_l2c_cb(p_cb); evt_data.async = false; if (p_cback) { tBTA_JV bta_jv; bta_jv.l2c_close = evt_data; p_cback(BTA_JV_L2CAP_CLOSE_EVT, &bta_jv, l2cap_socket_id); } break; } } } /* Write data to an L2CAP connection */ void bta_jv_l2cap_write(uint32_t handle, uint32_t req_id, BT_HDR* msg, uint32_t user_id, tBTA_JV_L2C_CB* p_cb) { /* As we check this callback exists before the tBTA_JV_API_L2CAP_WRITE can be * send through the API this check should not be needed. But the API is not * designed to be used (safely at least) in a multi-threaded scheduler, hence * if the peer device disconnects the l2cap link after the API is called, but * before this message is handled, the ->p_cback will be cleared at this * point. At first glanch this seems highly unlikely, but for all * obex-profiles with two channels connected - e.g. MAP, this happens around 1 * of 4 disconnects, as a disconnect on the server channel causes a disconnect * to be send on the client (notification) channel, but at the peer typically * disconnects both the OBEX disconnect request crosses the incoming l2cap * disconnect. If p_cback is cleared, we simply discard the data. RISK: The * caller must handle any cleanup based on another signal than * BTA_JV_L2CAP_WRITE_EVT, which is typically not possible, as the pointer to * the allocated buffer is stored in this message, and can therefore not be * freed, hence we have a mem-leak-by-design.*/ if (!p_cb->p_cback) { /* As this pointer is checked in the API function, this occurs only when the * channel is disconnected after the API function is called, but before the * message is handled. */ LOG(ERROR) << __func__ << ": p_cb->p_cback == NULL"; osi_free(msg); return; } tBTA_JV_L2CAP_WRITE evt_data; evt_data.status = BTA_JV_FAILURE; evt_data.handle = handle; evt_data.req_id = req_id; evt_data.cong = p_cb->cong; evt_data.len = msg->len; bta_jv_pm_conn_busy(p_cb->p_pm_cb); // TODO: this was set only for non-fixed channel packets. Is that needed ? msg->event = BT_EVT_TO_BTU_SP_DATA; if (evt_data.cong) { osi_free(msg); } else { if (GAP_ConnWriteData(handle, msg) == BT_PASS) evt_data.status = BTA_JV_SUCCESS; } tBTA_JV bta_jv; bta_jv.l2c_write = evt_data; p_cb->p_cback(BTA_JV_L2CAP_WRITE_EVT, &bta_jv, user_id); } /* Write data to an L2CAP connection using Fixed channels */ void bta_jv_l2cap_write_fixed(uint16_t channel, const RawAddress& addr, uint32_t req_id, BT_HDR* msg, uint32_t user_id, tBTA_JV_L2CAP_CBACK* p_cback) { tBTA_JV_L2CAP_WRITE_FIXED evt_data; evt_data.status = BTA_JV_FAILURE; evt_data.channel = channel; evt_data.addr = addr; evt_data.req_id = req_id; evt_data.len = 0; L2CA_SendFixedChnlData(channel, addr, msg); tBTA_JV bta_jv; bta_jv.l2c_write_fixed = evt_data; p_cback(BTA_JV_L2CAP_WRITE_FIXED_EVT, &bta_jv, user_id); } /******************************************************************************* * * Function bta_jv_port_data_co_cback * * Description port data callback function of rfcomm * connections * * Returns void * ******************************************************************************/ static int bta_jv_port_data_co_cback(uint16_t port_handle, uint8_t* buf, uint16_t len, int type) { tBTA_JV_RFC_CB* p_cb = bta_jv_rfc_port_to_cb(port_handle); tBTA_JV_PCB* p_pcb = bta_jv_rfc_port_to_pcb(port_handle); VLOG(2) << __func__ << ": p_cb=" << p_cb << ", p_pcb=" << p_pcb << ", len=" << len << ", type=" << type; if (p_pcb != NULL) { switch (type) { case DATA_CO_CALLBACK_TYPE_INCOMING: return bta_co_rfc_data_incoming(p_pcb->rfcomm_slot_id, (BT_HDR*)buf); case DATA_CO_CALLBACK_TYPE_OUTGOING_SIZE: return bta_co_rfc_data_outgoing_size(p_pcb->rfcomm_slot_id, (int*)buf); case DATA_CO_CALLBACK_TYPE_OUTGOING: return bta_co_rfc_data_outgoing(p_pcb->rfcomm_slot_id, buf, len); default: LOG(ERROR) << __func__ << ": unknown callout type=" << type; break; } } return 0; } /******************************************************************************* * * Function bta_jv_port_mgmt_cl_cback * * Description callback for port mamangement function of rfcomm * client connections * * Returns void * ******************************************************************************/ static void bta_jv_port_mgmt_cl_cback(uint32_t code, uint16_t port_handle) { tBTA_JV_RFC_CB* p_cb = bta_jv_rfc_port_to_cb(port_handle); tBTA_JV_PCB* p_pcb = bta_jv_rfc_port_to_pcb(port_handle); tBTA_JV evt_data; RawAddress rem_bda = RawAddress::kEmpty; uint16_t lcid; tBTA_JV_RFCOMM_CBACK* p_cback; /* the callback function */ VLOG(2) << __func__ << ": code=" << code << ", port_handle=" << port_handle; if (NULL == p_cb || NULL == p_cb->p_cback) return; VLOG(2) << __func__ << ": code=" << code << ", port_handle=" << port_handle << ", handle=" << p_cb->handle; PORT_CheckConnection(port_handle, &rem_bda, &lcid); if (code == PORT_SUCCESS) { evt_data.rfc_open.handle = p_cb->handle; evt_data.rfc_open.status = BTA_JV_SUCCESS; evt_data.rfc_open.rem_bda = rem_bda; p_pcb->state = BTA_JV_ST_CL_OPEN; p_cb->p_cback(BTA_JV_RFCOMM_OPEN_EVT, &evt_data, p_pcb->rfcomm_slot_id); } else { evt_data.rfc_close.handle = p_cb->handle; evt_data.rfc_close.status = BTA_JV_FAILURE; evt_data.rfc_close.port_status = code; evt_data.rfc_close.async = true; if (p_pcb->state == BTA_JV_ST_CL_CLOSING) { evt_data.rfc_close.async = false; } // p_pcb->state = BTA_JV_ST_NONE; // p_pcb->cong = false; p_cback = p_cb->p_cback; p_cback(BTA_JV_RFCOMM_CLOSE_EVT, &evt_data, p_pcb->rfcomm_slot_id); // bta_jv_free_rfc_cb(p_cb, p_pcb); } } /******************************************************************************* * * Function bta_jv_port_event_cl_cback * * Description Callback for RFCOMM client port events * * Returns void * ******************************************************************************/ static void bta_jv_port_event_cl_cback(uint32_t code, uint16_t port_handle) { tBTA_JV_RFC_CB* p_cb = bta_jv_rfc_port_to_cb(port_handle); tBTA_JV_PCB* p_pcb = bta_jv_rfc_port_to_pcb(port_handle); tBTA_JV evt_data; VLOG(2) << __func__ << ": port_handle=" << port_handle; if (NULL == p_cb || NULL == p_cb->p_cback) return; VLOG(2) << __func__ << ": code=" << loghex(code) << ", port_handle=" << port_handle << ", handle=" << p_cb->handle; if (code & PORT_EV_RXCHAR) { evt_data.data_ind.handle = p_cb->handle; p_cb->p_cback(BTA_JV_RFCOMM_DATA_IND_EVT, &evt_data, p_pcb->rfcomm_slot_id); } if (code & PORT_EV_FC) { p_pcb->cong = (code & PORT_EV_FCS) ? false : true; evt_data.rfc_cong.cong = p_pcb->cong; evt_data.rfc_cong.handle = p_cb->handle; evt_data.rfc_cong.status = BTA_JV_SUCCESS; p_cb->p_cback(BTA_JV_RFCOMM_CONG_EVT, &evt_data, p_pcb->rfcomm_slot_id); } if (code & PORT_EV_TXEMPTY) { bta_jv_pm_conn_idle(p_pcb->p_pm_cb); } } /* Client initiates an RFCOMM connection */ void bta_jv_rfcomm_connect(tBTA_SEC sec_mask, tBTA_JV_ROLE role, uint8_t remote_scn, const RawAddress& peer_bd_addr, tBTA_JV_RFCOMM_CBACK* p_cback, uint32_t rfcomm_slot_id) { uint16_t handle = 0; uint32_t event_mask = BTA_JV_RFC_EV_MASK; tPORT_STATE port_state; /* TODO DM role manager L2CA_SetDesireRole(role); */ uint8_t sec_id = bta_jv_alloc_sec_id(); tBTA_JV_RFCOMM_CL_INIT evt_data; memset(&evt_data, 0, sizeof(evt_data)); evt_data.sec_id = sec_id; evt_data.status = BTA_JV_SUCCESS; if (0 == sec_id || !BTM_SetSecurityLevel(true, "", sec_id, sec_mask, BT_PSM_RFCOMM, BTM_SEC_PROTO_RFCOMM, remote_scn)) { evt_data.status = BTA_JV_FAILURE; LOG(ERROR) << __func__ << ": sec_id=" << +sec_id << " is zero or BTM_SetSecurityLevel failed, remote_scn:" << +remote_scn; } if (evt_data.status == BTA_JV_SUCCESS && RFCOMM_CreateConnection(UUID_SERVCLASS_SERIAL_PORT, remote_scn, false, BTA_JV_DEF_RFC_MTU, peer_bd_addr, &handle, bta_jv_port_mgmt_cl_cback) != PORT_SUCCESS) { LOG(ERROR) << __func__ << ": RFCOMM_CreateConnection failed"; evt_data.status = BTA_JV_FAILURE; } if (evt_data.status == BTA_JV_SUCCESS) { tBTA_JV_PCB* p_pcb; tBTA_JV_RFC_CB* p_cb = bta_jv_alloc_rfc_cb(handle, &p_pcb); if (p_cb) { p_cb->p_cback = p_cback; p_cb->sec_id = sec_id; p_cb->scn = 0; p_pcb->state = BTA_JV_ST_CL_OPENING; p_pcb->rfcomm_slot_id = rfcomm_slot_id; evt_data.use_co = true; PORT_SetEventCallback(handle, bta_jv_port_event_cl_cback); PORT_SetEventMask(handle, event_mask); PORT_SetDataCOCallback(handle, bta_jv_port_data_co_cback); PORT_GetState(handle, &port_state); port_state.fc_type = (PORT_FC_CTS_ON_INPUT | PORT_FC_CTS_ON_OUTPUT); PORT_SetState(handle, &port_state); evt_data.handle = p_cb->handle; } else { evt_data.status = BTA_JV_FAILURE; LOG(ERROR) << __func__ << ": run out of rfc control block"; } } tBTA_JV bta_jv; bta_jv.rfc_cl_init = evt_data; p_cback(BTA_JV_RFCOMM_CL_INIT_EVT, &bta_jv, rfcomm_slot_id); if (bta_jv.rfc_cl_init.status == BTA_JV_FAILURE) { if (sec_id) bta_jv_free_sec_id(&sec_id); if (handle) RFCOMM_RemoveConnection(handle); } } static int find_rfc_pcb(uint32_t rfcomm_slot_id, tBTA_JV_RFC_CB** cb, tBTA_JV_PCB** pcb) { *cb = NULL; *pcb = NULL; int i; for (i = 0; i < MAX_RFC_PORTS; i++) { uint32_t rfc_handle = bta_jv_cb.port_cb[i].handle & BTA_JV_RFC_HDL_MASK; rfc_handle &= ~BTA_JV_RFCOMM_MASK; if (rfc_handle && bta_jv_cb.port_cb[i].rfcomm_slot_id == rfcomm_slot_id) { *pcb = &bta_jv_cb.port_cb[i]; *cb = &bta_jv_cb.rfc_cb[rfc_handle - 1]; VLOG(2) << __func__ << ": FOUND rfc_cb_handle=" << loghex(rfc_handle) << ", port.jv_handle=" << loghex((*pcb)->handle) << ", state=" << (*pcb)->state << ", rfc_cb->handle=" << loghex((*cb)->handle); return 1; } } VLOG(2) << __func__ << ": cannot find rfc_cb from user data:" << rfcomm_slot_id; return 0; } /* Close an RFCOMM connection */ void bta_jv_rfcomm_close(uint32_t handle, uint32_t rfcomm_slot_id) { if (!handle) { LOG(ERROR) << __func__ << ": rfc handle is null"; return; } VLOG(2) << __func__ << ": rfc handle=" << handle; tBTA_JV_RFC_CB* p_cb = NULL; tBTA_JV_PCB* p_pcb = NULL; if (!find_rfc_pcb(rfcomm_slot_id, &p_cb, &p_pcb)) return; bta_jv_free_rfc_cb(p_cb, p_pcb); VLOG(2) << __func__ << ": sec id in use=" << get_sec_id_used() << ", rfc_cb in use=" << get_rfc_cb_used(); } /******************************************************************************* * * Function bta_jv_port_mgmt_sr_cback * * Description callback for port mamangement function of rfcomm * server connections * * Returns void * ******************************************************************************/ static void bta_jv_port_mgmt_sr_cback(uint32_t code, uint16_t port_handle) { tBTA_JV_PCB* p_pcb = bta_jv_rfc_port_to_pcb(port_handle); tBTA_JV_RFC_CB* p_cb = bta_jv_rfc_port_to_cb(port_handle); tBTA_JV evt_data; RawAddress rem_bda = RawAddress::kEmpty; uint16_t lcid; VLOG(2) << __func__ << ": code=" << code << ", port_handle=" << port_handle; if (NULL == p_cb || NULL == p_cb->p_cback) { LOG(ERROR) << __func__ << ": p_cb=" << p_cb << ", p_cb->p_cback=" << (p_cb ? p_cb->p_cback : 0); return; } uint32_t rfcomm_slot_id = p_pcb->rfcomm_slot_id; VLOG(2) << __func__ << ": code=" << code << ", port_handle=" << loghex(port_handle) << ", handle=" << loghex(p_cb->handle) << ", p_pcb" << p_pcb << ", user=" << p_pcb->rfcomm_slot_id; int status = PORT_CheckConnection(port_handle, &rem_bda, &lcid); int failed = true; if (code == PORT_SUCCESS) { if (status != PORT_SUCCESS) { LOG(ERROR) << __func__ << ": PORT_CheckConnection returned " << status << ", although port is supposed to be connected"; } evt_data.rfc_srv_open.handle = p_pcb->handle; evt_data.rfc_srv_open.status = BTA_JV_SUCCESS; evt_data.rfc_srv_open.rem_bda = rem_bda; tBTA_JV_PCB* p_pcb_new_listen = bta_jv_add_rfc_port(p_cb, p_pcb); if (p_pcb_new_listen) { evt_data.rfc_srv_open.new_listen_handle = p_pcb_new_listen->handle; p_pcb_new_listen->rfcomm_slot_id = p_cb->p_cback(BTA_JV_RFCOMM_SRV_OPEN_EVT, &evt_data, rfcomm_slot_id); VLOG(2) << __func__ << ": curr_sess=" << p_cb->curr_sess << ", max_sess=" << p_cb->max_sess; failed = false; } else LOG(ERROR) << __func__ << ": failed to create new listen port"; } if (failed) { evt_data.rfc_close.handle = p_cb->handle; evt_data.rfc_close.status = BTA_JV_FAILURE; evt_data.rfc_close.async = true; evt_data.rfc_close.port_status = code; p_pcb->cong = false; tBTA_JV_RFCOMM_CBACK* p_cback = p_cb->p_cback; VLOG(2) << __func__ << ": PORT_CLOSED before BTA_JV_RFCOMM_CLOSE_EVT: curr_sess=" << p_cb->curr_sess << ", max_sess=" << p_cb->max_sess; if (BTA_JV_ST_SR_CLOSING == p_pcb->state) { evt_data.rfc_close.async = false; evt_data.rfc_close.status = BTA_JV_SUCCESS; } // p_pcb->state = BTA_JV_ST_NONE; p_cback(BTA_JV_RFCOMM_CLOSE_EVT, &evt_data, rfcomm_slot_id); // bta_jv_free_rfc_cb(p_cb, p_pcb); VLOG(2) << __func__ << ": PORT_CLOSED after BTA_JV_RFCOMM_CLOSE_EVT: curr_sess=" << p_cb->curr_sess << ", max_sess=" << p_cb->max_sess; } } /******************************************************************************* * * Function bta_jv_port_event_sr_cback * * Description Callback for RFCOMM server port events * * Returns void * ******************************************************************************/ static void bta_jv_port_event_sr_cback(uint32_t code, uint16_t port_handle) { tBTA_JV_PCB* p_pcb = bta_jv_rfc_port_to_pcb(port_handle); tBTA_JV_RFC_CB* p_cb = bta_jv_rfc_port_to_cb(port_handle); tBTA_JV evt_data; if (NULL == p_cb || NULL == p_cb->p_cback) { LOG(ERROR) << __func__ << ": p_cb=" << p_cb << ", p_cb->p_cback=" << (p_cb ? p_cb->p_cback : 0); return; } VLOG(2) << __func__ << ": code=" << loghex(code) << ", port_handle=" << port_handle << ", handle=" << p_cb->handle; uint32_t user_data = p_pcb->rfcomm_slot_id; if (code & PORT_EV_RXCHAR) { evt_data.data_ind.handle = p_cb->handle; p_cb->p_cback(BTA_JV_RFCOMM_DATA_IND_EVT, &evt_data, user_data); } if (code & PORT_EV_FC) { p_pcb->cong = (code & PORT_EV_FCS) ? false : true; evt_data.rfc_cong.cong = p_pcb->cong; evt_data.rfc_cong.handle = p_cb->handle; evt_data.rfc_cong.status = BTA_JV_SUCCESS; p_cb->p_cback(BTA_JV_RFCOMM_CONG_EVT, &evt_data, user_data); } if (code & PORT_EV_TXEMPTY) { bta_jv_pm_conn_idle(p_pcb->p_pm_cb); } } /******************************************************************************* * * Function bta_jv_add_rfc_port * * Description add a port for server when the existing posts is open * * Returns return a pointer to tBTA_JV_PCB just added * ******************************************************************************/ static tBTA_JV_PCB* bta_jv_add_rfc_port(tBTA_JV_RFC_CB* p_cb, tBTA_JV_PCB* p_pcb_open) { uint8_t used = 0, i, listen = 0; uint32_t si = 0; tPORT_STATE port_state; uint32_t event_mask = BTA_JV_RFC_EV_MASK; tBTA_JV_PCB* p_pcb = NULL; if (p_cb->max_sess > 1) { for (i = 0; i < p_cb->max_sess; i++) { if (p_cb->rfc_hdl[i] != 0) { p_pcb = &bta_jv_cb.port_cb[p_cb->rfc_hdl[i] - 1]; if (p_pcb->state == BTA_JV_ST_SR_LISTEN) { listen++; if (p_pcb_open == p_pcb) { VLOG(2) << __func__ << ": port_handle=" << p_pcb->port_handle << ", change the listen port to open state"; p_pcb->state = BTA_JV_ST_SR_OPEN; } else { LOG(ERROR) << __func__ << ": open pcb not matching listen one, count=" << listen << ", listen pcb handle=" << p_pcb->port_handle << ", open pcb=" << p_pcb_open->handle; return NULL; } } used++; } else if (si == 0) { si = i + 1; } } VLOG(2) << __func__ << ": max_sess=" << p_cb->max_sess << ", used=" << used << ", curr_sess=" << p_cb->curr_sess << ", listen=" << listen << ", si=" << si; if (used < p_cb->max_sess && listen == 1 && si) { si--; if (RFCOMM_CreateConnection(p_cb->sec_id, p_cb->scn, true, BTA_JV_DEF_RFC_MTU, RawAddress::kAny, &(p_cb->rfc_hdl[si]), bta_jv_port_mgmt_sr_cback) == PORT_SUCCESS) { p_cb->curr_sess++; p_pcb = &bta_jv_cb.port_cb[p_cb->rfc_hdl[si] - 1]; p_pcb->state = BTA_JV_ST_SR_LISTEN; p_pcb->port_handle = p_cb->rfc_hdl[si]; p_pcb->rfcomm_slot_id = p_pcb_open->rfcomm_slot_id; PORT_ClearKeepHandleFlag(p_pcb->port_handle); PORT_SetEventCallback(p_pcb->port_handle, bta_jv_port_event_sr_cback); PORT_SetDataCOCallback(p_pcb->port_handle, bta_jv_port_data_co_cback); PORT_SetEventMask(p_pcb->port_handle, event_mask); PORT_GetState(p_pcb->port_handle, &port_state); port_state.fc_type = (PORT_FC_CTS_ON_INPUT | PORT_FC_CTS_ON_OUTPUT); PORT_SetState(p_pcb->port_handle, &port_state); p_pcb->handle = BTA_JV_RFC_H_S_TO_HDL(p_cb->handle, si); VLOG(2) << __func__ << ": p_pcb->handle=" << loghex(p_pcb->handle) << ", curr_sess=" << p_cb->curr_sess; } } else { LOG(ERROR) << __func__ << ": cannot create new rfc listen port"; return NULL; } } VLOG(2) << __func__ << ": sec id in use=" << get_sec_id_used() << ", rfc_cb in use=" << get_rfc_cb_used(); return p_pcb; } /* waits for an RFCOMM client to connect */ void bta_jv_rfcomm_start_server(tBTA_SEC sec_mask, tBTA_JV_ROLE role, uint8_t local_scn, uint8_t max_session, tBTA_JV_RFCOMM_CBACK* p_cback, uint32_t rfcomm_slot_id) { uint16_t handle = 0; uint32_t event_mask = BTA_JV_RFC_EV_MASK; tPORT_STATE port_state; uint8_t sec_id = 0; tBTA_JV_RFC_CB* p_cb = NULL; tBTA_JV_PCB* p_pcb; tBTA_JV_RFCOMM_START evt_data; /* TODO DM role manager L2CA_SetDesireRole(role); */ memset(&evt_data, 0, sizeof(evt_data)); evt_data.status = BTA_JV_FAILURE; VLOG(2) << __func__ << ": sec id in use=" << get_sec_id_used() << ", rfc_cb in use=" << get_rfc_cb_used(); do { sec_id = bta_jv_alloc_sec_id(); if (0 == sec_id || !BTM_SetSecurityLevel(false, "JV PORT", sec_id, sec_mask, BT_PSM_RFCOMM, BTM_SEC_PROTO_RFCOMM, local_scn)) { LOG(ERROR) << __func__ << ": run out of sec_id"; break; } if (RFCOMM_CreateConnection(sec_id, local_scn, true, BTA_JV_DEF_RFC_MTU, RawAddress::kAny, &handle, bta_jv_port_mgmt_sr_cback) != PORT_SUCCESS) { LOG(ERROR) << __func__ << ": RFCOMM_CreateConnection failed"; break; } p_cb = bta_jv_alloc_rfc_cb(handle, &p_pcb); if (!p_cb) { LOG(ERROR) << __func__ << ": run out of rfc control block"; break; } p_cb->max_sess = max_session; p_cb->p_cback = p_cback; p_cb->sec_id = sec_id; p_cb->scn = local_scn; p_pcb->state = BTA_JV_ST_SR_LISTEN; p_pcb->rfcomm_slot_id = rfcomm_slot_id; evt_data.status = BTA_JV_SUCCESS; evt_data.handle = p_cb->handle; evt_data.sec_id = sec_id; evt_data.use_co = true; PORT_ClearKeepHandleFlag(handle); PORT_SetEventCallback(handle, bta_jv_port_event_sr_cback); PORT_SetEventMask(handle, event_mask); PORT_GetState(handle, &port_state); port_state.fc_type = (PORT_FC_CTS_ON_INPUT | PORT_FC_CTS_ON_OUTPUT); PORT_SetState(handle, &port_state); } while (0); tBTA_JV bta_jv; bta_jv.rfc_start = evt_data; p_cback(BTA_JV_RFCOMM_START_EVT, &bta_jv, rfcomm_slot_id); if (bta_jv.rfc_start.status == BTA_JV_SUCCESS) { PORT_SetDataCOCallback(handle, bta_jv_port_data_co_cback); } else { if (sec_id) bta_jv_free_sec_id(&sec_id); if (handle) RFCOMM_RemoveConnection(handle); } } /* stops an RFCOMM server */ void bta_jv_rfcomm_stop_server(uint32_t handle, uint32_t rfcomm_slot_id) { if (!handle) { LOG(ERROR) << __func__ << ": jv handle is null"; return; } VLOG(2) << __func__; tBTA_JV_RFC_CB* p_cb = NULL; tBTA_JV_PCB* p_pcb = NULL; if (!find_rfc_pcb(rfcomm_slot_id, &p_cb, &p_pcb)) return; VLOG(2) << __func__ << ": p_pcb=" << p_pcb << ", p_pcb->port_handle=" << p_pcb->port_handle; bta_jv_free_rfc_cb(p_cb, p_pcb); VLOG(2) << __func__ << ": sec id in use=" << get_sec_id_used() << ", rfc_cb in use=" << get_rfc_cb_used(); } /* write data to an RFCOMM connection */ void bta_jv_rfcomm_write(uint32_t handle, uint32_t req_id, tBTA_JV_RFC_CB* p_cb, tBTA_JV_PCB* p_pcb) { if (p_pcb->state == BTA_JV_ST_NONE) { LOG(ERROR) << __func__ << ": in state BTA_JV_ST_NONE - cannot write"; return; } tBTA_JV_RFCOMM_WRITE evt_data; evt_data.status = BTA_JV_FAILURE; evt_data.handle = handle; evt_data.req_id = req_id; evt_data.cong = p_pcb->cong; evt_data.len = 0; bta_jv_pm_conn_busy(p_pcb->p_pm_cb); if (!evt_data.cong && PORT_WriteDataCO(p_pcb->port_handle, &evt_data.len) == PORT_SUCCESS) { evt_data.status = BTA_JV_SUCCESS; } // Update congestion flag evt_data.cong = p_pcb->cong; if (!p_cb->p_cback) { LOG(ERROR) << __func__ << ": No JV callback set"; return; } tBTA_JV bta_jv; bta_jv.rfc_write = evt_data; p_cb->p_cback(BTA_JV_RFCOMM_WRITE_EVT, &bta_jv, p_pcb->rfcomm_slot_id); } /* Set or free power mode profile for a JV application */ void bta_jv_set_pm_profile(uint32_t handle, tBTA_JV_PM_ID app_id, tBTA_JV_CONN_STATE init_st) { tBTA_JV_STATUS status; tBTA_JV_PM_CB* p_cb; VLOG(2) << __func__ << " handle=" << loghex(handle) << ", app_id=" << app_id << ", init_st=" << +init_st; /* clear PM control block */ if (app_id == BTA_JV_PM_ID_CLEAR) { status = bta_jv_free_set_pm_profile_cb(handle); if (status != BTA_JV_SUCCESS) { LOG(WARNING) << __func__ << ": free pm cb failed: reason=" << +status; } } else /* set PM control block */ { p_cb = bta_jv_alloc_set_pm_profile_cb(handle, app_id); if (NULL != p_cb) bta_jv_pm_state_change(p_cb, init_st); else LOG(WARNING) << __func__ << ": failed"; } } /******************************************************************************* * * Function bta_jv_pm_conn_busy * * Description set pm connection busy state (input param safe) * * Params p_cb: pm control block of jv connection * * Returns void * ******************************************************************************/ static void bta_jv_pm_conn_busy(tBTA_JV_PM_CB* p_cb) { if ((NULL != p_cb) && (BTA_JV_PM_IDLE_ST == p_cb->state)) bta_jv_pm_state_change(p_cb, BTA_JV_CONN_BUSY); } /******************************************************************************* * * Function bta_jv_pm_conn_busy * * Description set pm connection busy state (input param safe) * * Params p_cb: pm control block of jv connection * * Returns void * ******************************************************************************/ static void bta_jv_pm_conn_idle(tBTA_JV_PM_CB* p_cb) { if ((NULL != p_cb) && (BTA_JV_PM_IDLE_ST != p_cb->state)) bta_jv_pm_state_change(p_cb, BTA_JV_CONN_IDLE); } /******************************************************************************* * * Function bta_jv_pm_state_change * * Description Notify power manager there is state change * * Params p_cb: must be NONE NULL * * Returns void * ******************************************************************************/ static void bta_jv_pm_state_change(tBTA_JV_PM_CB* p_cb, const tBTA_JV_CONN_STATE state) { VLOG(2) << __func__ << ": p_cb=" << p_cb << ", handle=" << loghex(p_cb->handle) << ", busy/idle_state=" << p_cb->state << ", app_id=" << p_cb->app_id << ", conn_state=" << state; switch (state) { case BTA_JV_CONN_OPEN: bta_sys_conn_open(BTA_ID_JV, p_cb->app_id, p_cb->peer_bd_addr); break; case BTA_JV_CONN_CLOSE: bta_sys_conn_close(BTA_ID_JV, p_cb->app_id, p_cb->peer_bd_addr); break; case BTA_JV_APP_OPEN: bta_sys_app_open(BTA_ID_JV, p_cb->app_id, p_cb->peer_bd_addr); break; case BTA_JV_APP_CLOSE: bta_sys_app_close(BTA_ID_JV, p_cb->app_id, p_cb->peer_bd_addr); break; case BTA_JV_SCO_OPEN: bta_sys_sco_open(BTA_ID_JV, p_cb->app_id, p_cb->peer_bd_addr); break; case BTA_JV_SCO_CLOSE: bta_sys_sco_close(BTA_ID_JV, p_cb->app_id, p_cb->peer_bd_addr); break; case BTA_JV_CONN_IDLE: p_cb->state = BTA_JV_PM_IDLE_ST; bta_sys_idle(BTA_ID_JV, p_cb->app_id, p_cb->peer_bd_addr); break; case BTA_JV_CONN_BUSY: p_cb->state = BTA_JV_PM_BUSY_ST; bta_sys_busy(BTA_ID_JV, p_cb->app_id, p_cb->peer_bd_addr); break; default: LOG(WARNING) << __func__ << ": Invalid state=" << +state; break; } } /******************************************************************************/ static struct fc_channel* fcchan_get(uint16_t chan, char create) { struct fc_channel* t = fc_channels; static tL2CAP_FIXED_CHNL_REG fcr = { .pL2CA_FixedConn_Cb = fcchan_conn_chng_cbk, .pL2CA_FixedData_Cb = fcchan_data_cbk, .default_idle_tout = 0xffff, .fixed_chnl_opts = { .mode = L2CAP_FCR_BASIC_MODE, .max_transmit = 0xFF, .rtrans_tout = 2000, .mon_tout = 12000, .mps = 670, .tx_win_sz = 1, }, }; while (t && t->chan != chan) t = t->next; if (t) return t; else if (!create) return NULL; /* we cannot alloc a struct if not asked to */ t = static_cast(osi_calloc(sizeof(*t))); t->chan = chan; if (!L2CA_RegisterFixedChannel(chan, &fcr)) { osi_free(t); return NULL; } // link it in t->next = fc_channels; fc_channels = t; return t; } /* pass NULL to find servers */ static struct fc_client* fcclient_find_by_addr(struct fc_client* start, const RawAddress* addr) { struct fc_client* t = start; while (t) { /* match client if have addr */ if (addr && addr == &t->remote_addr) break; /* match server if do not have addr */ if (!addr && t->server) break; t = t->next_all_list; } return t; } static struct fc_client* fcclient_find_by_id(uint32_t id) { struct fc_client* t = fc_clients; while (t && t->id != id) t = t->next_all_list; return t; } static struct fc_client* fcclient_alloc(uint16_t chan, char server, const uint8_t* sec_id_to_use) { struct fc_channel* fc = fcchan_get(chan, true); struct fc_client* t; uint8_t sec_id; if (!fc) return NULL; if (fc->has_server && server) return NULL; /* no way to have multiple servers on same channel */ if (sec_id_to_use) sec_id = *sec_id_to_use; else sec_id = bta_jv_alloc_sec_id(); t = static_cast(osi_calloc(sizeof(*t))); // Allocate it a unique ID do { t->id = ++fc_next_id; } while (!t->id || fcclient_find_by_id(t->id)); // Populate some params t->chan = chan; t->server = server; // Get a security id t->sec_id = sec_id; // Link it in to global list t->next_all_list = fc_clients; fc_clients = t; // Link it in to channel list t->next_chan_list = fc->clients; fc->clients = t; // Update channel if needed if (server) fc->has_server = true; return t; } static void fcclient_free(struct fc_client* fc) { struct fc_client* t = fc_clients; struct fc_channel* tc = fcchan_get(fc->chan, false); // remove from global list while (t && t->next_all_list != fc) t = t->next_all_list; if (!t && fc != fc_clients) return; /* prevent double-free */ if (t) t->next_all_list = fc->next_all_list; else fc_clients = fc->next_all_list; // remove from channel list if (tc) { t = tc->clients; while (t && t->next_chan_list != fc) t = t->next_chan_list; if (t) t->next_chan_list = fc->next_chan_list; else tc->clients = fc->next_chan_list; // if was server then channel no longer has a server if (fc->server) tc->has_server = false; } // free security id bta_jv_free_sec_id(&fc->sec_id); osi_free(fc); } static void fcchan_conn_chng_cbk(uint16_t chan, const RawAddress& bd_addr, bool connected, uint16_t reason, tBT_TRANSPORT transport) { tBTA_JV init_evt; tBTA_JV open_evt; struct fc_channel* tc; struct fc_client *t = NULL, *new_conn; tBTA_JV_L2CAP_CBACK* p_cback = NULL; char call_init = false; uint32_t l2cap_socket_id; tc = fcchan_get(chan, false); if (tc) { t = fcclient_find_by_addr( tc->clients, &bd_addr); // try to find an open socked for that addr if (t) { p_cback = t->p_cback; l2cap_socket_id = t->l2cap_socket_id; } else { t = fcclient_find_by_addr( tc->clients, NULL); // try to find a listening socked for that channel if (t) { // found: create a normal connection socket and assign the connection to // it new_conn = fcclient_alloc(chan, false, &t->sec_id); if (new_conn) { new_conn->remote_addr = bd_addr; new_conn->p_cback = NULL; // for now new_conn->init_called = true; /*nop need to do it again */ p_cback = t->p_cback; l2cap_socket_id = t->l2cap_socket_id; t = new_conn; } } else { // drop it return; } } } if (t) { if (!t->init_called) { call_init = true; t->init_called = true; init_evt.l2c_cl_init.handle = t->id; init_evt.l2c_cl_init.status = BTA_JV_SUCCESS; init_evt.l2c_cl_init.sec_id = t->sec_id; } open_evt.l2c_open.handle = t->id; open_evt.l2c_open.tx_mtu = 23; /* 23, why not ?*/ memcpy(&open_evt.l2c_le_open.rem_bda, &t->remote_addr, sizeof(open_evt.l2c_le_open.rem_bda)); // TODO: (apanicke) Change the way these functions work so that casting // isn't needed open_evt.l2c_le_open.p_p_cback = (void**)&t->p_cback; open_evt.l2c_le_open.p_user_data = (void**)&t->l2cap_socket_id; open_evt.l2c_le_open.status = BTA_JV_SUCCESS; if (connected) { open_evt.l2c_open.status = BTA_JV_SUCCESS; } else { fcclient_free(t); open_evt.l2c_open.status = BTA_JV_FAILURE; } } if (call_init) p_cback(BTA_JV_L2CAP_CL_INIT_EVT, &init_evt, l2cap_socket_id); // call this with lock taken so socket does not disappear from under us */ if (p_cback) { p_cback(BTA_JV_L2CAP_OPEN_EVT, &open_evt, l2cap_socket_id); if (!t->p_cback) /* no callback set, means they do not want this one... */ fcclient_free(t); } } static void fcchan_data_cbk(uint16_t chan, const RawAddress& bd_addr, BT_HDR* p_buf) { tBTA_JV evt_data; struct fc_channel* tc; struct fc_client* t = NULL; tBTA_JV_L2CAP_CBACK* sock_cback = NULL; uint32_t sock_id; tc = fcchan_get(chan, false); if (tc) { // try to find an open socked for that addr and channel t = fcclient_find_by_addr(tc->clients, &bd_addr); } if (!t) { // no socket -> drop it return; } sock_cback = t->p_cback; sock_id = t->l2cap_socket_id; evt_data.le_data_ind.handle = t->id; evt_data.le_data_ind.p_buf = p_buf; if (sock_cback) sock_cback(BTA_JV_L2CAP_DATA_IND_EVT, &evt_data, sock_id); } /** makes an le l2cap client connection */ void bta_jv_l2cap_connect_le(uint16_t remote_chan, const RawAddress& peer_bd_addr, tBTA_JV_L2CAP_CBACK* p_cback, uint32_t l2cap_socket_id) { tBTA_JV evt; uint32_t id; char call_init_f = true; struct fc_client* t; evt.l2c_cl_init.handle = GAP_INVALID_HANDLE; evt.l2c_cl_init.status = BTA_JV_FAILURE; t = fcclient_alloc(remote_chan, false, NULL); if (!t) { p_cback(BTA_JV_L2CAP_CL_INIT_EVT, &evt, l2cap_socket_id); return; } t->p_cback = p_cback; t->l2cap_socket_id = l2cap_socket_id; t->remote_addr = peer_bd_addr; id = t->id; t->init_called = false; if (L2CA_ConnectFixedChnl(t->chan, t->remote_addr)) { evt.l2c_cl_init.status = BTA_JV_SUCCESS; evt.l2c_cl_init.handle = id; } // it could have been deleted/moved from under us, so re-find it */ t = fcclient_find_by_id(id); if (t) { if (evt.l2c_cl_init.status == BTA_JV_SUCCESS) { call_init_f = !t->init_called; } else { fcclient_free(t); t = NULL; } } if (call_init_f) p_cback(BTA_JV_L2CAP_CL_INIT_EVT, &evt, l2cap_socket_id); if (t) { t->init_called = true; } } /* stops an LE L2CAP server */ void bta_jv_l2cap_stop_server_le(uint16_t local_chan) { struct fc_client* fcclient; tBTA_JV evt; evt.l2c_close.status = BTA_JV_FAILURE; evt.l2c_close.async = false; evt.l2c_close.handle = GAP_INVALID_HANDLE; struct fc_channel* fcchan = fcchan_get(local_chan, false); if (fcchan) { while ((fcclient = fcchan->clients)) { tBTA_JV_L2CAP_CBACK* p_cback = fcclient->p_cback; uint32_t l2cap_socket_id = fcclient->l2cap_socket_id; evt.l2c_close.handle = fcclient->id; evt.l2c_close.status = BTA_JV_SUCCESS; evt.l2c_close.async = false; fcclient_free(fcclient); if (p_cback) p_cback(BTA_JV_L2CAP_CLOSE_EVT, &evt, l2cap_socket_id); } } } /** starts an LE L2CAP server */ void bta_jv_l2cap_start_server_le(uint16_t local_chan, tBTA_JV_L2CAP_CBACK* p_cback, uint32_t l2cap_socket_id) { tBTA_JV_L2CAP_START evt_data; evt_data.handle = GAP_INVALID_HANDLE; evt_data.status = BTA_JV_FAILURE; struct fc_client* t = fcclient_alloc(local_chan, true, NULL); if (!t) goto out; t->p_cback = p_cback; t->l2cap_socket_id = l2cap_socket_id; // if we got here, we're registered... evt_data.status = BTA_JV_SUCCESS; evt_data.handle = t->id; evt_data.sec_id = t->sec_id; out: tBTA_JV bta_jv; bta_jv.l2c_start = evt_data; p_cback(BTA_JV_L2CAP_START_EVT, &bta_jv, l2cap_socket_id); } /* close a fixed channel connection. calls no callbacks. idempotent */ extern void bta_jv_l2cap_close_fixed(uint32_t handle) { struct fc_client* t = fcclient_find_by_id(handle); if (t) fcclient_free(t); }