| /****************************************************************************** |
| * |
| * Copyright 2003-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 the HeaLth device profile (HL) action functions for |
| * the state machine. |
| * |
| ******************************************************************************/ |
| |
| #include <string.h> |
| |
| #include "bt_target.h" |
| #if (HL_INCLUDED == TRUE) |
| |
| #include "bt_common.h" |
| #include "bta_hl_api.h" |
| #include "bta_hl_int.h" |
| #include "bta_sys.h" |
| #include "mca_api.h" |
| #include "mca_defs.h" |
| #include "osi/include/osi.h" |
| #include "port_api.h" |
| #include "sdp_api.h" |
| #include "utl.h" |
| |
| using bluetooth::Uuid; |
| |
| /***************************************************************************** |
| * Local Function prototypes |
| ****************************************************************************/ |
| #if (BTA_HL_DEBUG == TRUE) |
| static const char* bta_hl_mcap_evt_code(uint8_t evt_code); |
| static const char* bta_hl_dch_oper_code(tBTA_HL_DCH_OPER oper_code); |
| static const char* bta_hl_cback_evt_code(uint8_t evt_code); |
| #endif |
| static void bta_hl_sdp_cback(uint8_t sdp_op, uint8_t app_idx, uint8_t mcl_idx, |
| uint8_t mdl_idx, uint16_t status); |
| static void bta_hl_sdp_cback0(uint16_t status); |
| static void bta_hl_sdp_cback1(uint16_t status); |
| static void bta_hl_sdp_cback2(uint16_t status); |
| static void bta_hl_sdp_cback3(uint16_t status); |
| static void bta_hl_sdp_cback4(uint16_t status); |
| static void bta_hl_sdp_cback5(uint16_t status); |
| static void bta_hl_sdp_cback6(uint16_t status); |
| |
| static tSDP_DISC_CMPL_CB* const bta_hl_sdp_cback_arr[] = { |
| bta_hl_sdp_cback0, bta_hl_sdp_cback1, bta_hl_sdp_cback2, bta_hl_sdp_cback3, |
| bta_hl_sdp_cback4, bta_hl_sdp_cback5, bta_hl_sdp_cback6}; |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_mca_cong_change |
| * |
| * Description Action routine for processing congestion change notification |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_mca_cong_change(uint8_t app_idx, uint8_t mcl_idx, |
| uint8_t mdl_idx, tBTA_HL_DATA* p_data) { |
| tBTA_HL_APP_CB* p_acb = BTA_HL_GET_APP_CB_PTR(app_idx); |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| tMCA_CONG_CHG* p_cong_chg = &p_data->mca_evt.mca_data.cong_chg; |
| tBTA_HL evt_data; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_mca_cong_change mdl_id=%d cong=%d", |
| p_cong_chg->mdl_id, p_cong_chg->cong); |
| #endif |
| evt_data.dch_cong_ind.cong = p_dcb->cong = p_cong_chg->cong; |
| evt_data.dch_cong_ind.mdl_handle = p_dcb->mdl_handle; |
| evt_data.dch_cong_ind.mcl_handle = p_mcb->mcl_handle; |
| evt_data.dch_cong_ind.app_handle = p_acb->app_handle; |
| |
| p_acb->p_cback(BTA_HL_CONG_CHG_IND_EVT, (tBTA_HL*)&evt_data); |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_echo_test |
| * |
| * Description Action routine for processing echo test request |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_echo_test(uint8_t app_idx, uint8_t mcl_idx, uint8_t mdl_idx, |
| UNUSED_ATTR tBTA_HL_DATA* p_data) { |
| tBTA_HL_APP_CB* p_acb = BTA_HL_GET_APP_CB_PTR(app_idx); |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_echo_test"); |
| #endif |
| |
| p_dcb->echo_oper = BTA_HL_ECHO_OP_CI_GET_ECHO_DATA; |
| p_dcb->cout_oper |= BTA_HL_CO_GET_ECHO_DATA_MASK; |
| |
| bta_hl_co_get_echo_data( |
| p_acb->app_id, p_mcb->mcl_handle, p_dcb->p_echo_tx_pkt->len, |
| BTA_HL_GET_BUF_PTR(p_dcb->p_echo_tx_pkt), BTA_HL_CI_GET_ECHO_DATA_EVT); |
| } |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_sdp_init |
| * |
| * Description Action routine for processing DCH SDP initiation |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_sdp_init(uint8_t app_idx, uint8_t mcl_idx, uint8_t mdl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_sdp_init"); |
| #endif |
| if (p_mcb->sdp_oper == BTA_HL_SDP_OP_NONE) { |
| p_mcb->sdp_mdl_idx = mdl_idx; |
| if (p_dcb->dch_oper == BTA_HL_DCH_OP_LOCAL_OPEN) { |
| p_mcb->sdp_oper = BTA_HL_SDP_OP_DCH_OPEN_INIT; |
| |
| } else { |
| p_mcb->sdp_oper = BTA_HL_SDP_OP_DCH_RECONNECT_INIT; |
| } |
| |
| if (bta_hl_init_sdp(p_mcb->sdp_oper, app_idx, mcl_idx, mdl_idx) != |
| BTA_HL_STATUS_OK) { |
| APPL_TRACE_ERROR("SDP INIT failed"); |
| p_mcb->sdp_oper = BTA_HL_SDP_OP_NONE; |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, BTA_HL_DCH_SDP_FAIL_EVT, |
| p_data); |
| } |
| } else { |
| APPL_TRACE_ERROR("SDP in use"); |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, BTA_HL_DCH_SDP_FAIL_EVT, |
| p_data); |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_close_echo_test |
| * |
| * Description Action routine for processing the closing of echo test |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_close_echo_test(uint8_t app_idx, uint8_t mcl_idx, |
| uint8_t mdl_idx, tBTA_HL_DATA* p_data) { |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_close_echo_test"); |
| #endif |
| |
| switch (p_dcb->echo_oper) { |
| case BTA_HL_ECHO_OP_DCH_CLOSE_CFM: |
| case BTA_HL_ECHO_OP_OPEN_IND: |
| case BTA_HL_ECHO_OP_ECHO_PKT: |
| p_dcb->dch_oper = BTA_HL_DCH_OP_LOCAL_CLOSE_ECHO_TEST; |
| break; |
| case BTA_HL_ECHO_OP_MDL_CREATE_CFM: |
| case BTA_HL_ECHO_OP_DCH_OPEN_CFM: |
| case BTA_HL_ECHO_OP_LOOP_BACK: |
| default: |
| break; |
| } |
| |
| if (MCA_CloseReq((tMCA_DL)p_dcb->mdl_handle) != MCA_SUCCESS) { |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, BTA_HL_DCH_CLOSE_CMPL_EVT, |
| p_data); |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_mca_rcv_data |
| * |
| * Description Action routine for processing the received data |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_mca_rcv_data(uint8_t app_idx, uint8_t mcl_idx, uint8_t mdl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_APP_CB* p_acb = BTA_HL_GET_APP_CB_PTR(app_idx); |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_mca_rcv_data"); |
| #endif |
| |
| if (p_dcb->local_mdep_id == BTA_HL_ECHO_TEST_MDEP_ID) { |
| switch (p_dcb->echo_oper) { |
| case BTA_HL_ECHO_OP_ECHO_PKT: |
| |
| if (MCA_WriteReq((tMCA_DL)p_dcb->mdl_handle, |
| p_data->mca_rcv_data_evt.p_pkt) != MCA_SUCCESS) { |
| osi_free_and_reset((void**)&p_data->mca_rcv_data_evt.p_pkt); |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, |
| BTA_HL_DCH_CLOSE_ECHO_TEST_EVT, p_data); |
| } |
| break; |
| case BTA_HL_ECHO_OP_LOOP_BACK: |
| |
| p_dcb->p_echo_rx_pkt = p_data->mca_rcv_data_evt.p_pkt; |
| p_dcb->echo_oper = BTA_HL_ECHO_OP_CI_PUT_ECHO_DATA; |
| p_dcb->cout_oper |= BTA_HL_CO_PUT_ECHO_DATA_MASK; |
| p_dcb->ci_put_echo_data_status = BTA_HL_STATUS_FAIL; |
| |
| bta_hl_co_put_echo_data(p_acb->app_id, p_mcb->mcl_handle, |
| p_dcb->p_echo_rx_pkt->len, |
| BTA_HL_GET_BUF_PTR(p_dcb->p_echo_rx_pkt), |
| BTA_HL_CI_PUT_ECHO_DATA_EVT); |
| break; |
| default: |
| APPL_TRACE_ERROR("Unknonw echo_oper=%d", p_dcb->echo_oper); |
| break; |
| } |
| |
| } else { |
| p_dcb->cout_oper |= BTA_HL_CO_PUT_RX_DATA_MASK; |
| p_dcb->p_rx_pkt = p_data->mca_rcv_data_evt.p_pkt; |
| |
| bta_hl_co_put_rx_data( |
| p_acb->app_id, p_dcb->mdl_handle, p_dcb->p_rx_pkt->len, |
| BTA_HL_GET_BUF_PTR(p_dcb->p_rx_pkt), BTA_HL_CI_PUT_RX_DATA_EVT); |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_ci_put_echo_data |
| * |
| * Description Action routine for processing the call-in of the |
| * put echo data event |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_ci_put_echo_data(uint8_t app_idx, uint8_t mcl_idx, |
| uint8_t mdl_idx, tBTA_HL_DATA* p_data) { |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_ci_put_echo_data"); |
| #endif |
| |
| p_dcb->cout_oper &= ~BTA_HL_CO_PUT_ECHO_DATA_MASK; |
| osi_free_and_reset((void**)&p_dcb->p_echo_rx_pkt); |
| p_dcb->ci_put_echo_data_status = p_data->ci_get_put_echo_data.status; |
| |
| p_dcb->echo_oper = BTA_HL_ECHO_OP_DCH_CLOSE_CFM; |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, |
| BTA_HL_DCH_CLOSE_ECHO_TEST_EVT, p_data); |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_ci_get_echo_data |
| * |
| * Description Action routine for processing the call-in of the |
| * get echo data event |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_ci_get_echo_data(uint8_t app_idx, uint8_t mcl_idx, |
| uint8_t mdl_idx, tBTA_HL_DATA* p_data) { |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| tBTA_HL_STATUS status; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_ci_get_echo_data"); |
| #endif |
| |
| p_dcb->cout_oper &= ~BTA_HL_CO_GET_ECHO_DATA_MASK; |
| |
| if (!p_dcb->abort_oper) { |
| status = p_data->ci_get_put_echo_data.status; |
| if (status == BTA_HL_STATUS_OK) { |
| p_dcb->echo_oper = BTA_HL_ECHO_OP_MDL_CREATE_CFM; |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, BTA_HL_DCH_OPEN_EVT, |
| p_data); |
| } else { |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, |
| BTA_HL_DCH_CLOSE_CMPL_EVT, p_data); |
| } |
| } else { |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, BTA_HL_DCH_CLOSE_CMPL_EVT, |
| p_data); |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_ci_put_rx_data |
| * |
| * Description Action routine for processing the call-in of the |
| * put rx data event |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_ci_put_rx_data(uint8_t app_idx, uint8_t mcl_idx, |
| uint8_t mdl_idx, tBTA_HL_DATA* p_data) { |
| tBTA_HL_APP_CB* p_acb = BTA_HL_GET_APP_CB_PTR(app_idx); |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| tBTA_HL evt_data; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_ci_put_rx_data"); |
| #endif |
| |
| p_dcb->cout_oper &= ~BTA_HL_CO_PUT_RX_DATA_MASK; |
| osi_free_and_reset((void**)&p_dcb->p_rx_pkt); |
| bta_hl_build_rcv_data_ind(&evt_data, p_acb->app_handle, p_mcb->mcl_handle, |
| p_dcb->mdl_handle); |
| p_acb->p_cback(BTA_HL_DCH_RCV_DATA_IND_EVT, (tBTA_HL*)&evt_data); |
| if (p_dcb->close_pending) { |
| if (!p_dcb->cout_oper) { |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, BTA_HL_DCH_CLOSE_EVT, |
| p_data); |
| } |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_ci_get_tx_data |
| * |
| * Description Action routine for processing the call-in of the |
| * get tx data event |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_ci_get_tx_data(uint8_t app_idx, uint8_t mcl_idx, |
| uint8_t mdl_idx, tBTA_HL_DATA* p_data) { |
| tBTA_HL_APP_CB* p_acb = BTA_HL_GET_APP_CB_PTR(app_idx); |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| tMCA_RESULT result; |
| tBTA_HL_STATUS status = BTA_HL_STATUS_OK; |
| bool free_buf = false; |
| bool close_dch = false; |
| tBTA_HL evt_data; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_ci_get_tx_data"); |
| #endif |
| |
| if (p_data != NULL) { |
| status = p_data->ci_get_put_data.status; |
| APPL_TRACE_WARNING("%s: status=%d", __func__, status); |
| } |
| |
| p_dcb->cout_oper &= ~BTA_HL_CO_GET_TX_DATA_MASK; |
| |
| if (p_dcb->close_pending) { |
| status = BTA_HL_STATUS_FAIL; |
| free_buf = true; |
| |
| if (!p_dcb->cout_oper) { |
| close_dch = true; |
| } |
| } else if (status == BTA_HL_STATUS_FAIL) { |
| free_buf = TRUE; |
| } else { |
| result = MCA_WriteReq((tMCA_DL)p_dcb->mdl_handle, p_dcb->p_tx_pkt); |
| if (result != MCA_SUCCESS) { |
| if (result == MCA_BUSY) { |
| status = BTA_HL_STATUS_DCH_BUSY; |
| } else { |
| status = BTA_HL_STATUS_FAIL; |
| } |
| free_buf = true; |
| } else { |
| p_dcb->p_tx_pkt = NULL; |
| } |
| } |
| |
| if (free_buf) osi_free_and_reset((void**)&p_dcb->p_tx_pkt); |
| |
| bta_hl_build_send_data_cfm(&evt_data, p_acb->app_handle, p_mcb->mcl_handle, |
| p_dcb->mdl_handle, status); |
| p_acb->p_cback(BTA_HL_DCH_SEND_DATA_CFM_EVT, (tBTA_HL*)&evt_data); |
| |
| if (close_dch) { |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, BTA_HL_DCH_CLOSE_EVT, |
| p_data); |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_send_data |
| * |
| * Description Action routine for processing api send data request |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_send_data(uint8_t app_idx, uint8_t mcl_idx, uint8_t mdl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_APP_CB* p_acb = BTA_HL_GET_APP_CB_PTR(app_idx); |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| tBTA_HL evt_data; |
| bool success = true; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_send_data"); |
| #endif |
| |
| if (!(p_dcb->cout_oper & BTA_HL_CO_GET_TX_DATA_MASK)) { |
| // p_dcb->chnl_cfg.fcs may be BTA_HL_MCA_USE_FCS (0x11) or BTA_HL_MCA_NO_FCS |
| // (0x10) or BTA_HL_DEFAULT_SOURCE_FCS (1) |
| bool fcs_use = (bool)(p_dcb->chnl_cfg.fcs & BTA_HL_MCA_FCS_USE_MASK); |
| p_dcb->p_tx_pkt = bta_hl_get_buf(p_data->api_send_data.pkt_size, fcs_use); |
| if (p_dcb->p_tx_pkt != NULL) { |
| bta_hl_co_get_tx_data( |
| p_acb->app_id, p_dcb->mdl_handle, p_data->api_send_data.pkt_size, |
| BTA_HL_GET_BUF_PTR(p_dcb->p_tx_pkt), BTA_HL_CI_GET_TX_DATA_EVT); |
| p_dcb->cout_oper |= BTA_HL_CO_GET_TX_DATA_MASK; |
| } else { |
| success = false; |
| } |
| } else { |
| success = false; |
| } |
| |
| if (!success) { |
| bta_hl_build_send_data_cfm(&evt_data, p_acb->app_handle, p_mcb->mcl_handle, |
| p_dcb->mdl_handle, BTA_HL_STATUS_FAIL); |
| p_acb->p_cback(BTA_HL_DCH_SEND_DATA_CFM_EVT, (tBTA_HL*)&evt_data); |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_close_cmpl |
| * |
| * Description Action routine for processing the close complete event |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_close_cmpl(uint8_t app_idx, uint8_t mcl_idx, uint8_t mdl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_APP_CB* p_acb = BTA_HL_GET_APP_CB_PTR(app_idx); |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| tBTA_HL evt_data; |
| tBTA_HL_EVT event = 0; |
| bool send_evt = true; |
| tBTA_HL_STATUS status; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_close_cmpl dch oper=%s", |
| bta_hl_dch_oper_code(p_dcb->dch_oper)); |
| #endif |
| |
| switch (p_dcb->dch_oper) { |
| case BTA_HL_DCH_OP_LOCAL_OPEN: |
| case BTA_HL_DCH_OP_LOCAL_RECONNECT: |
| |
| if (p_dcb->abort_oper & BTA_HL_ABORT_LOCAL_MASK) { |
| bta_hl_build_abort_cfm(&evt_data, p_acb->app_handle, p_mcb->mcl_handle, |
| BTA_HL_STATUS_OK); |
| event = BTA_HL_DCH_ABORT_CFM_EVT; |
| } else if (p_dcb->abort_oper & BTA_HL_ABORT_REMOTE_MASK) { |
| bta_hl_build_abort_ind(&evt_data, p_acb->app_handle, p_mcb->mcl_handle); |
| event = BTA_HL_DCH_ABORT_IND_EVT; |
| } else { |
| bta_hl_build_dch_open_cfm(&evt_data, p_acb->app_handle, |
| p_mcb->mcl_handle, BTA_HL_INVALID_MDL_HANDLE, |
| 0, 0, 0, 0, 0, BTA_HL_STATUS_FAIL); |
| event = BTA_HL_DCH_OPEN_CFM_EVT; |
| if (p_dcb->dch_oper == BTA_HL_DCH_OP_LOCAL_RECONNECT) { |
| event = BTA_HL_DCH_RECONNECT_CFM_EVT; |
| } |
| } |
| break; |
| |
| case BTA_HL_DCH_OP_LOCAL_CLOSE: |
| case BTA_HL_DCH_OP_REMOTE_DELETE: |
| case BTA_HL_DCH_OP_LOCAL_CLOSE_RECONNECT: |
| case BTA_HL_DCH_OP_NONE: |
| |
| bta_hl_build_dch_close_cfm(&evt_data, p_acb->app_handle, |
| p_mcb->mcl_handle, p_dcb->mdl_handle, |
| BTA_HL_STATUS_OK); |
| event = BTA_HL_DCH_CLOSE_CFM_EVT; |
| break; |
| |
| case BTA_HL_DCH_OP_REMOTE_CLOSE: |
| bta_hl_build_dch_close_ind(&evt_data, p_acb->app_handle, |
| p_mcb->mcl_handle, p_dcb->mdl_handle, |
| p_dcb->intentional_close); |
| event = BTA_HL_DCH_CLOSE_IND_EVT; |
| break; |
| |
| case BTA_HL_DCH_OP_REMOTE_OPEN: |
| |
| if (p_dcb->abort_oper & BTA_HL_ABORT_LOCAL_MASK) { |
| bta_hl_build_abort_cfm(&evt_data, p_acb->app_handle, p_mcb->mcl_handle, |
| BTA_HL_STATUS_OK); |
| event = BTA_HL_DCH_ABORT_CFM_EVT; |
| } else if (p_dcb->abort_oper & BTA_HL_ABORT_REMOTE_MASK) { |
| bta_hl_build_abort_ind(&evt_data, p_acb->app_handle, p_mcb->mcl_handle); |
| event = BTA_HL_DCH_ABORT_IND_EVT; |
| } else { |
| bta_hl_build_dch_close_ind(&evt_data, p_acb->app_handle, |
| p_mcb->mcl_handle, p_dcb->mdl_handle, |
| p_dcb->intentional_close); |
| event = BTA_HL_DCH_CLOSE_IND_EVT; |
| } |
| break; |
| |
| case BTA_HL_DCH_OP_LOCAL_CLOSE_ECHO_TEST: |
| /* this is normal echo test close */ |
| case BTA_HL_DCH_OP_REMOTE_CREATE: |
| case BTA_HL_DCH_OP_REMOTE_RECONNECT: |
| send_evt = false; |
| break; |
| |
| default: |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_ERROR("DCH operation not found oper=%s", |
| bta_hl_dch_oper_code(p_dcb->dch_oper)); |
| #endif |
| send_evt = false; |
| break; |
| } |
| |
| if (p_dcb->local_mdep_id == BTA_HL_ECHO_TEST_MDEP_ID) { |
| p_mcb->echo_test = false; |
| send_evt = false; |
| |
| if (p_dcb->dch_oper != BTA_HL_DCH_OP_LOCAL_CLOSE_ECHO_TEST) { |
| switch (p_dcb->echo_oper) { |
| case BTA_HL_ECHO_OP_CI_GET_ECHO_DATA: |
| case BTA_HL_ECHO_OP_SDP_INIT: |
| case BTA_HL_ECHO_OP_MDL_CREATE_CFM: |
| case BTA_HL_ECHO_OP_DCH_OPEN_CFM: |
| case BTA_HL_ECHO_OP_LOOP_BACK: |
| |
| status = BTA_HL_STATUS_FAIL; |
| send_evt = true; |
| break; |
| case BTA_HL_ECHO_OP_OPEN_IND: |
| case BTA_HL_ECHO_OP_ECHO_PKT: |
| break; |
| default: |
| APPL_TRACE_ERROR("Invalid echo_oper=%d", p_dcb->echo_oper); |
| break; |
| } |
| } else { |
| status = p_dcb->ci_put_echo_data_status; |
| send_evt = true; |
| } |
| |
| if (send_evt) { |
| bta_hl_build_echo_test_cfm(&evt_data, p_acb->app_handle, |
| p_mcb->mcl_handle, status); |
| event = BTA_HL_DCH_ECHO_TEST_CFM_EVT; |
| } |
| } |
| |
| bta_hl_clean_mdl_cb(app_idx, mcl_idx, mdl_idx); |
| |
| if (send_evt) { |
| if (p_acb->p_cback) { |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("Send Event: %s", bta_hl_cback_evt_code(event)); |
| #endif |
| p_acb->p_cback(event, (tBTA_HL*)&evt_data); |
| } |
| } |
| /* check cch close is in progress or not */ |
| bta_hl_check_cch_close(app_idx, mcl_idx, p_data, false); |
| } |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_mca_close_ind |
| * |
| * Description Action routine for processing the close indication |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_mca_close_ind(uint8_t app_idx, uint8_t mcl_idx, uint8_t mdl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_mca_close_ind dch oper=%s", |
| bta_hl_dch_oper_code(p_dcb->dch_oper)); |
| #endif |
| |
| p_dcb->intentional_close = false; |
| if (p_data->mca_evt.mca_data.close_ind.reason == L2CAP_DISC_OK) { |
| p_dcb->intentional_close = true; |
| } |
| |
| if (!p_dcb->cout_oper) { |
| if ((p_dcb->dch_oper != BTA_HL_DCH_OP_REMOTE_OPEN) && |
| (p_dcb->dch_oper != BTA_HL_DCH_OP_REMOTE_RECONNECT)) { |
| p_dcb->dch_oper = BTA_HL_DCH_OP_REMOTE_CLOSE; |
| } |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, BTA_HL_DCH_CLOSE_CMPL_EVT, |
| p_data); |
| } else { |
| p_dcb->close_pending = true; |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_mca_close_cfm |
| * |
| * Description Action routine for processing the close confirmation |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_mca_close_cfm(uint8_t app_idx, uint8_t mcl_idx, uint8_t mdl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_mca_close_cfm dch_oper=%s", |
| bta_hl_dch_oper_code(p_dcb->dch_oper)); |
| #endif |
| |
| switch (p_dcb->dch_oper) { |
| case BTA_HL_DCH_OP_LOCAL_CLOSE: |
| case BTA_HL_DCH_OP_LOCAL_OPEN: |
| case BTA_HL_DCH_OP_LOCAL_RECONNECT: |
| case BTA_HL_DCH_OP_LOCAL_CLOSE_ECHO_TEST: |
| case BTA_HL_DCH_OP_REMOTE_DELETE: |
| case BTA_HL_DCH_OP_LOCAL_CLOSE_RECONNECT: |
| case BTA_HL_DCH_OP_NONE: |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, |
| BTA_HL_DCH_CLOSE_CMPL_EVT, p_data); |
| break; |
| default: |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_ERROR("Invalid dch_oper=%s for close cfm", |
| bta_hl_dch_oper_code(p_dcb->dch_oper)); |
| #endif |
| break; |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_mca_close |
| * |
| * Description Action routine for processing the DCH close request |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_mca_close(uint8_t app_idx, uint8_t mcl_idx, uint8_t mdl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_APP_CB* p_acb = BTA_HL_GET_APP_CB_PTR(app_idx); |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| tBTA_HL_STATUS status = BTA_HL_STATUS_OK; |
| tBTA_HL evt_data; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_mca_close"); |
| #endif |
| if (!p_dcb->cout_oper) { |
| p_dcb->close_pending = false; |
| if (MCA_CloseReq((tMCA_DL)p_dcb->mdl_handle) == MCA_SUCCESS) { |
| p_dcb->dch_oper = BTA_HL_DCH_OP_LOCAL_CLOSE; |
| } else { |
| status = BTA_HL_STATUS_FAIL; |
| } |
| |
| if ((status != BTA_HL_STATUS_OK) && |
| (p_mcb->cch_close_dch_oper != BTA_HL_CCH_CLOSE_OP_DCH_CLOSE)) { |
| bta_hl_build_dch_close_cfm(&evt_data, p_acb->app_handle, |
| p_mcb->mcl_handle, |
| p_data->api_dch_close.mdl_handle, status); |
| p_acb->p_cback(BTA_HL_DCH_CLOSE_CFM_EVT, (tBTA_HL*)&evt_data); |
| } |
| } else { |
| p_dcb->close_pending = true; |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_mca_open_ind |
| * |
| * Description Action routine for processing the open indication |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_mca_open_ind(uint8_t app_idx, uint8_t mcl_idx, uint8_t mdl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_APP_CB* p_acb = BTA_HL_GET_APP_CB_PTR(app_idx); |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| tMCA_DL_OPEN* p_open_ind = &p_data->mca_evt.mca_data.open_ind; |
| tBTA_HL evt_data; |
| tBTA_HL_EVT event; |
| uint8_t old_dch_oper = BTA_HL_DCH_OP_NONE; |
| bool send_event = false; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_mca_open_ind"); |
| #endif |
| if ((p_dcb->dch_oper == BTA_HL_DCH_OP_REMOTE_OPEN) || |
| (p_dcb->dch_oper == BTA_HL_DCH_OP_REMOTE_RECONNECT)) { |
| p_dcb->mdl_handle = (tBTA_HL_MDL_HANDLE)p_open_ind->mdl; |
| p_dcb->mtu = p_open_ind->mtu; |
| |
| evt_data.dch_open_ind.mdl_handle = p_dcb->mdl_handle; |
| evt_data.dch_open_ind.mcl_handle = p_mcb->mcl_handle; |
| evt_data.dch_open_ind.app_handle = p_acb->app_handle; |
| |
| evt_data.dch_open_ind.local_mdep_id = p_dcb->local_mdep_id; |
| evt_data.dch_open_ind.mdl_id = p_dcb->mdl_id; |
| evt_data.dch_open_ind.mtu = p_dcb->mtu; |
| |
| if (p_dcb->chnl_cfg.fcr_opt.mode == L2CAP_FCR_ERTM_MODE) { |
| evt_data.dch_open_ind.dch_mode = BTA_HL_DCH_MODE_RELIABLE; |
| if (!bta_hl_is_the_first_reliable_existed(app_idx, mcl_idx)) { |
| p_dcb->is_the_first_reliable = true; |
| } |
| } else { |
| evt_data.dch_open_ind.dch_mode = BTA_HL_DCH_MODE_STREAMING; |
| } |
| evt_data.dch_open_ind.first_reliable = p_dcb->is_the_first_reliable; |
| |
| old_dch_oper = p_dcb->dch_oper; |
| p_dcb->dch_oper = BTA_HL_DCH_OP_NONE; |
| } |
| |
| switch (old_dch_oper) { |
| case BTA_HL_DCH_OP_REMOTE_OPEN: |
| |
| p_dcb->dch_mode = evt_data.dch_open_ind.dch_mode; |
| if (p_dcb->local_mdep_id != BTA_HL_ECHO_TEST_MDEP_ID) { |
| bta_hl_save_mdl_cfg(app_idx, mcl_idx, mdl_idx); |
| event = BTA_HL_DCH_OPEN_IND_EVT; |
| send_event = true; |
| } else { |
| p_dcb->echo_oper = BTA_HL_ECHO_OP_ECHO_PKT; |
| } |
| |
| break; |
| |
| case BTA_HL_DCH_OP_REMOTE_RECONNECT: |
| |
| if (bta_hl_validate_chan_cfg(app_idx, mcl_idx, mdl_idx)) { |
| bta_hl_save_mdl_cfg(app_idx, mcl_idx, mdl_idx); |
| event = BTA_HL_DCH_RECONNECT_IND_EVT; |
| send_event = true; |
| } else { |
| if (MCA_CloseReq((tMCA_DL)p_dcb->mdl_handle) == MCA_SUCCESS) { |
| p_dcb->dch_oper = BTA_HL_DCH_OP_LOCAL_CLOSE_RECONNECT; |
| } else { |
| APPL_TRACE_ERROR("Unabel to close DCH for reconnect cfg mismatch"); |
| } |
| } |
| break; |
| default: |
| break; |
| } |
| |
| if (send_event) { |
| p_acb->p_cback(event, (tBTA_HL*)&evt_data); |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_mca_open_cfm |
| * |
| * Description Action routine for processing the open confirmation |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_mca_open_cfm(uint8_t app_idx, uint8_t mcl_idx, uint8_t mdl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_APP_CB* p_acb = BTA_HL_GET_APP_CB_PTR(app_idx); |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| tMCA_DL_OPEN* p_open_cfm = &p_data->mca_evt.mca_data.open_cfm; |
| tBTA_HL evt_data; |
| tBTA_HL_EVT event; |
| uint8_t old_dch_oper = BTA_HL_DCH_OP_NONE; |
| tBTA_HL_DCH_MODE dch_mode = BTA_HL_DCH_MODE_STREAMING; |
| bool send_event = false; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_mca_open_cfm"); |
| #endif |
| if ((p_dcb->dch_oper == BTA_HL_DCH_OP_LOCAL_OPEN) || |
| (p_dcb->dch_oper == BTA_HL_DCH_OP_LOCAL_RECONNECT)) { |
| p_dcb->mdl_handle = (tBTA_HL_MDL_HANDLE)p_open_cfm->mdl; |
| p_dcb->mtu = p_open_cfm->mtu; |
| |
| /*todo verify dch_mode, mtu and fcs for reconnect */ |
| if (p_dcb->chnl_cfg.fcr_opt.mode == L2CAP_FCR_ERTM_MODE) { |
| dch_mode = BTA_HL_DCH_MODE_RELIABLE; |
| } |
| |
| if (p_dcb->local_mdep_id != BTA_HL_ECHO_TEST_MDEP_ID) { |
| if (dch_mode == BTA_HL_DCH_MODE_RELIABLE) { |
| if (!bta_hl_is_the_first_reliable_existed(app_idx, mcl_idx)) { |
| p_dcb->is_the_first_reliable = true; |
| } |
| } |
| } |
| |
| bta_hl_build_dch_open_cfm( |
| &evt_data, p_acb->app_handle, p_mcb->mcl_handle, p_dcb->mdl_handle, |
| p_dcb->local_mdep_id, p_dcb->mdl_id, dch_mode, |
| p_dcb->is_the_first_reliable, p_dcb->mtu, BTA_HL_STATUS_OK); |
| |
| old_dch_oper = p_dcb->dch_oper; |
| p_dcb->dch_oper = BTA_HL_DCH_OP_NONE; |
| } else { |
| APPL_TRACE_ERROR("Error dch oper =%d", p_dcb->dch_oper); |
| return; |
| } |
| |
| switch (old_dch_oper) { |
| case BTA_HL_DCH_OP_LOCAL_OPEN: |
| |
| p_dcb->dch_mode = dch_mode; |
| if (p_dcb->local_mdep_id != BTA_HL_ECHO_TEST_MDEP_ID) { |
| bta_hl_save_mdl_cfg(app_idx, mcl_idx, mdl_idx); |
| event = BTA_HL_DCH_OPEN_CFM_EVT; |
| send_event = true; |
| } else { |
| p_dcb->echo_oper = BTA_HL_ECHO_OP_LOOP_BACK; |
| if (MCA_WriteReq((tMCA_DL)p_dcb->mdl_handle, p_dcb->p_echo_tx_pkt) != |
| MCA_SUCCESS) { |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, |
| BTA_HL_DCH_CLOSE_ECHO_TEST_EVT, p_data); |
| } else { |
| p_dcb->p_echo_tx_pkt = NULL; |
| } |
| } |
| break; |
| |
| case BTA_HL_DCH_OP_LOCAL_RECONNECT: |
| |
| if (bta_hl_validate_chan_cfg(app_idx, mcl_idx, mdl_idx)) { |
| bta_hl_save_mdl_cfg(app_idx, mcl_idx, mdl_idx); |
| event = BTA_HL_DCH_RECONNECT_CFM_EVT; |
| send_event = true; |
| } else { |
| if (MCA_CloseReq((tMCA_DL)p_dcb->mdl_handle) == MCA_SUCCESS) { |
| p_dcb->dch_oper = BTA_HL_DCH_OP_LOCAL_CLOSE_RECONNECT; |
| } else { |
| APPL_TRACE_ERROR("Unabel to close DCH for reconnect cfg mismatch"); |
| } |
| } |
| break; |
| default: |
| break; |
| } |
| |
| if (send_event) p_acb->p_cback(event, (tBTA_HL*)&evt_data); |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_mca_abort_ind |
| * |
| * Description Action routine for processing the abort indication |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_mca_abort_ind(uint8_t app_idx, uint8_t mcl_idx, uint8_t mdl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_mca_abort_ind"); |
| #endif |
| |
| p_dcb->abort_oper |= BTA_HL_ABORT_REMOTE_MASK; |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, BTA_HL_DCH_CLOSE_CMPL_EVT, |
| p_data); |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_mca_abort_cfm |
| * |
| * Description Action routine for processing the abort confirmation |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_mca_abort_cfm(uint8_t app_idx, uint8_t mcl_idx, uint8_t mdl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_APP_CB* p_acb = BTA_HL_GET_APP_CB_PTR(app_idx); |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| tBTA_HL evt_data; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_mca_abort_cfm"); |
| #endif |
| |
| if (p_dcb->abort_oper) { |
| if (p_data->mca_evt.mca_data.abort_cfm.rsp_code != MCA_RSP_SUCCESS) { |
| if (p_dcb->abort_oper & BTA_HL_ABORT_LOCAL_MASK) { |
| bta_hl_build_abort_cfm(&evt_data, p_acb->app_handle, p_mcb->mcl_handle, |
| BTA_HL_STATUS_FAIL); |
| p_acb->p_cback(BTA_HL_DCH_ABORT_CFM_EVT, (tBTA_HL*)&evt_data); |
| } |
| } else { |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, |
| BTA_HL_DCH_CLOSE_CMPL_EVT, p_data); |
| } |
| } else { |
| APPL_TRACE_ERROR("Not expecting Abort CFM "); |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_mca_abort |
| * |
| * Description Action routine for processing the abort request |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_mca_abort(uint8_t app_idx, uint8_t mcl_idx, uint8_t mdl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_APP_CB* p_acb = BTA_HL_GET_APP_CB_PTR(app_idx); |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| tMCA_RESULT mca_result; |
| tBTA_HL evt_data; |
| |
| if (((p_mcb->sdp_oper == BTA_HL_SDP_OP_DCH_OPEN_INIT) || |
| (p_mcb->sdp_oper == BTA_HL_SDP_OP_DCH_RECONNECT_INIT)) && |
| (p_mcb->sdp_mdl_idx == mdl_idx)) { |
| p_dcb->abort_oper |= BTA_HL_ABORT_PENDING_MASK; |
| return; |
| } else if (p_dcb->echo_oper == BTA_HL_ECHO_OP_CI_GET_ECHO_DATA) { |
| p_dcb->abort_oper |= BTA_HL_ABORT_PENDING_MASK; |
| return; |
| } |
| |
| p_dcb->abort_oper &= ~BTA_HL_ABORT_PENDING_MASK; |
| |
| mca_result = MCA_Abort((tMCA_CL)p_mcb->mcl_handle); |
| if (mca_result != MCA_SUCCESS) { |
| if (mca_result == MCA_NO_RESOURCES) { |
| p_dcb->abort_oper |= BTA_HL_ABORT_PENDING_MASK; |
| } else { |
| if (p_dcb->abort_oper & BTA_HL_ABORT_LOCAL_MASK) { |
| bta_hl_build_abort_cfm(&evt_data, p_acb->app_handle, p_mcb->mcl_handle, |
| BTA_HL_STATUS_FAIL); |
| p_acb->p_cback(BTA_HL_DCH_ABORT_CFM_EVT, (tBTA_HL*)&evt_data); |
| } |
| bta_hl_check_cch_close(app_idx, mcl_idx, p_data, false); |
| } |
| } |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_mca_abort abort_oper=0x%x", p_dcb->abort_oper); |
| #endif |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_mca_reconnect_ind |
| * |
| * Description Action routine for processing the reconnect indication |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_mca_reconnect_ind(uint8_t app_idx, uint8_t mcl_idx, |
| uint8_t mdl_idx, tBTA_HL_DATA* p_data) { |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| tBTA_HL_MDL_CFG* p_mdl_cfg; |
| tMCA_EVT_HDR* p_reconnect_ind = &p_data->mca_evt.mca_data.reconnect_ind; |
| uint8_t mdl_cfg_idx, in_use_mdl_idx, mdep_cfg_idx; |
| uint8_t rsp_code = MCA_RSP_SUCCESS; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_mca_reconnect_ind mdl_id=%d", |
| p_reconnect_ind->mdl_id); |
| #endif |
| |
| if (bta_hl_find_mdl_cfg_idx(app_idx, mcl_idx, p_reconnect_ind->mdl_id, |
| &mdl_cfg_idx)) { |
| if (!bta_hl_find_mdl_idx(app_idx, mcl_idx, p_reconnect_ind->mdl_id, |
| &in_use_mdl_idx)) { |
| p_mdl_cfg = BTA_HL_GET_MDL_CFG_PTR(app_idx, mdl_cfg_idx); |
| |
| if (bta_hl_find_mdep_cfg_idx(app_idx, p_mdl_cfg->local_mdep_id, |
| &mdep_cfg_idx)) { |
| p_dcb->in_use = true; |
| p_dcb->dch_oper = BTA_HL_DCH_OP_REMOTE_RECONNECT; |
| p_dcb->sec_mask = (BTA_SEC_AUTHENTICATE | BTA_SEC_ENCRYPT); |
| p_dcb->peer_mdep_id = 0xFF; |
| p_dcb->local_mdep_id = p_mdl_cfg->local_mdep_id; |
| p_dcb->local_mdep_cfg_idx = mdep_cfg_idx; |
| p_dcb->local_cfg = BTA_HL_DCH_CFG_UNKNOWN; |
| p_dcb->mdl_id = p_reconnect_ind->mdl_id; |
| p_dcb->mdl_cfg_idx_included = true; |
| p_dcb->mdl_cfg_idx = mdl_cfg_idx; |
| p_dcb->dch_mode = p_mdl_cfg->dch_mode; |
| bta_hl_find_rxtx_apdu_size(app_idx, mdep_cfg_idx, |
| &p_dcb->max_rx_apdu_size, |
| &p_dcb->max_tx_apdu_size); |
| bta_hl_set_dch_chan_cfg(app_idx, mcl_idx, mdl_idx, p_data); |
| } else { |
| rsp_code = MCA_RSP_BAD_MDL; |
| } |
| } else { |
| rsp_code = MCA_RSP_BAD_MDL; |
| } |
| } else { |
| rsp_code = MCA_RSP_BAD_MDL; |
| } |
| |
| if (MCA_ReconnectMdlRsp((tMCA_CL)p_mcb->mcl_handle, p_dcb->local_mdep_id, |
| p_dcb->mdl_id, rsp_code, |
| &p_dcb->chnl_cfg) != MCA_SUCCESS) { |
| MCA_Abort((tMCA_CL)p_mcb->mcl_handle); |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, BTA_HL_DCH_CLOSE_CMPL_EVT, |
| p_data); |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_mca_reconnect_cfm |
| * |
| * Description Action routine for processing the reconenct confirmation |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_mca_reconnect_cfm(uint8_t app_idx, uint8_t mcl_idx, |
| uint8_t mdl_idx, tBTA_HL_DATA* p_data) { |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| tMCA_RSP_EVT* p_reconnect_cfm = &p_data->mca_evt.mca_data.reconnect_cfm; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_mca_reconnect_cfm"); |
| #endif |
| if (p_dcb->abort_oper & BTA_HL_ABORT_PENDING_MASK) { |
| p_dcb->abort_oper &= ~BTA_HL_ABORT_PENDING_MASK; |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, BTA_HL_DCH_ABORT_EVT, |
| p_data); |
| return; |
| } |
| |
| if (p_dcb->dch_oper == BTA_HL_DCH_OP_LOCAL_RECONNECT) { |
| if (p_reconnect_cfm->rsp_code == MCA_RSP_SUCCESS) { |
| bta_hl_set_dch_chan_cfg(app_idx, mcl_idx, mdl_idx, p_data); |
| |
| if (MCA_DataChnlCfg((tMCA_CL)p_mcb->mcl_handle, &p_dcb->chnl_cfg) != |
| MCA_SUCCESS) { |
| /* should be able to abort so no checking of the return code */ |
| MCA_Abort((tMCA_CL)p_mcb->mcl_handle); |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, |
| BTA_HL_DCH_CLOSE_CMPL_EVT, p_data); |
| } |
| } else { |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, |
| BTA_HL_DCH_CLOSE_CMPL_EVT, p_data); |
| } |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_mca_reconnect |
| * |
| * Description Action routine for processing the reconnect request |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_mca_reconnect(uint8_t app_idx, uint8_t mcl_idx, uint8_t mdl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| tMCA_CHNL_CFG* p_chnl_cfg = NULL; |
| uint8_t sdp_idx; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_mca_reconnect"); |
| #endif |
| if (bta_hl_find_sdp_idx_using_ctrl_psm(&p_mcb->sdp, p_mcb->ctrl_psm, |
| &sdp_idx)) { |
| p_mcb->data_psm = p_mcb->sdp.sdp_rec[sdp_idx].data_psm; |
| if (MCA_ReconnectMdl((tMCA_CL)p_mcb->mcl_handle, p_dcb->local_mdep_id, |
| p_mcb->data_psm, p_dcb->mdl_id, |
| p_chnl_cfg) != MCA_SUCCESS) { |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, |
| BTA_HL_DCH_CLOSE_CMPL_EVT, p_data); |
| } |
| } else { |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, BTA_HL_DCH_CLOSE_CMPL_EVT, |
| p_data); |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_create_rsp |
| * |
| * Description Action routine for processing BTA_HL_API_DCH_CREATE_RSP_EVT |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_create_rsp(uint8_t app_idx, uint8_t mcl_idx, uint8_t mdl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| tBTA_HL_API_DCH_CREATE_RSP* p_create_rsp = &p_data->api_dch_create_rsp; |
| uint8_t mca_rsp_code = MCA_RSP_SUCCESS; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_create_rsp"); |
| #endif |
| if (p_create_rsp->rsp_code == BTA_HL_DCH_CREATE_RSP_SUCCESS) { |
| p_dcb->dch_oper = BTA_HL_DCH_OP_REMOTE_OPEN; |
| p_dcb->local_cfg = p_create_rsp->cfg_rsp; |
| |
| bta_hl_set_dch_chan_cfg(app_idx, mcl_idx, mdl_idx, p_data); |
| } else { |
| mca_rsp_code = MCA_RSP_CFG_REJ; |
| } |
| |
| if (MCA_CreateMdlRsp((tMCA_CL)p_mcb->mcl_handle, p_dcb->local_mdep_id, |
| p_dcb->mdl_id, p_dcb->local_cfg, mca_rsp_code, |
| &p_dcb->chnl_cfg) != MCA_SUCCESS) { |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, BTA_HL_DCH_CLOSE_CMPL_EVT, |
| p_data); |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_mca_create_ind |
| * |
| * Description Action routine for processing |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_mca_create_ind(uint8_t app_idx, uint8_t mcl_idx, |
| uint8_t mdl_idx, tBTA_HL_DATA* p_data) { |
| tBTA_HL_APP_CB* p_acb = BTA_HL_GET_APP_CB_PTR(app_idx); |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| tMCA_CREATE_IND* p_create_ind = &p_data->mca_evt.mca_data.create_ind; |
| uint8_t mdep_cfg_idx; |
| uint8_t cfg_rsp; |
| uint8_t rsp_code = MCA_RSP_SUCCESS; |
| bool send_create_ind_evt = false; |
| tBTA_HL evt_data; |
| tBTA_HL_ECHO_CFG* p_echo_cfg; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_mca_create_ind"); |
| #endif |
| |
| if (bta_hl_find_mdep_cfg_idx(app_idx, p_create_ind->dep_id, &mdep_cfg_idx)) { |
| if (p_create_ind->dep_id == BTA_HL_ECHO_TEST_MDEP_ID) { |
| if (bta_hl_find_echo_cfg_rsp(app_idx, mcl_idx, mdep_cfg_idx, |
| p_create_ind->cfg, &cfg_rsp)) { |
| p_dcb->in_use = true; |
| p_dcb->dch_oper = BTA_HL_DCH_OP_REMOTE_OPEN; |
| p_dcb->local_mdep_id = p_create_ind->dep_id; |
| p_dcb->local_mdep_cfg_idx = mdep_cfg_idx; |
| p_dcb->local_cfg = cfg_rsp; |
| p_dcb->remote_cfg = p_create_ind->cfg; |
| p_dcb->mdl_id = p_create_ind->mdl_id; |
| p_dcb->mdl_cfg_idx_included = false; |
| p_echo_cfg = BTA_HL_GET_ECHO_CFG_PTR(app_idx); |
| p_dcb->max_rx_apdu_size = p_echo_cfg->max_rx_apdu_size; |
| p_dcb->max_tx_apdu_size = p_echo_cfg->max_tx_apdu_size; |
| |
| bta_hl_set_dch_chan_cfg(app_idx, mcl_idx, mdl_idx, p_data); |
| } else { |
| rsp_code = MCA_RSP_CFG_REJ; |
| } |
| } else |
| |
| { |
| p_dcb->in_use = true; |
| p_dcb->dch_oper = BTA_HL_DCH_OP_REMOTE_CREATE; |
| p_dcb->local_mdep_id = p_create_ind->dep_id; |
| p_dcb->local_mdep_cfg_idx = mdep_cfg_idx; |
| p_dcb->local_cfg = BTA_HL_DCH_CFG_UNKNOWN; |
| p_dcb->remote_cfg = p_create_ind->cfg; |
| p_dcb->mdl_id = p_create_ind->mdl_id; |
| p_dcb->mdl_cfg_idx_included = false; |
| bta_hl_find_rxtx_apdu_size(app_idx, mdep_cfg_idx, |
| &p_dcb->max_rx_apdu_size, |
| &p_dcb->max_tx_apdu_size); |
| send_create_ind_evt = true; |
| } |
| } else { |
| rsp_code = MCA_RSP_BAD_MDEP; |
| } |
| |
| if (send_create_ind_evt) { |
| evt_data.dch_create_ind.mcl_handle = p_mcb->mcl_handle; |
| evt_data.dch_create_ind.app_handle = p_acb->app_handle; |
| evt_data.dch_create_ind.local_mdep_id = p_dcb->local_mdep_id; |
| evt_data.dch_create_ind.mdl_id = p_dcb->mdl_id; |
| evt_data.dch_create_ind.cfg = p_dcb->remote_cfg; |
| evt_data.dch_create_ind.bd_addr = p_mcb->bd_addr; |
| p_acb->p_cback(BTA_HL_DCH_CREATE_IND_EVT, (tBTA_HL*)&evt_data); |
| } else { |
| if (MCA_CreateMdlRsp((tMCA_CL)p_mcb->mcl_handle, p_dcb->local_mdep_id, |
| p_dcb->mdl_id, p_dcb->local_cfg, rsp_code, |
| &p_dcb->chnl_cfg) != MCA_SUCCESS) { |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, |
| BTA_HL_DCH_CLOSE_CMPL_EVT, p_data); |
| } else { |
| if (p_dcb->local_mdep_id == BTA_HL_ECHO_TEST_MDEP_ID) { |
| p_mcb->echo_test = true; |
| p_dcb->echo_oper = BTA_HL_ECHO_OP_OPEN_IND; |
| } |
| } |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_mca_create_cfm |
| * |
| * Description Action routine for processing |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_mca_create_cfm(uint8_t app_idx, uint8_t mcl_idx, |
| uint8_t mdl_idx, tBTA_HL_DATA* p_data) { |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| tMCA_CREATE_CFM* p_create_cfm = &p_data->mca_evt.mca_data.create_cfm; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_mca_create_cfm"); |
| #endif |
| |
| if (p_dcb->abort_oper & BTA_HL_ABORT_PENDING_MASK) { |
| p_dcb->abort_oper &= ~BTA_HL_ABORT_PENDING_MASK; |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, BTA_HL_DCH_ABORT_EVT, |
| p_data); |
| return; |
| } |
| |
| if (p_dcb->dch_oper == BTA_HL_DCH_OP_LOCAL_OPEN) { |
| if (p_create_cfm->rsp_code == MCA_RSP_SUCCESS) { |
| if (bta_hl_validate_cfg(app_idx, mcl_idx, mdl_idx, p_create_cfm->cfg)) { |
| bta_hl_set_dch_chan_cfg(app_idx, mcl_idx, mdl_idx, p_data); |
| |
| if (MCA_DataChnlCfg((tMCA_CL)p_mcb->mcl_handle, &p_dcb->chnl_cfg) != |
| MCA_SUCCESS) { |
| /* this should not happen */ |
| APPL_TRACE_ERROR("Unable to create data channel"); |
| MCA_Abort((tMCA_CL)p_mcb->mcl_handle); |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, |
| BTA_HL_DCH_CLOSE_CMPL_EVT, p_data); |
| } else { |
| if (p_dcb->local_mdep_id == BTA_HL_ECHO_TEST_MDEP_ID) { |
| p_dcb->echo_oper = BTA_HL_ECHO_OP_DCH_OPEN_CFM; |
| } |
| } |
| } else { |
| MCA_Abort((tMCA_CL)p_mcb->mcl_handle); |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, |
| BTA_HL_DCH_CLOSE_CMPL_EVT, p_data); |
| } |
| } else { |
| APPL_TRACE_ERROR("MCA Create- failed"); |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, |
| BTA_HL_DCH_CLOSE_CMPL_EVT, p_data); |
| } |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_mca_create |
| * |
| * Description Action routine for processing the MDL create request |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_mca_create(uint8_t app_idx, uint8_t mcl_idx, uint8_t mdl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| tMCA_RESULT result; |
| uint8_t sdp_idx; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_mca_create"); |
| #endif |
| |
| if (bta_hl_find_sdp_idx_using_ctrl_psm(&p_mcb->sdp, p_mcb->ctrl_psm, |
| &sdp_idx) && |
| bta_hl_validate_peer_cfg(app_idx, mcl_idx, mdl_idx, p_dcb->peer_mdep_id, |
| p_dcb->peer_mdep_role, sdp_idx)) { |
| p_mcb->data_psm = p_mcb->sdp.sdp_rec[sdp_idx].data_psm; |
| result = MCA_CreateMdl((tMCA_CL)p_mcb->mcl_handle, p_dcb->local_mdep_id, |
| p_mcb->data_psm, p_dcb->mdl_id, p_dcb->peer_mdep_id, |
| p_dcb->local_cfg, NULL); |
| if (result != MCA_SUCCESS) { |
| APPL_TRACE_ERROR("MCA_CreateMdl FAIL mca_result=%d", result); |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, |
| BTA_HL_DCH_CLOSE_CMPL_EVT, p_data); |
| } |
| } else { |
| APPL_TRACE_ERROR("MCA Create- SDP idx or peer MDEP cfg not found"); |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, BTA_HL_DCH_CLOSE_CMPL_EVT, |
| p_data); |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_sdp_fail |
| * |
| * Description Action routine for processing the SDP failed event |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_dch_sdp_fail(uint8_t app_idx, uint8_t mcl_idx, uint8_t mdl_idx, |
| tBTA_HL_DATA* p_data) { |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_dch_sdp_fail"); |
| #endif |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, mdl_idx, BTA_HL_DCH_CLOSE_CMPL_EVT, |
| p_data); |
| } |
| |
| /****************************************************************************** |
| * |
| * Function bta_hl_sdp_cback |
| * |
| * Description This is the SDP callback function used by HL. |
| * This function will be executed by SDP when the service |
| * search is completed. If the search is successful, it |
| * finds the first record in the database that matches the |
| * UUID of the search. Then retrieves the scn from the |
| * record. |
| * |
| * Returns void. |
| * |
| *****************************************************************************/ |
| static void bta_hl_sdp_cback(uint8_t sdp_oper, uint8_t app_idx, uint8_t mcl_idx, |
| uint8_t mdl_idx, uint16_t status) { |
| tBTA_HL_MCL_CB* p_cb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_SDP_REC* p_hdp_rec; |
| tBTA_HL_CCH_SDP* p_cch_buf; |
| tBTA_HL_DCH_SDP* p_dch_buf; |
| tSDP_DISC_REC* p_rec = NULL; |
| tSDP_PROTOCOL_ELEM pe; |
| tSDP_DISC_ATTR* p_attr; |
| uint8_t i, rec_cnt; |
| tBTA_HL_SUP_FEATURE_LIST_ELEM sup_feature; |
| bool sdp_parsing_ok = false, result = false; |
| uint16_t event; |
| tBTA_HL_MDL_CB* p_dcb; |
| uint16_t service_uuid; |
| uint16_t name_len; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG( |
| "bta_hl_sdp_cback status:%d sdp_oper=%d app_idx=%d, mcl_idx=%d, " |
| "mdl_idx=%d", |
| status, sdp_oper, app_idx, mcl_idx, mdl_idx); |
| #endif |
| |
| rec_cnt = 0; |
| service_uuid = bta_hl_get_service_uuids(sdp_oper, app_idx, mcl_idx, mdl_idx); |
| |
| if (status == SDP_SUCCESS || status == SDP_DB_FULL) { |
| memset(&p_cb->sdp, 0, sizeof(tBTA_HL_SDP)); |
| do { |
| if (bta_hl_find_service_in_db(app_idx, mcl_idx, service_uuid, &p_rec)) { |
| p_hdp_rec = &p_cb->sdp.sdp_rec[rec_cnt]; |
| p_cb->sdp.num_recs = rec_cnt + 1; |
| } else { |
| break; |
| } |
| |
| if (SDP_FindProtocolListElemInRec(p_rec, UUID_PROTOCOL_L2CAP, &pe)) { |
| p_hdp_rec->ctrl_psm = (uint16_t)pe.params[0]; |
| } else { |
| APPL_TRACE_WARNING("Control PSM not found"); |
| break; |
| } |
| if (SDP_FindAddProtoListsElemInRec(p_rec, UUID_PROTOCOL_L2CAP, &pe)) { |
| p_hdp_rec->data_psm = (uint16_t)pe.params[0]; |
| } else { |
| APPL_TRACE_WARNING("Data PSM not found"); |
| break; |
| } |
| |
| p_hdp_rec->srv_name[0] = '\0'; |
| p_attr = SDP_FindAttributeInRec(p_rec, ATTR_ID_SERVICE_NAME); |
| if (p_attr != NULL) { |
| if (SDP_DISC_ATTR_LEN(p_attr->attr_len_type) < BT_MAX_SERVICE_NAME_LEN) |
| name_len = (uint16_t)SDP_DISC_ATTR_LEN(p_attr->attr_len_type); |
| else |
| name_len = BT_MAX_SERVICE_NAME_LEN; |
| memcpy(p_hdp_rec->srv_name, p_attr->attr_value.v.array, name_len); |
| } |
| |
| p_hdp_rec->srv_desp[0] = '\0'; |
| p_attr = SDP_FindAttributeInRec(p_rec, ATTR_ID_SERVICE_DESCRIPTION); |
| if (p_attr != NULL) { |
| if (SDP_DISC_ATTR_LEN(p_attr->attr_len_type) < BT_MAX_SERVICE_NAME_LEN) |
| name_len = (uint16_t)SDP_DISC_ATTR_LEN(p_attr->attr_len_type); |
| else |
| name_len = BT_MAX_SERVICE_NAME_LEN; |
| memcpy(p_hdp_rec->srv_desp, p_attr->attr_value.v.array, name_len); |
| } |
| |
| p_hdp_rec->provider_name[0] = '\0'; |
| p_attr = SDP_FindAttributeInRec(p_rec, ATTR_ID_PROVIDER_NAME); |
| if (p_attr != NULL) { |
| if (SDP_DISC_ATTR_LEN(p_attr->attr_len_type) < BT_MAX_SERVICE_NAME_LEN) |
| name_len = (uint16_t)SDP_DISC_ATTR_LEN(p_attr->attr_len_type); |
| else |
| name_len = BT_MAX_SERVICE_NAME_LEN; |
| memcpy(p_hdp_rec->provider_name, p_attr->attr_value.v.array, name_len); |
| } |
| |
| p_attr = SDP_FindAttributeInRec(p_rec, ATTR_ID_HDP_MCAP_SUP_PROC); |
| if (p_attr != NULL) { |
| p_hdp_rec->mcap_sup_proc = p_attr->attr_value.v.u8; |
| } else { |
| APPL_TRACE_WARNING("MCAP SUP PROC not found"); |
| break; |
| } |
| |
| p_attr = SDP_FindAttributeInRec(p_rec, ATTR_ID_HDP_SUP_FEAT_LIST); |
| if (p_attr != NULL) { |
| if (bta_hl_fill_sup_feature_list(p_attr, &sup_feature)) { |
| p_hdp_rec->num_mdeps = (uint8_t)sup_feature.num_elems; |
| APPL_TRACE_WARNING("bta_hl_sdp_cback num_mdeps %d", |
| sup_feature.num_elems); |
| for (i = 0; i < sup_feature.num_elems; i++) { |
| p_hdp_rec->mdep_cfg[i].data_type = |
| sup_feature.list_elem[i].data_type; |
| p_hdp_rec->mdep_cfg[i].mdep_id = sup_feature.list_elem[i].mdep_id; |
| p_hdp_rec->mdep_cfg[i].mdep_role = |
| sup_feature.list_elem[i].mdep_role; |
| /* Check MDEP Description pointer to prevent crash due to null |
| * pointer */ |
| if (sup_feature.list_elem[i].p_mdep_desp != NULL) { |
| strlcpy(p_hdp_rec->mdep_cfg[i].mdep_desp, |
| sup_feature.list_elem[i].p_mdep_desp, |
| BTA_HL_MDEP_DESP_LEN); |
| } else { |
| APPL_TRACE_ERROR( |
| "bta_hl_sdp_cback Incorrect Mdep[%d] Description (Null ptr)", |
| i); |
| } |
| } |
| |
| sdp_parsing_ok = true; |
| } else { |
| APPL_TRACE_WARNING("HDP supported feature list fill failed"); |
| break; |
| } |
| } else { |
| APPL_TRACE_WARNING("HDP supported feature list not found"); |
| break; |
| } |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("record=%d ctrl_psm=%0x data_psm=%x", rec_cnt + 1, |
| p_hdp_rec->ctrl_psm, p_hdp_rec->data_psm); |
| APPL_TRACE_DEBUG("srv_name=[%s]", (p_hdp_rec->srv_name[0] != '\0') |
| ? p_hdp_rec->srv_name |
| : "NULL"); |
| APPL_TRACE_DEBUG("srv_desp=[%s]", (p_hdp_rec->srv_desp[0] != '\0') |
| ? p_hdp_rec->srv_desp |
| : "NULL"); |
| for (i = 0; i < sup_feature.num_elems; i++) { |
| APPL_TRACE_DEBUG( |
| "index=0x%02x mdep_id=0x%04x data type=0x%04x mdep role=%s(0x%02x)", |
| (i + 1), p_hdp_rec->mdep_cfg[i].mdep_id, |
| p_hdp_rec->mdep_cfg[i].data_type, |
| (p_hdp_rec->mdep_cfg[i].mdep_role == BTA_HL_MDEP_ROLE_SOURCE) |
| ? "Src" |
| : "Snk", |
| p_hdp_rec->mdep_cfg[i].mdep_role); |
| } |
| APPL_TRACE_DEBUG("provider_name=[%s]", |
| (p_hdp_rec->provider_name[0] != '\0') |
| ? p_hdp_rec->provider_name |
| : "NULL"); |
| APPL_TRACE_DEBUG("found MCAP sup procedure=%d", |
| p_cb->sdp.sdp_rec[rec_cnt].mcap_sup_proc); |
| #endif |
| rec_cnt++; |
| if (rec_cnt >= BTA_HL_NUM_SDP_RECS) { |
| APPL_TRACE_WARNING("No more spaces for SDP recs max_rec_cnt=%d", |
| BTA_HL_NUM_SDP_RECS); |
| break; |
| } |
| |
| } while (true); |
| } |
| |
| osi_free_and_reset((void**)&p_cb->p_db); |
| |
| if ((status == SDP_SUCCESS || status == SDP_DB_FULL) && p_cb->sdp.num_recs && |
| sdp_parsing_ok) { |
| result = true; |
| } else { |
| APPL_TRACE_WARNING( |
| "SDP Failed sdp_status=%d num_recs=%d sdp_parsing_ok=%d ", status, |
| p_cb->sdp.num_recs, sdp_parsing_ok); |
| } |
| |
| p_cb->sdp_oper = BTA_HL_SDP_OP_NONE; |
| |
| switch (sdp_oper) { |
| case BTA_HL_SDP_OP_CCH_INIT: |
| case BTA_HL_SDP_OP_SDP_QUERY_NEW: |
| case BTA_HL_SDP_OP_SDP_QUERY_CURRENT: |
| |
| /* send result in event back to BTA */ |
| p_cch_buf = (tBTA_HL_CCH_SDP*)osi_malloc(sizeof(tBTA_HL_CCH_SDP)); |
| if (result) { |
| if (sdp_oper == BTA_HL_SDP_OP_CCH_INIT) { |
| event = BTA_HL_CCH_SDP_OK_EVT; |
| if (p_cb->close_pending) event = BTA_HL_CCH_SDP_FAIL_EVT; |
| } else { |
| event = BTA_HL_SDP_QUERY_OK_EVT; |
| } |
| } else { |
| if (sdp_oper == BTA_HL_SDP_OP_CCH_INIT) |
| event = BTA_HL_CCH_SDP_FAIL_EVT; |
| else |
| event = BTA_HL_SDP_QUERY_FAIL_EVT; |
| } |
| p_cch_buf->hdr.event = event; |
| |
| p_cch_buf->app_idx = app_idx; |
| p_cch_buf->mcl_idx = mcl_idx; |
| p_cch_buf->release_mcl_cb = false; |
| if (sdp_oper == BTA_HL_SDP_OP_SDP_QUERY_NEW) |
| p_cch_buf->release_mcl_cb = true; |
| |
| bta_sys_sendmsg(p_cch_buf); |
| break; |
| case BTA_HL_SDP_OP_DCH_OPEN_INIT: |
| case BTA_HL_SDP_OP_DCH_RECONNECT_INIT: |
| p_dch_buf = (tBTA_HL_DCH_SDP*)osi_malloc(sizeof(tBTA_HL_DCH_SDP)); |
| p_dch_buf->hdr.event = BTA_HL_DCH_SDP_FAIL_EVT; |
| p_dch_buf->app_idx = app_idx; |
| p_dch_buf->mcl_idx = mcl_idx; |
| p_dch_buf->mdl_idx = mdl_idx; |
| p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, mdl_idx); |
| if (p_dcb->abort_oper & BTA_HL_ABORT_PENDING_MASK) { |
| p_dcb->abort_oper &= ~BTA_HL_ABORT_PENDING_MASK; |
| result = false; |
| } |
| if (result) { |
| if (sdp_oper == BTA_HL_SDP_OP_DCH_OPEN_INIT) { |
| if (p_dcb->local_mdep_id == BTA_HL_ECHO_TEST_MDEP_ID) { |
| p_dch_buf->hdr.event = BTA_HL_DCH_ECHO_TEST_EVT; |
| } else { |
| p_dch_buf->hdr.event = BTA_HL_DCH_OPEN_EVT; |
| } |
| } else { |
| p_dch_buf->hdr.event = BTA_HL_DCH_RECONNECT_EVT; |
| } |
| } |
| bta_sys_sendmsg(p_dch_buf); |
| break; |
| default: |
| break; |
| } |
| } |
| |
| /****************************************************************************** |
| * |
| * Function bta_hl_sdp_cback0 |
| * |
| * Description This is the SDP callback function used by index = 0 |
| * |
| * Returns void. |
| * |
| *****************************************************************************/ |
| static void bta_hl_sdp_cback0(uint16_t status) { |
| bta_hl_sdp_cback(bta_hl_cb.scb[0].sdp_oper, bta_hl_cb.scb[0].app_idx, |
| bta_hl_cb.scb[0].mcl_idx, bta_hl_cb.scb[0].mdl_idx, status); |
| bta_hl_deallocate_spd_cback(0); |
| } |
| |
| /****************************************************************************** |
| * |
| * Function bta_hl_sdp_cback1 |
| * |
| * Description This is the SDP callback function used by index = 1 |
| * |
| * Parameters status - status of the SDP callabck |
| * |
| * Returns void. |
| * |
| *****************************************************************************/ |
| static void bta_hl_sdp_cback1(uint16_t status) { |
| bta_hl_sdp_cback(bta_hl_cb.scb[1].sdp_oper, bta_hl_cb.scb[1].app_idx, |
| bta_hl_cb.scb[1].mcl_idx, bta_hl_cb.scb[1].mdl_idx, status); |
| bta_hl_deallocate_spd_cback(1); |
| } |
| |
| /****************************************************************************** |
| * |
| * Function bta_hl_sdp_cback2 |
| * |
| * Description This is the SDP callback function used by index = 2 |
| * |
| * Returns void. |
| * |
| *****************************************************************************/ |
| static void bta_hl_sdp_cback2(uint16_t status) { |
| bta_hl_sdp_cback(bta_hl_cb.scb[2].sdp_oper, bta_hl_cb.scb[2].app_idx, |
| bta_hl_cb.scb[2].mcl_idx, bta_hl_cb.scb[2].mdl_idx, status); |
| bta_hl_deallocate_spd_cback(2); |
| } |
| |
| /****************************************************************************** |
| * |
| * Function bta_hl_sdp_cback3 |
| * |
| * Description This is the SDP callback function used by index = 3 |
| * |
| * Returns void. |
| * |
| *****************************************************************************/ |
| static void bta_hl_sdp_cback3(uint16_t status) { |
| bta_hl_sdp_cback(bta_hl_cb.scb[3].sdp_oper, bta_hl_cb.scb[3].app_idx, |
| bta_hl_cb.scb[3].mcl_idx, bta_hl_cb.scb[3].mdl_idx, status); |
| bta_hl_deallocate_spd_cback(3); |
| } |
| |
| /****************************************************************************** |
| * |
| * Function bta_hl_sdp_cback4 |
| * |
| * Description This is the SDP callback function used by index = 4 |
| * |
| * Parameters status - status of the SDP callabck |
| * |
| * Returns void. |
| * |
| *****************************************************************************/ |
| static void bta_hl_sdp_cback4(uint16_t status) { |
| bta_hl_sdp_cback(bta_hl_cb.scb[4].sdp_oper, bta_hl_cb.scb[4].app_idx, |
| bta_hl_cb.scb[4].mcl_idx, bta_hl_cb.scb[4].mdl_idx, status); |
| bta_hl_deallocate_spd_cback(4); |
| } |
| |
| /****************************************************************************** |
| * |
| * Function bta_hl_sdp_cback5 |
| * |
| * Description This is the SDP callback function used by index = 5 |
| * |
| * Parameters status - status of the SDP callabck |
| * |
| * Returns void. |
| * |
| *****************************************************************************/ |
| static void bta_hl_sdp_cback5(uint16_t status) { |
| bta_hl_sdp_cback(bta_hl_cb.scb[5].sdp_oper, bta_hl_cb.scb[5].app_idx, |
| bta_hl_cb.scb[5].mcl_idx, bta_hl_cb.scb[5].mdl_idx, status); |
| bta_hl_deallocate_spd_cback(5); |
| } |
| |
| /****************************************************************************** |
| * |
| * Function bta_hl_sdp_cback6 |
| * |
| * Description This is the SDP callback function used by index = 6 |
| * |
| * Returns void. |
| * |
| *****************************************************************************/ |
| static void bta_hl_sdp_cback6(uint16_t status) { |
| bta_hl_sdp_cback(bta_hl_cb.scb[6].sdp_oper, bta_hl_cb.scb[6].app_idx, |
| bta_hl_cb.scb[6].mcl_idx, bta_hl_cb.scb[6].mdl_idx, status); |
| bta_hl_deallocate_spd_cback(6); |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_deallocate_spd_cback |
| * |
| * Description Deallocate a SDP control block |
| * |
| * Returns bool - true found |
| * false not found |
| * |
| ******************************************************************************/ |
| void bta_hl_deallocate_spd_cback(uint8_t sdp_cback_idx) { |
| tBTA_HL_SDP_CB* p_spd_cb = &bta_hl_cb.scb[sdp_cback_idx]; |
| |
| memset(p_spd_cb, 0, sizeof(tBTA_HL_SDP_CB)); |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_deallocate_spd_cback index=%d", sdp_cback_idx); |
| #endif |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_allocate_spd_cback |
| * |
| * Description Finds a not in used SDP control block index |
| * |
| * |
| * Returns bool - true found |
| * false not found |
| * |
| ******************************************************************************/ |
| tSDP_DISC_CMPL_CB* bta_hl_allocate_spd_cback(tBTA_HL_SDP_OPER sdp_oper, |
| uint8_t app_idx, uint8_t mcl_idx, |
| uint8_t mdl_idx, |
| uint8_t* p_sdp_cback_idx) { |
| uint8_t i; |
| tSDP_DISC_CMPL_CB* p_cbcak = NULL; |
| |
| for (i = 0; i < BTA_HL_NUM_SDP_CBACKS; i++) { |
| if (!bta_hl_cb.scb[i].in_use) { |
| p_cbcak = bta_hl_sdp_cback_arr[i]; |
| bta_hl_cb.scb[i].in_use = true; |
| bta_hl_cb.scb[i].sdp_oper = sdp_oper; |
| bta_hl_cb.scb[i].app_idx = app_idx; |
| bta_hl_cb.scb[i].mcl_idx = mcl_idx; |
| bta_hl_cb.scb[i].mdl_idx = mdl_idx; |
| *p_sdp_cback_idx = i; |
| break; |
| } |
| } |
| |
| if (i == BTA_HL_NUM_SDP_CBACKS) { |
| APPL_TRACE_WARNING("No scb is available to allocate") |
| } else { |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_allocate_spd_cback cback_idx=%d ", i); |
| APPL_TRACE_DEBUG("sdp_oper=%d, app_idx=%d, mcl_idx=%d, mdl_idx=%d", |
| bta_hl_cb.scb[i].sdp_oper, bta_hl_cb.scb[i].app_idx, |
| bta_hl_cb.scb[i].mcl_idx, bta_hl_cb.scb[i].mdl_idx); |
| #endif |
| } |
| return p_cbcak; |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_init_sdp |
| * |
| * Description Action routine for processing the SDP initiattion request |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| tBTA_HL_STATUS bta_hl_init_sdp(tBTA_HL_SDP_OPER sdp_oper, uint8_t app_idx, |
| uint8_t mcl_idx, uint8_t mdl_idx) { |
| tBTA_HL_MCL_CB* p_cb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| uint16_t attr_list[BTA_HL_NUM_SRCH_ATTR]; |
| uint16_t num_attrs = BTA_HL_NUM_SRCH_ATTR; |
| tBTA_HL_STATUS status; |
| uint8_t sdp_cback_idx; |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG( |
| "bta_hl_init_sdp sdp_oper=%d app_idx=%d mcl_idx=%d, mdl_idx=%d", sdp_oper, |
| app_idx, mcl_idx, mdl_idx); |
| #endif |
| p_cb->sdp_cback = bta_hl_allocate_spd_cback(sdp_oper, app_idx, mcl_idx, |
| mdl_idx, &sdp_cback_idx); |
| if (p_cb->sdp_cback != NULL) { |
| if (p_cb->p_db == NULL) |
| (p_cb->p_db = (tSDP_DISCOVERY_DB*)osi_malloc(BTA_HL_DISC_SIZE)); |
| attr_list[0] = ATTR_ID_SERVICE_CLASS_ID_LIST; |
| attr_list[1] = ATTR_ID_PROTOCOL_DESC_LIST; |
| attr_list[2] = ATTR_ID_BT_PROFILE_DESC_LIST; |
| attr_list[3] = ATTR_ID_ADDITION_PROTO_DESC_LISTS; |
| attr_list[4] = ATTR_ID_SERVICE_NAME; |
| attr_list[5] = ATTR_ID_SERVICE_DESCRIPTION; |
| attr_list[6] = ATTR_ID_PROVIDER_NAME; |
| attr_list[7] = ATTR_ID_HDP_SUP_FEAT_LIST; |
| attr_list[8] = ATTR_ID_HDP_DATA_EXCH_SPEC; |
| attr_list[9] = ATTR_ID_HDP_MCAP_SUP_PROC; |
| |
| Uuid uuid_list = Uuid::From16Bit(UUID_SERVCLASS_HDP_PROFILE); |
| SDP_InitDiscoveryDb(p_cb->p_db, BTA_HL_DISC_SIZE, 1, &uuid_list, num_attrs, |
| attr_list); |
| |
| if (!SDP_ServiceSearchAttributeRequest(p_cb->bd_addr, p_cb->p_db, |
| p_cb->sdp_cback)) { |
| status = BTA_HL_STATUS_FAIL; |
| } else { |
| status = BTA_HL_STATUS_OK; |
| } |
| } else { |
| status = BTA_HL_STATUS_SDP_NO_RESOURCE; |
| } |
| |
| if (status != BTA_HL_STATUS_OK) { |
| osi_free_and_reset((void**)&p_cb->p_db); |
| if (status != BTA_HL_STATUS_SDP_NO_RESOURCE) |
| bta_hl_deallocate_spd_cback(sdp_cback_idx); |
| } |
| |
| return status; |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_cch_sdp_init |
| * |
| * Description Action routine for processing the CCH SDP init event |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_cch_sdp_init(uint8_t app_idx, uint8_t mcl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_MCL_CB* p_cb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_cch_init_sdp"); |
| #endif |
| if (p_cb->sdp_oper == BTA_HL_SDP_OP_NONE) { |
| p_cb->app_id = p_data->api_cch_open.app_id; |
| p_cb->sdp_oper = BTA_HL_SDP_OP_CCH_INIT; |
| |
| if (bta_hl_init_sdp(p_cb->sdp_oper, app_idx, mcl_idx, 0xFF) != |
| BTA_HL_STATUS_OK) { |
| p_cb->sdp_oper = BTA_HL_SDP_OP_NONE; |
| bta_hl_cch_sm_execute(app_idx, mcl_idx, BTA_HL_CCH_SDP_FAIL_EVT, p_data); |
| } |
| } else { |
| APPL_TRACE_ERROR("SDP in use"); |
| bta_hl_cch_sm_execute(app_idx, mcl_idx, BTA_HL_CCH_SDP_FAIL_EVT, p_data); |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_cch_mca_open |
| * |
| * Description Action routine for processing the CCH open request |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_cch_mca_open(uint8_t app_idx, uint8_t mcl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_APP_CB* p_acb = BTA_HL_GET_APP_CB_PTR(app_idx); |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| uint8_t sdp_idx; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_cch_mca_open"); |
| #endif |
| |
| if (bta_hl_find_sdp_idx_using_ctrl_psm(&p_mcb->sdp, p_mcb->req_ctrl_psm, |
| &sdp_idx)) { |
| p_mcb->ctrl_psm = p_mcb->sdp.sdp_rec[sdp_idx].ctrl_psm; |
| p_mcb->data_psm = p_mcb->sdp.sdp_rec[sdp_idx].data_psm; |
| if (MCA_ConnectReq((tMCA_HANDLE)p_acb->app_handle, p_mcb->bd_addr, |
| p_mcb->ctrl_psm, p_mcb->sec_mask) != MCA_SUCCESS) { |
| bta_hl_cch_sm_execute(app_idx, mcl_idx, BTA_HL_CCH_CLOSE_CMPL_EVT, |
| p_data); |
| } |
| } else { |
| bta_hl_cch_sm_execute(app_idx, mcl_idx, BTA_HL_CCH_CLOSE_CMPL_EVT, p_data); |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_cch_mca_close |
| * |
| * Description Action routine for processing the CCH close request |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_cch_mca_close(uint8_t app_idx, uint8_t mcl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_cch_mca_close mcl_handle=%d", p_mcb->mcl_handle); |
| #endif |
| if (p_mcb->sdp_oper != BTA_HL_SDP_OP_CCH_INIT) { |
| if (p_mcb->mcl_handle) { |
| if (MCA_DisconnectReq((tMCA_HANDLE)p_mcb->mcl_handle) != MCA_SUCCESS) { |
| bta_hl_cch_sm_execute(app_idx, mcl_idx, BTA_HL_CCH_CLOSE_CMPL_EVT, |
| p_data); |
| } |
| } else { |
| p_mcb->close_pending = true; |
| APPL_TRACE_DEBUG( |
| "No valid mcl_handle to stop the CCH setup now so wait until CCH is " |
| "up then close it"); |
| } |
| } else { |
| p_mcb->close_pending = true; |
| APPL_TRACE_DEBUG( |
| "can not stop the CCH setup becasue SDP is in progress so wait until " |
| "it is done"); |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_cch_close_cmpl |
| * |
| * Description Action routine for processing the CCH close complete event |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_cch_close_cmpl(uint8_t app_idx, uint8_t mcl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_APP_CB* p_acb = BTA_HL_GET_APP_CB_PTR(app_idx); |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| |
| tBTA_HL evt_data; |
| tBTA_HL_EVT event; |
| bool send_evt = true; |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_cch_close_cmpl"); |
| #endif |
| bta_sys_conn_close(BTA_ID_HL, p_acb->app_id, p_mcb->bd_addr); |
| |
| if (p_mcb->cch_oper == BTA_HL_CCH_OP_LOCAL_CLOSE && |
| p_mcb->force_close_local_cch_opening) { |
| p_mcb->cch_oper = BTA_HL_CCH_OP_LOCAL_OPEN; |
| APPL_TRACE_DEBUG( |
| "change cch_oper from BTA_HL_CCH_OP_LOCAL_CLOSE to " |
| "BTA_HL_CCH_OP_LOCAL_OPEN"); |
| } |
| |
| switch (p_mcb->cch_oper) { |
| case BTA_HL_CCH_OP_LOCAL_OPEN: |
| bta_hl_build_cch_open_cfm(&evt_data, p_mcb->app_id, p_acb->app_handle, |
| p_mcb->mcl_handle, p_mcb->bd_addr, |
| BTA_HL_STATUS_FAIL); |
| event = BTA_HL_CCH_OPEN_CFM_EVT; |
| break; |
| case BTA_HL_CCH_OP_LOCAL_CLOSE: |
| bta_hl_build_cch_close_cfm(&evt_data, p_acb->app_handle, |
| p_mcb->mcl_handle, BTA_HL_STATUS_OK); |
| event = BTA_HL_CCH_CLOSE_CFM_EVT; |
| break; |
| case BTA_HL_CCH_OP_REMOTE_CLOSE: |
| bta_hl_build_cch_close_ind(&evt_data, p_acb->app_handle, |
| p_mcb->mcl_handle, p_mcb->intentional_close); |
| event = BTA_HL_CCH_CLOSE_IND_EVT; |
| break; |
| default: |
| send_evt = false; |
| break; |
| } |
| |
| memset(p_mcb, 0, sizeof(tBTA_HL_MCL_CB)); |
| |
| if (send_evt) p_acb->p_cback(event, (tBTA_HL*)&evt_data); |
| |
| bta_hl_check_deregistration(app_idx, p_data); |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_cch_mca_disconnect |
| * |
| * Description Action routine for processing the CCH disconnect indication |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_cch_mca_disconnect(uint8_t app_idx, uint8_t mcl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL_MDL_CB* p_dcb; |
| uint8_t i; |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_cch_mca_disconnect"); |
| #endif |
| |
| p_mcb->intentional_close = false; |
| if (p_data->mca_evt.mca_data.disconnect_ind.reason == L2CAP_DISC_OK) { |
| p_mcb->intentional_close = true; |
| } |
| |
| for (i = 0; i < BTA_HL_NUM_MDLS_PER_MCL; i++) { |
| p_dcb = BTA_HL_GET_MDL_CB_PTR(app_idx, mcl_idx, i); |
| if (p_dcb->in_use && (p_dcb->dch_state != BTA_HL_DCH_IDLE_ST)) { |
| if (p_mcb->cch_oper == BTA_HL_CCH_OP_LOCAL_CLOSE) { |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, i, BTA_HL_DCH_CLOSE_CMPL_EVT, |
| p_data); |
| } else { |
| bta_hl_dch_sm_execute(app_idx, mcl_idx, i, BTA_HL_MCA_CLOSE_IND_EVT, |
| p_data); |
| } |
| } |
| } |
| bta_hl_cch_sm_execute(app_idx, mcl_idx, BTA_HL_CCH_CLOSE_CMPL_EVT, p_data); |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_cch_mca_disc_open |
| * |
| * Description Action routine for disconnect the just opened Control |
| * channel |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_cch_mca_disc_open(uint8_t app_idx, uint8_t mcl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_cch_mca_disc_open mcl_handle=0x%x close_pending=%d", |
| p_data->mca_evt.mcl_handle, p_mcb->close_pending); |
| #endif |
| |
| p_mcb->close_pending = false; |
| p_mcb->mcl_handle = p_data->mca_evt.mcl_handle; |
| bta_hl_cch_mca_close(app_idx, mcl_idx, p_data); |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_cch_mca_rsp_tout |
| * |
| * Description Action routine for processing the MCAP response timeout |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_cch_mca_rsp_tout(uint8_t app_idx, uint8_t mcl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_cch_mca_rsp_tout"); |
| #endif |
| |
| p_mcb->rsp_tout = true; |
| |
| bta_hl_check_cch_close(app_idx, mcl_idx, p_data, true); |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_cch_mca_connect |
| * |
| * Description Action routine for processing the CCH connect indication |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_cch_mca_connect(uint8_t app_idx, uint8_t mcl_idx, |
| tBTA_HL_DATA* p_data) { |
| tBTA_HL_APP_CB* p_acb = BTA_HL_GET_APP_CB_PTR(app_idx); |
| tBTA_HL_MCL_CB* p_mcb = BTA_HL_GET_MCL_CB_PTR(app_idx, mcl_idx); |
| tBTA_HL evt_data; |
| tBTA_HL_EVT event; |
| bool send_event = true; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_hl_cch_mca_connect mcl_handle=%d ", |
| p_data->mca_evt.mcl_handle); |
| #endif |
| |
| p_mcb->mcl_handle = p_data->mca_evt.mcl_handle; |
| p_mcb->bd_addr = p_data->mca_evt.mca_data.connect_ind.bd_addr; |
| p_mcb->cch_mtu = p_data->mca_evt.mca_data.connect_ind.mtu; |
| |
| bta_sys_conn_open(BTA_ID_HL, p_acb->app_id, p_mcb->bd_addr); |
| switch (p_mcb->cch_oper) { |
| case BTA_HL_CCH_OP_LOCAL_OPEN: |
| bta_hl_build_cch_open_cfm(&evt_data, p_mcb->app_id, p_acb->app_handle, |
| p_mcb->mcl_handle, p_mcb->bd_addr, |
| BTA_HL_STATUS_OK); |
| event = BTA_HL_CCH_OPEN_CFM_EVT; |
| break; |
| case BTA_HL_CCH_OP_REMOTE_OPEN: |
| bta_hl_build_cch_open_ind(&evt_data, p_acb->app_handle, p_mcb->mcl_handle, |
| p_mcb->bd_addr); |
| event = BTA_HL_CCH_OPEN_IND_EVT; |
| break; |
| default: |
| send_event = false; |
| break; |
| } |
| |
| p_mcb->cch_oper = BTA_HL_CCH_OP_NONE; |
| if (send_event) p_acb->p_cback(event, (tBTA_HL*)&evt_data); |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_mcap_ctrl_cback |
| * |
| * Description MCAP control callback function for HL. |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_mcap_ctrl_cback(tMCA_HANDLE handle, tMCA_CL mcl, uint8_t event, |
| tMCA_CTRL* p_data) { |
| bool send_event = true; |
| uint16_t mca_event; |
| |
| #if (BTA_HL_DEBUG == TRUE) |
| APPL_TRACE_EVENT("bta_hl_mcap_ctrl_cback event[%s]", |
| bta_hl_mcap_evt_code(event)); |
| #endif |
| |
| switch (event) { |
| case MCA_CREATE_IND_EVT: |
| mca_event = (uint16_t)BTA_HL_MCA_CREATE_IND_EVT; |
| break; |
| case MCA_CREATE_CFM_EVT: |
| mca_event = (uint16_t)BTA_HL_MCA_CREATE_CFM_EVT; |
| break; |
| case MCA_RECONNECT_IND_EVT: |
| mca_event = (uint16_t)BTA_HL_MCA_RECONNECT_IND_EVT; |
| break; |
| case MCA_RECONNECT_CFM_EVT: |
| mca_event = (uint16_t)BTA_HL_MCA_RECONNECT_CFM_EVT; |
| break; |
| case MCA_ABORT_IND_EVT: |
| mca_event = (uint16_t)BTA_HL_MCA_ABORT_IND_EVT; |
| break; |
| case MCA_ABORT_CFM_EVT: |
| mca_event = (uint16_t)BTA_HL_MCA_ABORT_CFM_EVT; |
| break; |
| case MCA_DELETE_IND_EVT: |
| mca_event = (uint16_t)BTA_HL_MCA_DELETE_IND_EVT; |
| break; |
| case MCA_DELETE_CFM_EVT: |
| mca_event = (uint16_t)BTA_HL_MCA_DELETE_CFM_EVT; |
| break; |
| case MCA_CONNECT_IND_EVT: |
| mca_event = (uint16_t)BTA_HL_MCA_CONNECT_IND_EVT; |
| break; |
| case MCA_DISCONNECT_IND_EVT: |
| mca_event = (uint16_t)BTA_HL_MCA_DISCONNECT_IND_EVT; |
| break; |
| case MCA_OPEN_IND_EVT: |
| mca_event = (uint16_t)BTA_HL_MCA_OPEN_IND_EVT; |
| break; |
| case MCA_OPEN_CFM_EVT: |
| mca_event = (uint16_t)BTA_HL_MCA_OPEN_CFM_EVT; |
| break; |
| case MCA_CLOSE_IND_EVT: |
| mca_event = (uint16_t)BTA_HL_MCA_CLOSE_IND_EVT; |
| break; |
| case MCA_CLOSE_CFM_EVT: |
| mca_event = (uint16_t)BTA_HL_MCA_CLOSE_CFM_EVT; |
| break; |
| case MCA_CONG_CHG_EVT: |
| mca_event = (uint16_t)BTA_HL_MCA_CONG_CHG_EVT; |
| break; |
| case MCA_RSP_TOUT_IND_EVT: |
| mca_event = (uint16_t)BTA_HL_MCA_RSP_TOUT_IND_EVT; |
| break; |
| case MCA_ERROR_RSP_EVT: |
| |
| default: |
| send_event = false; |
| break; |
| } |
| |
| if (send_event) { |
| tBTA_HL_MCA_EVT* p_msg = |
| (tBTA_HL_MCA_EVT*)osi_malloc(sizeof(tBTA_HL_MCA_EVT)); |
| p_msg->hdr.event = mca_event; |
| p_msg->app_handle = (tBTA_HL_APP_HANDLE)handle; |
| p_msg->mcl_handle = (tBTA_HL_MCL_HANDLE)mcl; |
| memcpy(&p_msg->mca_data, p_data, sizeof(tMCA_CTRL)); |
| bta_sys_sendmsg(p_msg); |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_mcap_data_cback |
| * |
| * Description MCAP data callback function for HL. |
| * |
| * Returns void |
| * |
| ******************************************************************************/ |
| void bta_hl_mcap_data_cback(tMCA_DL mdl, BT_HDR* p_pkt) { |
| uint8_t app_idx, mcl_idx, mdl_idx; |
| if (bta_hl_find_mdl_idx_using_handle((tBTA_HL_MDL_HANDLE)mdl, &app_idx, |
| &mcl_idx, &mdl_idx)) { |
| tBTA_HL_MCA_RCV_DATA_EVT* p_msg = |
| (tBTA_HL_MCA_RCV_DATA_EVT*)osi_malloc(sizeof(tBTA_HL_MCA_RCV_DATA_EVT)); |
| p_msg->hdr.event = BTA_HL_MCA_RCV_DATA_EVT; |
| p_msg->app_idx = app_idx; |
| p_msg->mcl_idx = mcl_idx; |
| p_msg->mdl_idx = mdl_idx; |
| p_msg->p_pkt = p_pkt; |
| bta_sys_sendmsg(p_msg); |
| } |
| } |
| |
| /***************************************************************************** |
| * Debug Functions |
| ****************************************************************************/ |
| #if (BTA_HL_DEBUG == TRUE) |
| |
| #define CASE_RETURN_STR(const) \ |
| case const: \ |
| return #const; |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_mcap_evt_code |
| * |
| * Description get the MCAP event string pointer |
| * |
| * Returns char * - event string pointer |
| * |
| ******************************************************************************/ |
| static const char* bta_hl_mcap_evt_code(uint8_t evt_code) { |
| switch (evt_code) { |
| CASE_RETURN_STR(MCA_ERROR_RSP_EVT) |
| CASE_RETURN_STR(MCA_CREATE_IND_EVT) |
| CASE_RETURN_STR(MCA_CREATE_CFM_EVT) |
| CASE_RETURN_STR(MCA_RECONNECT_IND_EVT) |
| CASE_RETURN_STR(MCA_RECONNECT_CFM_EVT) |
| CASE_RETURN_STR(MCA_ABORT_IND_EVT) |
| CASE_RETURN_STR(MCA_ABORT_CFM_EVT) |
| CASE_RETURN_STR(MCA_DELETE_IND_EVT) |
| CASE_RETURN_STR(MCA_DELETE_CFM_EVT) |
| CASE_RETURN_STR(MCA_CONNECT_IND_EVT) |
| CASE_RETURN_STR(MCA_DISCONNECT_IND_EVT) |
| CASE_RETURN_STR(MCA_OPEN_IND_EVT) |
| CASE_RETURN_STR(MCA_OPEN_CFM_EVT) |
| CASE_RETURN_STR(MCA_CLOSE_IND_EVT) |
| CASE_RETURN_STR(MCA_CLOSE_CFM_EVT) |
| CASE_RETURN_STR(MCA_CONG_CHG_EVT) |
| CASE_RETURN_STR(MCA_RSP_TOUT_IND_EVT) |
| default: |
| return "Unknown MCAP event code"; |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_cback_evt_code |
| * |
| * Description get the HDP event string pointer |
| * |
| * Returns char * - event string pointer |
| * |
| ******************************************************************************/ |
| static const char* bta_hl_cback_evt_code(uint8_t evt_code) { |
| switch (evt_code) { |
| CASE_RETURN_STR(BTA_HL_CCH_OPEN_IND_EVT) |
| CASE_RETURN_STR(BTA_HL_CCH_OPEN_CFM_EVT) |
| CASE_RETURN_STR(BTA_HL_CCH_CLOSE_IND_EVT) |
| CASE_RETURN_STR(BTA_HL_CCH_CLOSE_CFM_EVT) |
| CASE_RETURN_STR(BTA_HL_DCH_OPEN_IND_EVT) |
| CASE_RETURN_STR(BTA_HL_DCH_OPEN_CFM_EVT) |
| CASE_RETURN_STR(BTA_HL_DCH_CLOSE_IND_EVT) |
| CASE_RETURN_STR(BTA_HL_DCH_CLOSE_CFM_EVT) |
| CASE_RETURN_STR(BTA_HL_DCH_RCV_DATA_IND_EVT) |
| CASE_RETURN_STR(BTA_HL_REGISTER_CFM_EVT) |
| CASE_RETURN_STR(BTA_HL_DEREGISTER_CFM_EVT) |
| CASE_RETURN_STR(BTA_HL_DCH_RECONNECT_CFM_EVT) |
| CASE_RETURN_STR(BTA_HL_DCH_RECONNECT_IND_EVT) |
| CASE_RETURN_STR(BTA_HL_DCH_ECHO_TEST_CFM_EVT) |
| CASE_RETURN_STR(BTA_HL_SDP_QUERY_CFM_EVT) |
| CASE_RETURN_STR(BTA_HL_CONG_CHG_IND_EVT) |
| CASE_RETURN_STR(BTA_HL_DCH_CREATE_IND_EVT) |
| CASE_RETURN_STR(BTA_HL_DELETE_MDL_IND_EVT) |
| CASE_RETURN_STR(BTA_HL_DELETE_MDL_CFM_EVT) |
| CASE_RETURN_STR(BTA_HL_DCH_ABORT_IND_EVT) |
| CASE_RETURN_STR(BTA_HL_DCH_ABORT_CFM_EVT) |
| default: |
| return "Unknown HDP event code"; |
| } |
| } |
| |
| /******************************************************************************* |
| * |
| * Function bta_hl_dch_oper_code |
| * |
| * Description Get the DCH operation string |
| * |
| * Returns char * - DCH operation string pointer |
| * |
| ******************************************************************************/ |
| static const char* bta_hl_dch_oper_code(tBTA_HL_DCH_OPER oper_code) { |
| switch (oper_code) { |
| CASE_RETURN_STR(BTA_HL_DCH_OP_NONE) |
| CASE_RETURN_STR(BTA_HL_DCH_OP_REMOTE_CREATE) |
| CASE_RETURN_STR(BTA_HL_DCH_OP_LOCAL_OPEN) |
| CASE_RETURN_STR(BTA_HL_DCH_OP_REMOTE_OPEN) |
| CASE_RETURN_STR(BTA_HL_DCH_OP_LOCAL_CLOSE) |
| CASE_RETURN_STR(BTA_HL_DCH_OP_REMOTE_CLOSE) |
| CASE_RETURN_STR(BTA_HL_DCH_OP_LOCAL_DELETE) |
| CASE_RETURN_STR(BTA_HL_DCH_OP_REMOTE_DELETE) |
| CASE_RETURN_STR(BTA_HL_DCH_OP_LOCAL_RECONNECT) |
| CASE_RETURN_STR(BTA_HL_DCH_OP_REMOTE_RECONNECT) |
| CASE_RETURN_STR(BTA_HL_DCH_OP_LOCAL_CLOSE_ECHO_TEST) |
| CASE_RETURN_STR(BTA_HL_DCH_OP_LOCAL_CLOSE_RECONNECT) |
| default: |
| return "Unknown DCH oper code"; |
| } |
| } |
| |
| #endif /* Debug Functions */ |
| #endif /* HL_INCLUDED */ |