| /****************************************************************************** |
| * |
| * 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 is the private interface file for the BTA audio gateway. |
| * |
| ******************************************************************************/ |
| #ifndef BTA_AG_INT_H |
| #define BTA_AG_INT_H |
| |
| #include "bta_ag_api.h" |
| #include "bta_ag_at.h" |
| #include "bta_api.h" |
| #include "bta_sys.h" |
| |
| /***************************************************************************** |
| * Constants |
| ****************************************************************************/ |
| /* Time to wait for retry in case of collision */ |
| #ifndef BTA_AG_COLLISION_TIMEOUT_MS |
| #define BTA_AG_COLLISION_TIMEOUT_MS (2 * 1000) /* 2 seconds */ |
| #endif |
| |
| /* RFCOMM MTU SIZE */ |
| #define BTA_AG_MTU 256 |
| |
| /* Max number of peer and local HF indicators */ |
| #define BTA_AG_MAX_NUM_PEER_HF_IND 20 |
| #define BTA_AG_MAX_NUM_LOCAL_HF_IND 4 |
| |
| /* Internal profile indexes */ |
| #define BTA_AG_HSP 0 /* index for HSP */ |
| #define BTA_AG_HFP 1 /* index for HFP */ |
| #define BTA_AG_NUM_IDX 2 /* number of profile indexes */ |
| |
| /* profile role for connection */ |
| #define BTA_AG_ACP 0 /* accepted connection */ |
| #define BTA_AG_INT 1 /* initiating connection */ |
| |
| #define BTA_AG_SDP_FEAT_SPEC \ |
| (BTA_AG_FEAT_3WAY | BTA_AG_FEAT_ECNR | BTA_AG_FEAT_VREC | \ |
| BTA_AG_FEAT_INBAND | BTA_AG_FEAT_VTAG) |
| |
| enum { |
| /* these events are handled by the state machine */ |
| BTA_AG_API_REGISTER_EVT = BTA_SYS_EVT_START(BTA_ID_AG), |
| BTA_AG_API_DEREGISTER_EVT, |
| BTA_AG_API_OPEN_EVT, |
| BTA_AG_API_CLOSE_EVT, |
| BTA_AG_API_AUDIO_OPEN_EVT, |
| BTA_AG_API_AUDIO_CLOSE_EVT, |
| BTA_AG_API_RESULT_EVT, |
| BTA_AG_API_SETCODEC_EVT, |
| BTA_AG_RFC_OPEN_EVT, |
| BTA_AG_RFC_CLOSE_EVT, |
| BTA_AG_RFC_SRV_CLOSE_EVT, |
| BTA_AG_RFC_DATA_EVT, |
| BTA_AG_SCO_OPEN_EVT, |
| BTA_AG_SCO_CLOSE_EVT, |
| BTA_AG_DISC_ACP_RES_EVT, |
| BTA_AG_DISC_INT_RES_EVT, |
| BTA_AG_DISC_OK_EVT, |
| BTA_AG_DISC_FAIL_EVT, |
| BTA_AG_RING_TIMEOUT_EVT, |
| BTA_AG_SVC_TIMEOUT_EVT, |
| BTA_AG_MAX_EVT, |
| }; |
| |
| /* Actions to perform after a SCO event */ |
| enum { |
| BTA_AG_POST_SCO_NONE, /* no action */ |
| BTA_AG_POST_SCO_CLOSE_RFC, /* close RFCOMM channel after SCO closes */ |
| BTA_AG_POST_SCO_RING, /* send RING result code after SCO opens */ |
| BTA_AG_POST_SCO_CALL_CONN, /* send call indicators after SCO opens/closes */ |
| BTA_AG_POST_SCO_CALL_ORIG, /* send call indicators after SCO closes */ |
| BTA_AG_POST_SCO_CALL_END, /* send call indicators after SCO closes */ |
| BTA_AG_POST_SCO_CALL_END_INCALL /* send call indicators for end call & |
| incoming call after SCO closes */ |
| }; |
| |
| /* sco states */ |
| enum { |
| BTA_AG_SCO_SHUTDOWN_ST, /* no sco listening, all sco connections closed */ |
| BTA_AG_SCO_LISTEN_ST, /* sco listening */ |
| BTA_AG_SCO_CODEC_ST, /* sco codec negotiation */ |
| BTA_AG_SCO_OPENING_ST, /* sco connection opening */ |
| BTA_AG_SCO_OPEN_CL_ST, /* opening sco connection being closed */ |
| BTA_AG_SCO_OPEN_XFER_ST, /* opening sco connection being transferred */ |
| BTA_AG_SCO_OPEN_ST, /* sco open */ |
| BTA_AG_SCO_CLOSING_ST, /* sco closing */ |
| BTA_AG_SCO_CLOSE_OP_ST, /* closing sco being opened */ |
| BTA_AG_SCO_CLOSE_XFER_ST, /* closing sco being transferred */ |
| BTA_AG_SCO_SHUTTING_ST /* sco shutting down */ |
| }; |
| |
| /***************************************************************************** |
| * Data types |
| ****************************************************************************/ |
| |
| /* data type for BTA_AG_API_REGISTER_EVT */ |
| typedef struct { |
| char p_name[2][BTA_SERVICE_NAME_LEN + 1]; |
| tBTA_SERVICE_MASK services; |
| tBTA_SEC sec_mask; |
| tBTA_AG_FEAT features; |
| uint8_t app_id; |
| } tBTA_AG_API_REGISTER; |
| |
| /* data type for BTA_AG_API_OPEN_EVT */ |
| typedef struct { |
| RawAddress bd_addr; |
| tBTA_SERVICE_MASK services; |
| tBTA_SEC sec_mask; |
| } tBTA_AG_API_OPEN; |
| |
| /* data type for BTA_AG_API_RESULT_EVT */ |
| typedef struct { |
| tBTA_AG_RES result; |
| tBTA_AG_RES_DATA data; |
| } tBTA_AG_API_RESULT; |
| |
| /* data type for BTA_AG_API_SETCODEC_EVT */ |
| typedef struct { |
| tBTA_AG_PEER_CODEC codec; |
| } tBTA_AG_API_SETCODEC; |
| |
| /* data type for BTA_AG_DISC_RESULT_EVT */ |
| typedef struct { |
| uint16_t status; |
| } tBTA_AG_DISC_RESULT; |
| |
| /* data type for RFCOMM events */ |
| typedef struct { |
| uint16_t port_handle; |
| } tBTA_AG_RFC; |
| |
| /* union of all event datatypes */ |
| union tBTA_AG_DATA { |
| tBTA_AG_API_REGISTER api_register; |
| tBTA_AG_API_OPEN api_open; |
| tBTA_AG_API_RESULT api_result; |
| tBTA_AG_API_SETCODEC api_setcodec; |
| tBTA_AG_DISC_RESULT disc_result; |
| tBTA_AG_RFC rfc; |
| static const tBTA_AG_DATA kEmpty; |
| |
| /** |
| * Check if two tBTA_AG_DATA are equal in memory |
| * |
| * @param rhs other tBTA_AG_DATA |
| * @return true if both unions are equal in memory |
| */ |
| bool operator==(const tBTA_AG_DATA& rhs) const { |
| return (std::memcmp(this, &rhs, sizeof(tBTA_AG_DATA)) == 0); |
| } |
| |
| /** |
| * Check if this union is empty by comparing it to the kEmpty constant |
| * |
| * @return true if this union is empty |
| */ |
| bool IsEmpty() const { return *this == kEmpty; } |
| }; |
| |
| /* type for each profile */ |
| typedef struct { |
| uint32_t sdp_handle; |
| uint8_t scn; |
| } tBTA_AG_PROFILE; |
| |
| typedef enum { |
| BTA_AG_SCO_MSBC_SETTINGS_T2 = 0, /* preferred/default when codec is mSBC */ |
| BTA_AG_SCO_MSBC_SETTINGS_T1, |
| } tBTA_AG_SCO_MSBC_SETTINGS; |
| |
| /* type for each service control block */ |
| struct tBTA_AG_SCB { |
| char clip[BTA_AG_AT_MAX_LEN + 1]; /* number string used for CLIP */ |
| uint16_t serv_handle[BTA_AG_NUM_IDX]; /* RFCOMM server handles */ |
| tBTA_AG_AT_CB at_cb; /* AT command interpreter */ |
| RawAddress peer_addr; /* peer bd address */ |
| tSDP_DISCOVERY_DB* p_disc_db; /* pointer to discovery database */ |
| tBTA_SERVICE_MASK reg_services; /* services specified in register API */ |
| tBTA_SERVICE_MASK open_services; /* services specified in open API */ |
| uint16_t conn_handle; /* RFCOMM handle of connected service */ |
| tBTA_SEC serv_sec_mask; /* server security mask */ |
| tBTA_SEC cli_sec_mask; /* client security mask */ |
| tBTA_AG_FEAT features; /* features registered by application */ |
| tBTA_AG_PEER_FEAT peer_features; /* peer device features */ |
| uint16_t peer_version; /* profile version of peer device */ |
| uint16_t hsp_version; /* HSP profile version before SDP */ |
| uint16_t sco_idx; /* SCO handle */ |
| bool in_use; /* scb in use */ |
| bool dealloc; /* true if service shutting down */ |
| bool clip_enabled; /* set to true if HF enables CLIP reporting */ |
| bool ccwa_enabled; /* set to true if HF enables CCWA reporting */ |
| bool cmer_enabled; /* set to true if HF enables CMER reporting */ |
| bool cmee_enabled; /* set to true if HF enables CME ERROR reporting */ |
| bool inband_enabled; /* set to true if inband ring enabled */ |
| bool svc_conn; /* set to true when service level connection up */ |
| uint8_t state; /* state machine state */ |
| uint8_t conn_service; /* connected service */ |
| uint8_t peer_scn; /* peer scn */ |
| uint8_t app_id; /* application id */ |
| uint8_t role; /* initiator/acceptor role */ |
| uint8_t post_sco; /* action to perform after sco event */ |
| uint8_t call_ind; /* CIEV call indicator value */ |
| uint8_t callsetup_ind; /* CIEV callsetup indicator value */ |
| uint8_t service_ind; /* CIEV service indicator value */ |
| uint8_t signal_ind; /* CIEV signal indicator value */ |
| uint8_t roam_ind; /* CIEV roam indicator value */ |
| uint8_t battchg_ind; /* CIEV battery charge indicator value */ |
| uint8_t callheld_ind; /* CIEV call held indicator value */ |
| uint32_t bia_masked_out; /* indicators HF does not want us to send */ |
| alarm_t* collision_timer; |
| alarm_t* ring_timer; |
| alarm_t* codec_negotiation_timer; |
| tBTA_AG_PEER_CODEC peer_codecs; /* codecs for eSCO supported by the peer */ |
| tBTA_AG_PEER_CODEC sco_codec; /* codec to be used for eSCO connection */ |
| tBTA_AG_PEER_CODEC |
| inuse_codec; /* codec being used for the current SCO connection */ |
| bool codec_updated; /* set to true whenever the app updates codec type */ |
| bool codec_fallback; /* If sco nego fails for mSBC, fallback to CVSD */ |
| tBTA_AG_SCO_MSBC_SETTINGS |
| codec_msbc_settings; /* settings to be used for the impending eSCO */ |
| |
| tBTA_AG_HF_IND |
| peer_hf_indicators[BTA_AG_MAX_NUM_PEER_HF_IND]; /* Peer supported |
| HF indicators */ |
| tBTA_AG_HF_IND |
| local_hf_indicators[BTA_AG_MAX_NUM_LOCAL_HF_IND]; /* Local supported |
| HF indicators */ |
| }; |
| |
| /* type for sco data */ |
| typedef struct { |
| tBTM_ESCO_CONN_REQ_EVT_DATA conn_data; /* SCO data for pending conn request */ |
| tBTA_AG_SCB* p_curr_scb; /* SCB associated with SCO connection */ |
| tBTA_AG_SCB* p_xfer_scb; /* SCB associated with SCO transfer */ |
| uint16_t cur_idx; /* SCO handle */ |
| uint8_t state; /* SCO state variable */ |
| bool is_local; /* SCO connection initiated locally or remotely */ |
| } tBTA_AG_SCO_CB; |
| |
| /* type for AG control block */ |
| typedef struct { |
| tBTA_AG_SCB scb[BTA_AG_MAX_NUM_CLIENTS]; /* service control blocks */ |
| tBTA_AG_PROFILE profile[BTA_AG_NUM_IDX]; /* profile-specific data */ |
| tBTA_AG_SCO_CB sco; /* SCO data */ |
| tBTA_AG_CBACK* p_cback; /* application callback */ |
| } tBTA_AG_CB; |
| |
| /***************************************************************************** |
| * Global data |
| ****************************************************************************/ |
| |
| /* constant lookup tables */ |
| extern const uint16_t bta_ag_uuid[BTA_AG_NUM_IDX]; |
| extern const uint8_t bta_ag_sec_id[BTA_AG_NUM_IDX]; |
| extern const tBTA_AG_AT_CMD* bta_ag_at_tbl[BTA_AG_NUM_IDX]; |
| |
| /* control block declaration */ |
| extern tBTA_AG_CB bta_ag_cb; |
| |
| /* config struct */ |
| extern const tBTA_AG_CFG* p_bta_ag_cfg; |
| extern const tBTA_AG_HF_IND bta_ag_local_hf_ind_cfg[]; |
| |
| /***************************************************************************** |
| * Function prototypes |
| ****************************************************************************/ |
| bool bta_ag_hdl_event(BT_HDR* p_msg); |
| |
| /* API functions */ |
| extern void bta_ag_api_enable(tBTA_AG_CBACK* p_cback); |
| extern void bta_ag_api_disable(); |
| extern void bta_ag_api_set_active_device(const RawAddress& new_active_device); |
| extern void bta_ag_api_register(tBTA_SERVICE_MASK services, tBTA_SEC sec_mask, |
| tBTA_AG_FEAT features, |
| const std::vector<std::string>& service_names, |
| uint8_t app_id); |
| extern void bta_ag_api_result(uint16_t handle, tBTA_AG_RES result, |
| const tBTA_AG_RES_DATA& result_data); |
| |
| /* main functions */ |
| extern void bta_ag_scb_dealloc(tBTA_AG_SCB* p_scb); |
| extern uint16_t bta_ag_scb_to_idx(tBTA_AG_SCB* p_scb); |
| extern tBTA_AG_SCB* bta_ag_scb_by_idx(uint16_t idx); |
| extern uint8_t bta_ag_service_to_idx(tBTA_SERVICE_MASK services); |
| extern uint16_t bta_ag_idx_by_bdaddr(const RawAddress* peer_addr); |
| extern bool bta_ag_other_scb_open(tBTA_AG_SCB* p_curr_scb); |
| extern bool bta_ag_scb_open(tBTA_AG_SCB* p_curr_scb); |
| extern tBTA_AG_SCB* bta_ag_get_other_idle_scb(tBTA_AG_SCB* p_curr_scb); |
| extern void bta_ag_sm_execute(tBTA_AG_SCB* p_scb, uint16_t event, |
| const tBTA_AG_DATA& data); |
| extern void bta_ag_sm_execute_by_handle(uint16_t handle, uint16_t event, |
| const tBTA_AG_DATA& data); |
| extern void bta_ag_collision_cback(tBTA_SYS_CONN_STATUS status, uint8_t id, |
| uint8_t app_id, const RawAddress& peer_addr); |
| extern void bta_ag_resume_open(tBTA_AG_SCB* p_scb); |
| |
| /* SDP functions */ |
| extern bool bta_ag_add_record(uint16_t service_uuid, const char* p_service_name, |
| uint8_t scn, tBTA_AG_FEAT features, |
| uint32_t sdp_handle); |
| extern void bta_ag_create_records(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_del_records(tBTA_AG_SCB* p_scb); |
| extern bool bta_ag_sdp_find_attr(tBTA_AG_SCB* p_scb, tBTA_SERVICE_MASK service); |
| extern void bta_ag_do_disc(tBTA_AG_SCB* p_scb, tBTA_SERVICE_MASK service); |
| extern void bta_ag_free_db(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| |
| /* RFCOMM functions */ |
| extern void bta_ag_start_servers(tBTA_AG_SCB* p_scb, |
| tBTA_SERVICE_MASK services); |
| extern void bta_ag_close_servers(tBTA_AG_SCB* p_scb, |
| tBTA_SERVICE_MASK services); |
| extern bool bta_ag_is_server_closed(tBTA_AG_SCB* p_scb); |
| extern void bta_ag_rfc_do_close(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_rfc_do_open(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| |
| /* SCO functions */ |
| extern bool bta_ag_sco_is_active_device(const RawAddress& bd_addr); |
| extern bool bta_ag_sco_is_open(tBTA_AG_SCB* p_scb); |
| extern bool bta_ag_sco_is_opening(tBTA_AG_SCB* p_scb); |
| extern void bta_ag_sco_conn_rsp(tBTA_AG_SCB* p_scb, |
| tBTM_ESCO_CONN_REQ_EVT_DATA* data); |
| |
| /* AT command functions */ |
| extern void bta_ag_at_hsp_cback(tBTA_AG_SCB* p_scb, uint16_t cmd, |
| uint8_t arg_type, char* p_arg, int16_t int_arg); |
| extern void bta_ag_at_hfp_cback(tBTA_AG_SCB* p_scb, uint16_t cmd, |
| uint8_t arg_type, char* p_arg, int16_t int_arg); |
| extern void bta_ag_at_err_cback(tBTA_AG_SCB* p_scb, bool unknown, |
| const char* p_arg); |
| extern bool bta_ag_inband_enabled(tBTA_AG_SCB* p_scb); |
| extern void bta_ag_send_call_inds(tBTA_AG_SCB* p_scb, tBTA_AG_RES result); |
| |
| /* Action functions */ |
| extern void bta_ag_register(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_deregister(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_start_dereg(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_start_close(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_start_open(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_disc_int_res(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_disc_acp_res(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_disc_fail(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_open_fail(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_rfc_fail(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_rfc_close(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_rfc_open(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_rfc_acp_open(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_rfc_data(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_sco_listen(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_sco_open(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_sco_close(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_sco_shutdown(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_sco_conn_open(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_sco_conn_close(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_post_sco_open(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_post_sco_close(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_svc_conn_open(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_result(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_setcodec(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| extern void bta_ag_send_ring(tBTA_AG_SCB* p_scb, const tBTA_AG_DATA& data); |
| |
| /* Internal utility functions */ |
| extern void bta_ag_sco_codec_nego(tBTA_AG_SCB* p_scb, bool result); |
| extern void bta_ag_codec_negotiate(tBTA_AG_SCB* p_scb); |
| extern void bta_ag_send_bcs(tBTA_AG_SCB* p_scb); |
| extern void bta_ag_set_sco_allowed(bool value); |
| extern const RawAddress& bta_ag_get_active_device(); |
| extern void bta_clear_active_device(); |
| |
| #endif /* BTA_AG_INT_H */ |