/****************************************************************************** * * Copyright (c) 2014 The Android Open Source Project * 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 action functions for the handsfree client. * ******************************************************************************/ #include <string.h> #include "bt_utils.h" #include "bta_api.h" #include "bta_dm_api.h" #include "bta_hf_client_api.h" #include "bta_hf_client_int.h" #include "bta_sys.h" #include "l2c_api.h" #include "osi/include/compat.h" #include "osi/include/osi.h" #include "port_api.h" #include "utl.h" /***************************************************************************** * Constants ****************************************************************************/ /* maximum length of data to read from RFCOMM */ #define BTA_HF_CLIENT_RFC_READ_MAX 512 /******************************************************************************* * * Function bta_hf_client_start_close * * Description Start the process of closing SCO and RFCOMM connection. * * * Returns void * ******************************************************************************/ void bta_hf_client_start_close(tBTA_HF_CLIENT_DATA* p_data) { tBTA_HF_CLIENT_CB* client_cb = bta_hf_client_find_cb_by_handle(p_data->hdr.layer_specific); if (client_cb == NULL) { APPL_TRACE_ERROR("%s: wrong handle to control block %d", __func__, p_data->hdr.layer_specific); return; } /* Take the link out of sniff and set L2C idle time to 0 */ bta_dm_pm_active(client_cb->peer_addr); L2CA_SetIdleTimeoutByBdAddr(client_cb->peer_addr, 0, BT_TRANSPORT_BR_EDR); /* if SCO is open close SCO and wait on RFCOMM close */ if (client_cb->sco_state == BTA_HF_CLIENT_SCO_OPEN_ST) { client_cb->sco_close_rfc = true; } else { bta_hf_client_rfc_do_close(p_data); } /* always do SCO shutdown to handle all SCO corner cases */ bta_hf_client_sco_shutdown(client_cb); } /******************************************************************************* * * Function bta_hf_client_start_open * * Description This starts an HF Client open. * * * Returns void * ******************************************************************************/ void bta_hf_client_start_open(tBTA_HF_CLIENT_DATA* p_data) { tBTA_HF_CLIENT_CB* client_cb = bta_hf_client_find_cb_by_handle(p_data->hdr.layer_specific); if (client_cb == NULL) { APPL_TRACE_ERROR("%s: wrong handle to control block %d", __func__, p_data->hdr.layer_specific); return; } /* store parameters */ if (p_data) { client_cb->peer_addr = p_data->api_open.bd_addr; client_cb->cli_sec_mask = p_data->api_open.sec_mask; } /* Check if RFCOMM has any incoming connection to avoid collision. */ RawAddress pending_bd_addr = RawAddress::kEmpty; if (PORT_IsOpening(&pending_bd_addr)) { /* Let the incoming connection goes through. */ /* Issue collision for now. */ /* We will decide what to do when we find incoming connection later.*/ bta_hf_client_collision_cback(0, BTA_ID_HS, 0, client_cb->peer_addr); return; } /* set role */ client_cb->role = BTA_HF_CLIENT_INT; /* do service search */ bta_hf_client_do_disc(client_cb); } /******************************************************************************* * * Function bta_hf_client_rfc_open * * Description Handle RFCOMM channel open. * * * Returns void * ******************************************************************************/ void bta_hf_client_rfc_open(tBTA_HF_CLIENT_DATA* p_data) { APPL_TRACE_DEBUG("%s", __func__); tBTA_HF_CLIENT_CB* client_cb = bta_hf_client_find_cb_by_handle(p_data->hdr.layer_specific); if (client_cb == NULL) { APPL_TRACE_ERROR("%s: cb not found for handle %d", __func__, p_data->hdr.layer_specific); return; } bta_sys_conn_open(BTA_ID_HS, 1, client_cb->peer_addr); /* start SLC procedure */ bta_hf_client_slc_seq(client_cb, false); } /******************************************************************************* * * Function bta_hf_client_rfc_acp_open * * Description Handle RFCOMM channel open when accepting connection. * * * Returns void * ******************************************************************************/ void bta_hf_client_rfc_acp_open(tBTA_HF_CLIENT_DATA* p_data) { APPL_TRACE_DEBUG("%s", __func__); tBTA_HF_CLIENT_CB* client_cb = bta_hf_client_find_cb_by_handle(p_data->hdr.layer_specific); if (client_cb == NULL) { APPL_TRACE_ERROR("%s: cb not found for handle %d", __func__, p_data->hdr.layer_specific); return; } /* set role */ client_cb->role = BTA_HF_CLIENT_ACP; APPL_TRACE_DEBUG("%s: conn_handle %d", __func__, client_cb->conn_handle); /* get bd addr of peer */ uint16_t lcid = 0; RawAddress dev_addr = RawAddress::kEmpty; int status = PORT_CheckConnection(client_cb->conn_handle, &dev_addr, &lcid); if (status != PORT_SUCCESS) { LOG(ERROR) << __func__ << ": PORT_CheckConnection returned " << status; } /* Collision Handling */ if (alarm_is_scheduled(client_cb->collision_timer)) { alarm_cancel(client_cb->collision_timer); if (dev_addr == client_cb->peer_addr) { /* If incoming and outgoing device are same, nothing more to do. */ /* Outgoing conn will be aborted because we have successful incoming conn. */ } else { /* Resume outgoing connection. */ bta_hf_client_resume_open(client_cb); } } client_cb->peer_addr = dev_addr; /* do service discovery to get features */ bta_hf_client_do_disc(client_cb); /* continue with open processing */ bta_hf_client_rfc_open(p_data); } /******************************************************************************* * * Function bta_hf_client_rfc_fail * * Description RFCOMM connection failed. * * * Returns void * ******************************************************************************/ void bta_hf_client_rfc_fail(tBTA_HF_CLIENT_DATA* p_data) { tBTA_HF_CLIENT_CB* client_cb = bta_hf_client_find_cb_by_handle(p_data->hdr.layer_specific); if (client_cb == NULL) { APPL_TRACE_ERROR("%s: cb not found for handle %d", __func__, p_data->hdr.layer_specific); return; } /* reinitialize stuff */ client_cb->peer_features = 0; client_cb->chld_features = 0; client_cb->role = BTA_HF_CLIENT_ACP; client_cb->svc_conn = false; client_cb->send_at_reply = false; client_cb->negotiated_codec = BTM_SCO_CODEC_CVSD; bta_hf_client_at_reset(client_cb); } /******************************************************************************* * * Function bta_hf_client_disc_fail * * Description This function handles a discovery failure. * * * Returns void * ******************************************************************************/ void bta_hf_client_disc_fail(tBTA_HF_CLIENT_DATA* p_data) { tBTA_HF_CLIENT_CB* client_cb = bta_hf_client_find_cb_by_handle(p_data->hdr.layer_specific); if (client_cb == NULL) { APPL_TRACE_ERROR("%s: cb not found for handle %d", __func__, p_data->hdr.layer_specific); return; } } /******************************************************************************* * * Function bta_hf_client_open_fail * * Description open connection failed. * * * Returns void * ******************************************************************************/ void bta_hf_client_open_fail(tBTA_HF_CLIENT_DATA* p_data) { tBTA_HF_CLIENT_CB* client_cb = bta_hf_client_find_cb_by_handle(p_data->hdr.layer_specific); if (client_cb == NULL) { APPL_TRACE_ERROR("%s: cb not found for handle %d", __func__, p_data->hdr.layer_specific); return; } } /******************************************************************************* * * Function bta_hf_client_rfc_close * * Description RFCOMM connection closed. * * * Returns void * ******************************************************************************/ void bta_hf_client_rfc_close(tBTA_HF_CLIENT_DATA* p_data) { tBTA_HF_CLIENT_CB* client_cb = bta_hf_client_find_cb_by_handle(p_data->hdr.layer_specific); if (client_cb == NULL) { APPL_TRACE_ERROR("%s: cb not found for handle %d", __func__, p_data->hdr.layer_specific); return; } bta_hf_client_at_reset(client_cb); bta_sys_conn_close(BTA_ID_HS, 1, client_cb->peer_addr); /* call close cback */ tBTA_HF_CLIENT evt; memset(&evt, 0, sizeof(evt)); evt.conn.bd_addr = client_cb->peer_addr; /* if not deregistering reopen server */ if (!bta_hf_client_cb_arr.deregister) { /* Make sure SCO is shutdown */ bta_hf_client_sco_shutdown(client_cb); bta_sys_sco_unuse(BTA_ID_HS, 1, client_cb->peer_addr); } /* else close port and deallocate scb */ else { tBTA_HF_CLIENT evt; memset(&evt, 0, sizeof(evt)); evt.reg.bd_addr = client_cb->peer_addr; bta_hf_client_app_callback(BTA_HF_CLIENT_DISABLE_EVT, &evt); } } /******************************************************************************* * * Function bta_hf_client_disc_int_res * * Description This function handles a discovery result when initiator. * * * Returns void * ******************************************************************************/ void bta_hf_client_disc_int_res(tBTA_HF_CLIENT_DATA* p_data) { uint16_t event = BTA_HF_CLIENT_DISC_FAIL_EVT; APPL_TRACE_DEBUG("%s: Status: %d", __func__, p_data->disc_result.status); tBTA_HF_CLIENT_CB* client_cb = bta_hf_client_find_cb_by_handle(p_data->hdr.layer_specific); if (client_cb == NULL) { APPL_TRACE_ERROR("%s: cb not found for handle %d", __func__, p_data->hdr.layer_specific); return; } /* if found service */ if (p_data->disc_result.status == SDP_SUCCESS || p_data->disc_result.status == SDP_DB_FULL) { /* get attributes */ if (bta_hf_client_sdp_find_attr(client_cb)) { event = BTA_HF_CLIENT_DISC_OK_EVT; } } /* free discovery db */ bta_hf_client_free_db(p_data); /* send ourselves sdp ok/fail event */ bta_hf_client_sm_execute(event, p_data); } /******************************************************************************* * * Function bta_hf_client_disc_acp_res * * Description This function handles a discovery result when acceptor. * * * Returns void * ******************************************************************************/ void bta_hf_client_disc_acp_res(tBTA_HF_CLIENT_DATA* p_data) { tBTA_HF_CLIENT_CB* client_cb = bta_hf_client_find_cb_by_handle(p_data->hdr.layer_specific); if (client_cb == NULL) { APPL_TRACE_ERROR("%s: cb not found for handle %d", __func__, p_data->hdr.layer_specific); return; } /* if found service */ if (p_data->disc_result.status == SDP_SUCCESS || p_data->disc_result.status == SDP_DB_FULL) { /* get attributes */ bta_hf_client_sdp_find_attr(client_cb); } /* free discovery db */ bta_hf_client_free_db(p_data); } /******************************************************************************* * * Function bta_hf_client_rfc_data * * Description Read and process data from RFCOMM. * * * Returns void * ******************************************************************************/ void bta_hf_client_rfc_data(tBTA_HF_CLIENT_DATA* p_data) { tBTA_HF_CLIENT_CB* client_cb = bta_hf_client_find_cb_by_handle(p_data->hdr.layer_specific); if (client_cb == NULL) { APPL_TRACE_ERROR("%s: cb not found for handle %d", __func__, p_data->hdr.layer_specific); return; } uint16_t len; char buf[BTA_HF_CLIENT_RFC_READ_MAX]; memset(buf, 0, sizeof(buf)); /* read data from rfcomm; if bad status, we're done */ while (PORT_ReadData(client_cb->conn_handle, buf, BTA_HF_CLIENT_RFC_READ_MAX, &len) == PORT_SUCCESS) { /* if no data, we're done */ if (len == 0) { break; } bta_hf_client_at_parse(client_cb, buf, len); /* no more data to read, we're done */ if (len < BTA_HF_CLIENT_RFC_READ_MAX) { break; } } } /******************************************************************************* * * Function bta_hf_client_svc_conn_open * * Description Service level connection opened * * * Returns void * ******************************************************************************/ void bta_hf_client_svc_conn_open(tBTA_HF_CLIENT_DATA* p_data) { APPL_TRACE_DEBUG("%s", __func__); tBTA_HF_CLIENT_CB* client_cb = bta_hf_client_find_cb_by_handle(p_data->hdr.layer_specific); if (client_cb == NULL) { APPL_TRACE_ERROR("%s: cb not found for handle %d", __func__, p_data->hdr.layer_specific); return; } tBTA_HF_CLIENT evt; memset(&evt, 0, sizeof(evt)); if (!client_cb->svc_conn) { /* set state variable */ client_cb->svc_conn = true; /* call callback */ evt.conn.bd_addr = client_cb->peer_addr; evt.conn.peer_feat = client_cb->peer_features; evt.conn.chld_feat = client_cb->chld_features; bta_hf_client_app_callback(BTA_HF_CLIENT_CONN_EVT, &evt); } }