13deb3ec6SMatthias Ringwald /* 23deb3ec6SMatthias Ringwald * Copyright (C) 2014 BlueKitchen GmbH 33deb3ec6SMatthias Ringwald * 43deb3ec6SMatthias Ringwald * Redistribution and use in source and binary forms, with or without 53deb3ec6SMatthias Ringwald * modification, are permitted provided that the following conditions 63deb3ec6SMatthias Ringwald * are met: 73deb3ec6SMatthias Ringwald * 83deb3ec6SMatthias Ringwald * 1. Redistributions of source code must retain the above copyright 93deb3ec6SMatthias Ringwald * notice, this list of conditions and the following disclaimer. 103deb3ec6SMatthias Ringwald * 2. Redistributions in binary form must reproduce the above copyright 113deb3ec6SMatthias Ringwald * notice, this list of conditions and the following disclaimer in the 123deb3ec6SMatthias Ringwald * documentation and/or other materials provided with the distribution. 133deb3ec6SMatthias Ringwald * 3. Neither the name of the copyright holders nor the names of 143deb3ec6SMatthias Ringwald * contributors may be used to endorse or promote products derived 153deb3ec6SMatthias Ringwald * from this software without specific prior written permission. 163deb3ec6SMatthias Ringwald * 4. Any redistribution, use, or modification is done solely for 173deb3ec6SMatthias Ringwald * personal benefit and not for any commercial purpose or for 183deb3ec6SMatthias Ringwald * monetary gain. 193deb3ec6SMatthias Ringwald * 203deb3ec6SMatthias Ringwald * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS 213deb3ec6SMatthias Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 223deb3ec6SMatthias Ringwald * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 232fca4dadSMilanka Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL BLUEKITCHEN 242fca4dadSMilanka Ringwald * GMBH OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 253deb3ec6SMatthias Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 263deb3ec6SMatthias Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS 273deb3ec6SMatthias Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 283deb3ec6SMatthias Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 293deb3ec6SMatthias Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF 303deb3ec6SMatthias Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 313deb3ec6SMatthias Ringwald * SUCH DAMAGE. 323deb3ec6SMatthias Ringwald * 333deb3ec6SMatthias Ringwald * Please inquire about commercial licensing options at 343deb3ec6SMatthias Ringwald * [email protected] 353deb3ec6SMatthias Ringwald * 363deb3ec6SMatthias Ringwald */ 37ab2c6ae4SMatthias Ringwald 38e501bae0SMatthias Ringwald #define BTSTACK_FILE__ "hfp_hf.c" 393deb3ec6SMatthias Ringwald 403deb3ec6SMatthias Ringwald // ***************************************************************************** 413deb3ec6SMatthias Ringwald // 42fffdd288SMatthias Ringwald // HFP Hands-Free (HF) unit 433deb3ec6SMatthias Ringwald // 443deb3ec6SMatthias Ringwald // ***************************************************************************** 453deb3ec6SMatthias Ringwald 467907f069SMatthias Ringwald #include "btstack_config.h" 473deb3ec6SMatthias Ringwald 483deb3ec6SMatthias Ringwald #include <stdint.h> 493cfa4086SMatthias Ringwald #include <stdio.h> 503deb3ec6SMatthias Ringwald #include <string.h> 513deb3ec6SMatthias Ringwald 52235946f1SMatthias Ringwald #include "bluetooth_sdp.h" 5359c6af15SMatthias Ringwald #include "btstack_debug.h" 54d4dd47ffSMatthias Ringwald #include "btstack_event.h" 553deb3ec6SMatthias Ringwald #include "btstack_memory.h" 5659c6af15SMatthias Ringwald #include "btstack_run_loop.h" 5759c6af15SMatthias Ringwald #include "classic/core.h" 5859c6af15SMatthias Ringwald #include "classic/hfp.h" 5959c6af15SMatthias Ringwald #include "classic/hfp_hf.h" 60efda0b48SMatthias Ringwald #include "classic/sdp_client_rfcomm.h" 61746ccb7eSMatthias Ringwald #include "classic/sdp_server.h" 62023f2764SMatthias Ringwald #include "classic/sdp_util.h" 6359c6af15SMatthias Ringwald #include "hci.h" 6459c6af15SMatthias Ringwald #include "hci_cmd.h" 6559c6af15SMatthias Ringwald #include "hci_dump.h" 6659c6af15SMatthias Ringwald #include "l2cap.h" 673deb3ec6SMatthias Ringwald 6820b2edb6SMatthias Ringwald // const 69aeb0f0feSMatthias Ringwald static const char hfp_hf_default_service_name[] = "Hands-Free unit"; 7020b2edb6SMatthias Ringwald 7120b2edb6SMatthias Ringwald // globals 72aeb0f0feSMatthias Ringwald 73aeb0f0feSMatthias Ringwald // higher layer callbacks 74aeb0f0feSMatthias Ringwald static btstack_packet_handler_t hfp_hf_callback; 75aeb0f0feSMatthias Ringwald 761c6a0fc0SMatthias Ringwald static btstack_packet_callback_registration_t hfp_hf_hci_event_callback_registration; 7727950165SMatthias Ringwald 78aeb0f0feSMatthias Ringwald static uint16_t hfp_hf_supported_features; 79aeb0f0feSMatthias Ringwald static uint8_t hfp_hf_codecs_nr; 80aeb0f0feSMatthias Ringwald static uint8_t hfp_hf_codecs[HFP_MAX_NUM_CODECS]; 813deb3ec6SMatthias Ringwald 82aeb0f0feSMatthias Ringwald static uint8_t hfp_hf_indicators_nr; 83aeb0f0feSMatthias Ringwald static uint8_t hfp_hf_indicators[HFP_MAX_NUM_INDICATORS]; 84aeb0f0feSMatthias Ringwald static uint32_t hfp_hf_indicators_value[HFP_MAX_NUM_INDICATORS]; 85667ec068SMatthias Ringwald 8620b2edb6SMatthias Ringwald static uint8_t hfp_hf_speaker_gain; 8720b2edb6SMatthias Ringwald static uint8_t hfp_hf_microphone_gain; 883deb3ec6SMatthias Ringwald 89aeb0f0feSMatthias Ringwald static hfp_call_status_t hfp_hf_call_status; 90aeb0f0feSMatthias Ringwald static hfp_callsetup_status_t hfp_hf_callsetup_status; 91aeb0f0feSMatthias Ringwald static hfp_callheld_status_t hfp_hf_callheld_status; 923deb3ec6SMatthias Ringwald 93aeb0f0feSMatthias Ringwald static char hfp_hf_phone_number[25]; 94ce263fc8SMatthias Ringwald 95ce263fc8SMatthias Ringwald 9676cc1527SMatthias Ringwald static int has_codec_negotiation_feature(hfp_connection_t * hfp_connection){ 97aeb0f0feSMatthias Ringwald int hf = get_bit(hfp_hf_supported_features, HFP_HFSF_CODEC_NEGOTIATION); 9876cc1527SMatthias Ringwald int ag = get_bit(hfp_connection->remote_supported_features, HFP_AGSF_CODEC_NEGOTIATION); 9976cc1527SMatthias Ringwald return hf && ag; 10076cc1527SMatthias Ringwald } 10176cc1527SMatthias Ringwald 10276cc1527SMatthias Ringwald static int has_call_waiting_and_3way_calling_feature(hfp_connection_t * hfp_connection){ 103aeb0f0feSMatthias Ringwald int hf = get_bit(hfp_hf_supported_features, HFP_HFSF_THREE_WAY_CALLING); 10476cc1527SMatthias Ringwald int ag = get_bit(hfp_connection->remote_supported_features, HFP_AGSF_THREE_WAY_CALLING); 10576cc1527SMatthias Ringwald return hf && ag; 10676cc1527SMatthias Ringwald } 10776cc1527SMatthias Ringwald 10876cc1527SMatthias Ringwald 10976cc1527SMatthias Ringwald static int has_hf_indicators_feature(hfp_connection_t * hfp_connection){ 110aeb0f0feSMatthias Ringwald int hf = get_bit(hfp_hf_supported_features, HFP_HFSF_HF_INDICATORS); 11176cc1527SMatthias Ringwald int ag = get_bit(hfp_connection->remote_supported_features, HFP_AGSF_HF_INDICATORS); 11276cc1527SMatthias Ringwald return hf && ag; 11376cc1527SMatthias Ringwald } 11476cc1527SMatthias Ringwald 115fcf4ede6SMilanka Ringwald static bool hfp_hf_vra_flag_supported(hfp_connection_t * hfp_connection){ 116fcf4ede6SMilanka Ringwald int hf = get_bit(hfp_hf_supported_features, HFP_HFSF_VOICE_RECOGNITION_FUNCTION); 117fcf4ede6SMilanka Ringwald int ag = get_bit(hfp_connection->remote_supported_features, HFP_AGSF_VOICE_RECOGNITION_FUNCTION); 118fcf4ede6SMilanka Ringwald return hf && ag; 119fcf4ede6SMilanka Ringwald } 120fcf4ede6SMilanka Ringwald 121fcf4ede6SMilanka Ringwald static bool hfp_hf_enhanced_vra_flag_supported(hfp_connection_t * hfp_connection){ 122fcf4ede6SMilanka Ringwald int hf = get_bit(hfp_hf_supported_features, HFP_HFSF_ENHANCED_VOICE_RECOGNITION_STATUS); 123fcf4ede6SMilanka Ringwald int ag = get_bit(hfp_connection->remote_supported_features, HFP_AGSF_ENHANCED_VOICE_RECOGNITION_STATUS); 124fcf4ede6SMilanka Ringwald return hf && ag; 125fcf4ede6SMilanka Ringwald } 12676cc1527SMatthias Ringwald 1279c9c64c1SMatthias Ringwald static hfp_connection_t * get_hfp_hf_connection_context_for_acl_handle(uint16_t handle){ 1289c9c64c1SMatthias Ringwald btstack_linked_list_iterator_t it; 1299c9c64c1SMatthias Ringwald btstack_linked_list_iterator_init(&it, hfp_get_connections()); 1309c9c64c1SMatthias Ringwald while (btstack_linked_list_iterator_has_next(&it)){ 1319c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = (hfp_connection_t *)btstack_linked_list_iterator_next(&it); 1329c9c64c1SMatthias Ringwald if (hfp_connection->acl_handle != handle) continue; 1339c9c64c1SMatthias Ringwald if (hfp_connection->local_role != HFP_ROLE_HF) continue; 1349c9c64c1SMatthias Ringwald return hfp_connection; 1359c9c64c1SMatthias Ringwald } 1369c9c64c1SMatthias Ringwald return NULL; 1379c9c64c1SMatthias Ringwald } 1389c9c64c1SMatthias Ringwald 139c10fde09SMatthias Ringwald /* emit functions */ 1403deb3ec6SMatthias Ringwald 141a473a009SMatthias Ringwald static void hfp_hf_emit_subscriber_information(const hfp_connection_t * hfp_connection, uint8_t status){ 142a473a009SMatthias Ringwald if (hfp_hf_callback == NULL) return; 143c10fde09SMatthias Ringwald uint16_t bnip_number_len = btstack_min(strlen(hfp_connection->bnip_number), sizeof(hfp_connection->bnip_number)-1); 144c10fde09SMatthias Ringwald uint8_t event[7 + sizeof(hfp_connection->bnip_number)]; 145a0ffb263SMatthias Ringwald event[0] = HCI_EVENT_HFP_META; 146c10fde09SMatthias Ringwald event[1] = 6 + bnip_number_len; 147a473a009SMatthias Ringwald event[2] = HFP_SUBEVENT_SUBSCRIBER_NUMBER_INFORMATION; 148d703d377SMatthias Ringwald little_endian_store_16(event, 3, hfp_connection->acl_handle); 149d703d377SMatthias Ringwald event[5] = status; 150d703d377SMatthias Ringwald event[6] = hfp_connection->bnip_type; 151c10fde09SMatthias Ringwald memcpy(&event[7], hfp_connection->bnip_number, bnip_number_len); 152c10fde09SMatthias Ringwald event[7 + bnip_number_len] = 0; 153c10fde09SMatthias Ringwald (*hfp_hf_callback)(HCI_EVENT_PACKET, 0, event, 8 + bnip_number_len); 154a0ffb263SMatthias Ringwald } 155a0ffb263SMatthias Ringwald 156598d4936SMatthias Ringwald static void hfp_hf_emit_type_number_alpha(const hfp_connection_t * hfp_connection, uint8_t event_subtype){ 157a473a009SMatthias Ringwald if (hfp_hf_callback == NULL) return; 158c10fde09SMatthias Ringwald uint16_t bnip_number_len = btstack_min(strlen(hfp_connection->bnip_number), sizeof(hfp_connection->bnip_number)-1); 159598d4936SMatthias Ringwald // 10 fixed - 1 (bnip_number_len <= sizeof(hfp_connection->bnip_number)-1) + 1 (trailing \0 for line buffer) 160598d4936SMatthias Ringwald uint8_t event[10 + sizeof(hfp_connection->bnip_number) + sizeof(hfp_connection->line_buffer)]; 161598d4936SMatthias Ringwald uint8_t alpha_len = hfp_connection->clip_have_alpha ? strlen((const char *) hfp_connection->line_buffer) : 0; 162598d4936SMatthias Ringwald uint8_t pos = 0; 163598d4936SMatthias Ringwald event[pos++] = HCI_EVENT_HFP_META; 164598d4936SMatthias Ringwald event[pos++] = 8 + bnip_number_len + alpha_len; 165598d4936SMatthias Ringwald event[pos++] = event_subtype; 166d703d377SMatthias Ringwald little_endian_store_16(event, 3, hfp_connection->acl_handle); 167598d4936SMatthias Ringwald pos += 2; 168598d4936SMatthias Ringwald event[pos++] = hfp_connection->bnip_type; 169598d4936SMatthias Ringwald event[pos++] = bnip_number_len; 170598d4936SMatthias Ringwald memcpy(&event[7], hfp_connection->bnip_number, bnip_number_len); 171598d4936SMatthias Ringwald pos += bnip_number_len; 172598d4936SMatthias Ringwald event[pos++] = 0; 173598d4936SMatthias Ringwald event[pos++] = alpha_len; 174598d4936SMatthias Ringwald memcpy(&event[pos], hfp_connection->line_buffer, alpha_len); 175598d4936SMatthias Ringwald pos += alpha_len; 176598d4936SMatthias Ringwald event[pos++] = 0; 177598d4936SMatthias Ringwald (*hfp_hf_callback)(HCI_EVENT_PACKET, 0, event, pos); 178a0ffb263SMatthias Ringwald } 179a0ffb263SMatthias Ringwald 180a473a009SMatthias Ringwald static void hfp_hf_emit_enhanced_call_status(const hfp_connection_t * hfp_connection){ 181a473a009SMatthias Ringwald if (hfp_hf_callback == NULL) return; 182c10fde09SMatthias Ringwald uint16_t bnip_number_len = btstack_min(strlen(hfp_connection->bnip_number), sizeof(hfp_connection->bnip_number)-1); 183c10fde09SMatthias Ringwald uint8_t event[11 + sizeof(hfp_connection->bnip_number)]; 184c10fde09SMatthias Ringwald event[0] = HCI_EVENT_HFP_META; 185c10fde09SMatthias Ringwald event[1] = 10 + bnip_number_len; 186c10fde09SMatthias Ringwald event[2] = HFP_SUBEVENT_ENHANCED_CALL_STATUS; 187c10fde09SMatthias Ringwald little_endian_store_16(event, 3, hfp_connection->acl_handle); 188c10fde09SMatthias Ringwald event[4] = hfp_connection->clcc_idx; 189c10fde09SMatthias Ringwald event[5] = hfp_connection->clcc_dir; 190c10fde09SMatthias Ringwald event[6] = hfp_connection->clcc_status; 191c10fde09SMatthias Ringwald event[7] = hfp_connection->clcc_mode; 192c10fde09SMatthias Ringwald event[8] = hfp_connection->clcc_mpty; 193c10fde09SMatthias Ringwald event[9] = hfp_connection->bnip_type; 194c10fde09SMatthias Ringwald memcpy(&event[10], hfp_connection->bnip_number, bnip_number_len); 195c10fde09SMatthias Ringwald event[11+bnip_number_len] = 0; 196c10fde09SMatthias Ringwald (*hfp_hf_callback)(HCI_EVENT_PACKET, 0, event, 12+bnip_number_len); 197a0ffb263SMatthias Ringwald } 198a0ffb263SMatthias Ringwald 1991ac1f60fSMilanka Ringwald static void hfp_emit_ag_indicator_mapping_event(const hfp_connection_t * hfp_connection, const hfp_ag_indicator_t * indicator){ 2001ac1f60fSMilanka Ringwald if (hfp_hf_callback == NULL) return; 201c10fde09SMatthias Ringwald uint8_t event[8 + HFP_MAX_INDICATOR_DESC_SIZE]; 202c10fde09SMatthias Ringwald uint16_t indicator_len = btstack_min(strlen(indicator->name), HFP_MAX_INDICATOR_DESC_SIZE-1); 203c10fde09SMatthias Ringwald event[0] = HCI_EVENT_HFP_META; 204c10fde09SMatthias Ringwald event[1] = 7 + indicator_len; 205c10fde09SMatthias Ringwald event[2] = HFP_SUBEVENT_AG_INDICATOR_MAPPING; 206c10fde09SMatthias Ringwald little_endian_store_16(event, 3, hfp_connection->acl_handle); 207c10fde09SMatthias Ringwald event[5] = indicator->index; 208c10fde09SMatthias Ringwald event[6] = indicator->min_range; 209c10fde09SMatthias Ringwald event[7] = indicator->max_range; 210c10fde09SMatthias Ringwald memcpy(&event[8], indicator->name, indicator_len); 211c10fde09SMatthias Ringwald event[8+indicator_len] = 0; 212c10fde09SMatthias Ringwald (*hfp_hf_callback)(HCI_EVENT_PACKET, 0, event, 9 + indicator_len); 2131ac1f60fSMilanka Ringwald } 21476cc1527SMatthias Ringwald 215ce3797e1SMilanka Ringwald static void hfp_emit_ag_indicator_status_event(const hfp_connection_t * hfp_connection, const hfp_ag_indicator_t * indicator){ 216a473a009SMatthias Ringwald if (hfp_hf_callback == NULL) return; 217c10fde09SMatthias Ringwald uint8_t event[12+HFP_MAX_INDICATOR_DESC_SIZE]; 218c10fde09SMatthias Ringwald uint16_t indicator_len = btstack_min(strlen(indicator->name), HFP_MAX_INDICATOR_DESC_SIZE-1); 219c10fde09SMatthias Ringwald event[0] = HCI_EVENT_HFP_META; 220c10fde09SMatthias Ringwald event[1] = 11 + indicator_len; 221c10fde09SMatthias Ringwald event[2] = HFP_SUBEVENT_AG_INDICATOR_STATUS_CHANGED; 222c10fde09SMatthias Ringwald little_endian_store_16(event, 3, hfp_connection->acl_handle); 223c10fde09SMatthias Ringwald event[5] = indicator->index; 224c10fde09SMatthias Ringwald event[6] = indicator->status; 225c10fde09SMatthias Ringwald event[7] = indicator->min_range; 226c10fde09SMatthias Ringwald event[8] = indicator->max_range; 227c10fde09SMatthias Ringwald event[9] = indicator->mandatory; 228c10fde09SMatthias Ringwald event[10] = indicator->enabled; 229c10fde09SMatthias Ringwald event[11] = indicator->status_changed; 230c10fde09SMatthias Ringwald memcpy(&event[12], indicator->name, indicator_len); 231c10fde09SMatthias Ringwald event[12+indicator_len] = 0; 232c10fde09SMatthias Ringwald (*hfp_hf_callback)(HCI_EVENT_PACKET, 0, event, 13 + indicator_len); 2333deb3ec6SMatthias Ringwald } 2343deb3ec6SMatthias Ringwald 235a473a009SMatthias Ringwald static void hfp_emit_network_operator_event(const hfp_connection_t * hfp_connection){ 236a473a009SMatthias Ringwald if (hfp_hf_callback == NULL) return; 237c10fde09SMatthias Ringwald uint16_t operator_len = btstack_min(strlen(hfp_connection->network_operator.name), HFP_MAX_NETWORK_OPERATOR_NAME_SIZE-1); 238c10fde09SMatthias Ringwald uint8_t event[7+HFP_MAX_NETWORK_OPERATOR_NAME_SIZE]; 23976cc1527SMatthias Ringwald event[0] = HCI_EVENT_HFP_META; 24076cc1527SMatthias Ringwald event[1] = sizeof(event) - 2; 241c10fde09SMatthias Ringwald event[2] = 6 + operator_len; 242d703d377SMatthias Ringwald little_endian_store_16(event, 3, hfp_connection->acl_handle); 243d703d377SMatthias Ringwald event[5] = hfp_connection->network_operator.mode; 244d703d377SMatthias Ringwald event[6] = hfp_connection->network_operator.format; 245c10fde09SMatthias Ringwald memcpy(&event[7], hfp_connection->network_operator.name, operator_len); 246c10fde09SMatthias Ringwald event[7+operator_len] = 0; 247c10fde09SMatthias Ringwald (*hfp_hf_callback)(HCI_EVENT_PACKET, 0, event, 8 + operator_len); 2483deb3ec6SMatthias Ringwald } 2493deb3ec6SMatthias Ringwald 250b95cac54SMilanka Ringwald 251b95cac54SMilanka Ringwald static void hfp_hf_emit_enhanced_voice_recognition_text(hfp_connection_t * hfp_connection){ 252b95cac54SMilanka Ringwald btstack_assert(hfp_connection != NULL); 25351aa5d5aSMilanka Ringwald uint8_t event[HFP_MAX_VR_TEXT_SIZE + 11]; 254b95cac54SMilanka Ringwald int pos = 0; 255b95cac54SMilanka Ringwald event[pos++] = HCI_EVENT_HFP_META; 256b95cac54SMilanka Ringwald event[pos++] = sizeof(event) - 2; 257b95cac54SMilanka Ringwald event[pos++] = HFP_SUBEVENT_ENHANCED_VOICE_RECOGNITION_AG_MESSAGE; 258b95cac54SMilanka Ringwald little_endian_store_16(event, pos, hfp_connection->acl_handle); 259b95cac54SMilanka Ringwald pos += 2; 260b95cac54SMilanka Ringwald little_endian_store_16(event, pos, hfp_connection->ag_msg.text_id); 261b95cac54SMilanka Ringwald pos += 2; 262b95cac54SMilanka Ringwald event[pos++] = hfp_connection->ag_msg.text_type; 263e83f1be7SMilanka Ringwald event[pos++] = hfp_connection->ag_msg.text_operation; 264b95cac54SMilanka Ringwald 26551aa5d5aSMilanka Ringwald // length, zero ending is already in message 266b95cac54SMilanka Ringwald uint8_t * value = &hfp_connection->line_buffer[0]; 26751aa5d5aSMilanka Ringwald uint16_t value_length = hfp_connection->ag_vra_msg_length; 26851aa5d5aSMilanka Ringwald 26951aa5d5aSMilanka Ringwald little_endian_store_16(event, pos, value_length); 270b95cac54SMilanka Ringwald pos += 2; 27151aa5d5aSMilanka Ringwald memcpy(&event[pos], value, value_length); 27251aa5d5aSMilanka Ringwald pos += value_length; 27351aa5d5aSMilanka Ringwald 274b95cac54SMilanka Ringwald (*hfp_hf_callback)(HCI_EVENT_PACKET, 0, event, pos); 275b95cac54SMilanka Ringwald } 276b95cac54SMilanka Ringwald 27776cc1527SMatthias Ringwald /* send commands */ 27889425bfcSMilanka Ringwald 27989425bfcSMilanka Ringwald static inline int hfp_hf_send_cmd(uint16_t cid, const char * cmd){ 2803deb3ec6SMatthias Ringwald char buffer[20]; 2811599fe57SMatthias Ringwald snprintf(buffer, sizeof(buffer), "AT%s\r", cmd); 28289425bfcSMilanka Ringwald return send_str_over_rfcomm(cid, buffer); 28389425bfcSMilanka Ringwald } 28489425bfcSMilanka Ringwald 28589425bfcSMilanka Ringwald static inline int hfp_hf_send_cmd_with_mark(uint16_t cid, const char * cmd, const char * mark){ 28689425bfcSMilanka Ringwald char buffer[20]; 2871599fe57SMatthias Ringwald snprintf(buffer, sizeof(buffer), "AT%s%s\r", cmd, mark); 28889425bfcSMilanka Ringwald return send_str_over_rfcomm(cid, buffer); 28989425bfcSMilanka Ringwald } 29089425bfcSMilanka Ringwald 29186da9d74SMatthias Ringwald static inline int hfp_hf_send_cmd_with_int(uint16_t cid, const char * cmd, uint16_t value){ 29289425bfcSMilanka Ringwald char buffer[40]; 2931599fe57SMatthias Ringwald snprintf(buffer, sizeof(buffer), "AT%s=%d\r", cmd, value); 2943deb3ec6SMatthias Ringwald return send_str_over_rfcomm(cid, buffer); 2953deb3ec6SMatthias Ringwald } 2963deb3ec6SMatthias Ringwald 2973deb3ec6SMatthias Ringwald static int hfp_hf_cmd_notify_on_codecs(uint16_t cid){ 2983deb3ec6SMatthias Ringwald char buffer[30]; 29989425bfcSMilanka Ringwald const int size = sizeof(buffer); 30089425bfcSMilanka Ringwald int offset = snprintf(buffer, size, "AT%s=", HFP_AVAILABLE_CODECS); 301aeb0f0feSMatthias Ringwald offset += join(buffer+offset, size-offset, hfp_hf_codecs, hfp_hf_codecs_nr); 3021599fe57SMatthias Ringwald offset += snprintf(buffer+offset, size-offset, "\r"); 3033deb3ec6SMatthias Ringwald return send_str_over_rfcomm(cid, buffer); 3043deb3ec6SMatthias Ringwald } 3053deb3ec6SMatthias Ringwald 3063deb3ec6SMatthias Ringwald static int hfp_hf_cmd_activate_status_update_for_ag_indicator(uint16_t cid, uint32_t indicators_status, int indicators_nr){ 3073deb3ec6SMatthias Ringwald char buffer[50]; 30889425bfcSMilanka Ringwald const int size = sizeof(buffer); 30989425bfcSMilanka Ringwald int offset = snprintf(buffer, size, "AT%s=", HFP_UPDATE_ENABLE_STATUS_FOR_INDIVIDUAL_AG_INDICATORS); 31089425bfcSMilanka Ringwald offset += join_bitmap(buffer+offset, size-offset, indicators_status, indicators_nr); 3111599fe57SMatthias Ringwald offset += snprintf(buffer+offset, size-offset, "\r"); 3123deb3ec6SMatthias Ringwald return send_str_over_rfcomm(cid, buffer); 3133deb3ec6SMatthias Ringwald } 3143deb3ec6SMatthias Ringwald 3153deb3ec6SMatthias Ringwald static int hfp_hf_cmd_list_supported_generic_status_indicators(uint16_t cid){ 3163deb3ec6SMatthias Ringwald char buffer[30]; 31789425bfcSMilanka Ringwald const int size = sizeof(buffer); 31889425bfcSMilanka Ringwald int offset = snprintf(buffer, size, "AT%s=", HFP_GENERIC_STATUS_INDICATOR); 319aeb0f0feSMatthias Ringwald offset += join(buffer+offset, size-offset, hfp_hf_indicators, hfp_hf_indicators_nr); 3201599fe57SMatthias Ringwald offset += snprintf(buffer+offset, size-offset, "\r"); 3213deb3ec6SMatthias Ringwald return send_str_over_rfcomm(cid, buffer); 3223deb3ec6SMatthias Ringwald } 3233deb3ec6SMatthias Ringwald 32489425bfcSMilanka Ringwald static int hfp_hf_cmd_activate_status_update_for_all_ag_indicators(uint16_t cid, uint8_t activate){ 3253deb3ec6SMatthias Ringwald char buffer[20]; 3261599fe57SMatthias Ringwald snprintf(buffer, sizeof(buffer), "AT%s=3,0,0,%d\r", HFP_ENABLE_STATUS_UPDATE_FOR_AG_INDICATORS, activate); 327ce263fc8SMatthias Ringwald return send_str_over_rfcomm(cid, buffer); 328ce263fc8SMatthias Ringwald } 329ce263fc8SMatthias Ringwald 330ce263fc8SMatthias Ringwald static int hfp_hf_initiate_outgoing_call_cmd(uint16_t cid){ 331ce263fc8SMatthias Ringwald char buffer[40]; 332aeb0f0feSMatthias Ringwald snprintf(buffer, sizeof(buffer), "%s%s;\r", HFP_CALL_PHONE_NUMBER, hfp_hf_phone_number); 333ce263fc8SMatthias Ringwald return send_str_over_rfcomm(cid, buffer); 334ce263fc8SMatthias Ringwald } 335ce263fc8SMatthias Ringwald 336a0ffb263SMatthias Ringwald static int hfp_hf_send_memory_dial_cmd(uint16_t cid, int memory_id){ 337ce263fc8SMatthias Ringwald char buffer[40]; 3381599fe57SMatthias Ringwald snprintf(buffer, sizeof(buffer), "%s>%d;\r", HFP_CALL_PHONE_NUMBER, memory_id); 339ce263fc8SMatthias Ringwald return send_str_over_rfcomm(cid, buffer); 340ce263fc8SMatthias Ringwald } 341ce263fc8SMatthias Ringwald 342f04a0c31SMatthias Ringwald static int hfp_hf_send_chld(uint16_t cid, unsigned int number){ 34389425bfcSMilanka Ringwald char buffer[40]; 3441599fe57SMatthias Ringwald snprintf(buffer, sizeof(buffer), "AT%s=%u\r", HFP_SUPPORT_CALL_HOLD_AND_MULTIPARTY_SERVICES, number); 345ce263fc8SMatthias Ringwald return send_str_over_rfcomm(cid, buffer); 346ce263fc8SMatthias Ringwald } 347ce263fc8SMatthias Ringwald 348ce263fc8SMatthias Ringwald static int hfp_hf_send_dtmf(uint16_t cid, char code){ 349ce263fc8SMatthias Ringwald char buffer[20]; 3501599fe57SMatthias Ringwald snprintf(buffer, sizeof(buffer), "AT%s=%c\r", HFP_TRANSMIT_DTMF_CODES, code); 351ce263fc8SMatthias Ringwald return send_str_over_rfcomm(cid, buffer); 352ce263fc8SMatthias Ringwald } 353ce263fc8SMatthias Ringwald 35497d2cadbSMatthias Ringwald static int hfp_hf_cmd_ata(uint16_t cid){ 3551599fe57SMatthias Ringwald return send_str_over_rfcomm(cid, (char *) "ATA\r"); 35697d2cadbSMatthias Ringwald } 35797d2cadbSMatthias Ringwald 35889425bfcSMilanka Ringwald static int hfp_hf_cmd_exchange_supported_features(uint16_t cid){ 359aeb0f0feSMatthias Ringwald return hfp_hf_send_cmd_with_int(cid, HFP_SUPPORTED_FEATURES, hfp_hf_supported_features); 36089425bfcSMilanka Ringwald } 36189425bfcSMilanka Ringwald 36289425bfcSMilanka Ringwald static int hfp_hf_cmd_retrieve_indicators(uint16_t cid){ 36389425bfcSMilanka Ringwald return hfp_hf_send_cmd_with_mark(cid, HFP_INDICATOR, "=?"); 36489425bfcSMilanka Ringwald } 36589425bfcSMilanka Ringwald 36689425bfcSMilanka Ringwald static int hfp_hf_cmd_retrieve_indicators_status(uint16_t cid){ 36789425bfcSMilanka Ringwald return hfp_hf_send_cmd_with_mark(cid, HFP_INDICATOR, "?"); 36889425bfcSMilanka Ringwald } 36989425bfcSMilanka Ringwald 37089425bfcSMilanka Ringwald static int hfp_hf_cmd_retrieve_can_hold_call(uint16_t cid){ 37189425bfcSMilanka Ringwald return hfp_hf_send_cmd_with_mark(cid, HFP_SUPPORT_CALL_HOLD_AND_MULTIPARTY_SERVICES, "=?"); 37289425bfcSMilanka Ringwald } 37389425bfcSMilanka Ringwald 37489425bfcSMilanka Ringwald static int hfp_hf_cmd_retrieve_supported_generic_status_indicators(uint16_t cid){ 37589425bfcSMilanka Ringwald return hfp_hf_send_cmd_with_mark(cid, HFP_GENERIC_STATUS_INDICATOR, "=?"); 37689425bfcSMilanka Ringwald } 37789425bfcSMilanka Ringwald 37889425bfcSMilanka Ringwald static int hfp_hf_cmd_list_initital_supported_generic_status_indicators(uint16_t cid){ 37989425bfcSMilanka Ringwald return hfp_hf_send_cmd_with_mark(cid, HFP_GENERIC_STATUS_INDICATOR, "?"); 38089425bfcSMilanka Ringwald } 38189425bfcSMilanka Ringwald 38289425bfcSMilanka Ringwald static int hfp_hf_cmd_query_operator_name_format(uint16_t cid){ 38389425bfcSMilanka Ringwald return hfp_hf_send_cmd_with_mark(cid, HFP_QUERY_OPERATOR_SELECTION, "=3,0"); 38489425bfcSMilanka Ringwald } 38589425bfcSMilanka Ringwald 38689425bfcSMilanka Ringwald static int hfp_hf_cmd_query_operator_name(uint16_t cid){ 38789425bfcSMilanka Ringwald return hfp_hf_send_cmd_with_mark(cid, HFP_QUERY_OPERATOR_SELECTION, "?"); 38889425bfcSMilanka Ringwald } 38989425bfcSMilanka Ringwald 39089425bfcSMilanka Ringwald static int hfp_hf_cmd_trigger_codec_connection_setup(uint16_t cid){ 39189425bfcSMilanka Ringwald return hfp_hf_send_cmd(cid, HFP_TRIGGER_CODEC_CONNECTION_SETUP); 39289425bfcSMilanka Ringwald } 39389425bfcSMilanka Ringwald 39489425bfcSMilanka Ringwald static int hfp_hf_set_microphone_gain_cmd(uint16_t cid, int gain){ 39589425bfcSMilanka Ringwald return hfp_hf_send_cmd_with_int(cid, HFP_SET_MICROPHONE_GAIN, gain); 39689425bfcSMilanka Ringwald } 39789425bfcSMilanka Ringwald 39889425bfcSMilanka Ringwald static int hfp_hf_set_speaker_gain_cmd(uint16_t cid, int gain){ 39989425bfcSMilanka Ringwald return hfp_hf_send_cmd_with_int(cid, HFP_SET_SPEAKER_GAIN, gain); 40089425bfcSMilanka Ringwald } 40189425bfcSMilanka Ringwald 40289425bfcSMilanka Ringwald static int hfp_hf_set_calling_line_notification_cmd(uint16_t cid, uint8_t activate){ 40389425bfcSMilanka Ringwald return hfp_hf_send_cmd_with_int(cid, HFP_ENABLE_CLIP, activate); 40489425bfcSMilanka Ringwald } 40589425bfcSMilanka Ringwald 40689425bfcSMilanka Ringwald static int hfp_hf_set_voice_recognition_notification_cmd(uint16_t cid, uint8_t activate){ 40789425bfcSMilanka Ringwald return hfp_hf_send_cmd_with_int(cid, HFP_ACTIVATE_VOICE_RECOGNITION, activate); 40889425bfcSMilanka Ringwald } 40989425bfcSMilanka Ringwald 41089425bfcSMilanka Ringwald static int hfp_hf_set_call_waiting_notification_cmd(uint16_t cid, uint8_t activate){ 41189425bfcSMilanka Ringwald return hfp_hf_send_cmd_with_int(cid, HFP_ENABLE_CALL_WAITING_NOTIFICATION, activate); 41289425bfcSMilanka Ringwald } 41389425bfcSMilanka Ringwald 41489425bfcSMilanka Ringwald static int hfp_hf_cmd_confirm_codec(uint16_t cid, uint8_t codec){ 41589425bfcSMilanka Ringwald return hfp_hf_send_cmd_with_int(cid, HFP_CONFIRM_COMMON_CODEC, codec); 41689425bfcSMilanka Ringwald } 41789425bfcSMilanka Ringwald 41889425bfcSMilanka Ringwald static int hfp_hf_cmd_enable_extended_audio_gateway_error_report(uint16_t cid, uint8_t enable){ 41989425bfcSMilanka Ringwald return hfp_hf_send_cmd_with_int(cid, HFP_ENABLE_EXTENDED_AUDIO_GATEWAY_ERROR, enable); 42089425bfcSMilanka Ringwald } 42189425bfcSMilanka Ringwald 42289425bfcSMilanka Ringwald static int hfp_hf_send_redial_last_number_cmd(uint16_t cid){ 42389425bfcSMilanka Ringwald return hfp_hf_send_cmd(cid, HFP_REDIAL_LAST_NUMBER); 42489425bfcSMilanka Ringwald } 42589425bfcSMilanka Ringwald 42689425bfcSMilanka Ringwald static int hfp_hf_send_chup(uint16_t cid){ 42789425bfcSMilanka Ringwald return hfp_hf_send_cmd(cid, HFP_HANG_UP_CALL); 42889425bfcSMilanka Ringwald } 42989425bfcSMilanka Ringwald 430ce263fc8SMatthias Ringwald static int hfp_hf_send_binp(uint16_t cid){ 43189425bfcSMilanka Ringwald return hfp_hf_send_cmd_with_mark(cid, HFP_PHONE_NUMBER_FOR_VOICE_TAG, "=1"); 432ce263fc8SMatthias Ringwald } 433ce263fc8SMatthias Ringwald 434667ec068SMatthias Ringwald static int hfp_hf_send_clcc(uint16_t cid){ 43589425bfcSMilanka Ringwald return hfp_hf_send_cmd(cid, HFP_LIST_CURRENT_CALLS); 436667ec068SMatthias Ringwald } 437667ec068SMatthias Ringwald 43876cc1527SMatthias Ringwald /* state machines */ 4393deb3ec6SMatthias Ringwald 440a0ffb263SMatthias Ringwald static int hfp_hf_run_for_context_service_level_connection(hfp_connection_t * hfp_connection){ 441a0ffb263SMatthias Ringwald if (hfp_connection->state >= HFP_SERVICE_LEVEL_CONNECTION_ESTABLISHED) return 0; 442a0ffb263SMatthias Ringwald if (hfp_connection->ok_pending) return 0; 443aa4dd815SMatthias Ringwald int done = 1; 444498a8121SMilanka Ringwald log_info("hfp_hf_run_for_context_service_level_connection state %d\n", hfp_connection->state); 445a0ffb263SMatthias Ringwald switch (hfp_connection->state){ 4463deb3ec6SMatthias Ringwald case HFP_EXCHANGE_SUPPORTED_FEATURES: 447aeb0f0feSMatthias Ringwald hfp_hf_drop_mSBC_if_eSCO_not_supported(hfp_hf_codecs, &hfp_hf_codecs_nr); 448a0ffb263SMatthias Ringwald hfp_connection->state = HFP_W4_EXCHANGE_SUPPORTED_FEATURES; 449a0ffb263SMatthias Ringwald hfp_hf_cmd_exchange_supported_features(hfp_connection->rfcomm_cid); 4503deb3ec6SMatthias Ringwald break; 4513deb3ec6SMatthias Ringwald case HFP_NOTIFY_ON_CODECS: 452a0ffb263SMatthias Ringwald hfp_connection->state = HFP_W4_NOTIFY_ON_CODECS; 453a0ffb263SMatthias Ringwald hfp_hf_cmd_notify_on_codecs(hfp_connection->rfcomm_cid); 4543deb3ec6SMatthias Ringwald break; 4553deb3ec6SMatthias Ringwald case HFP_RETRIEVE_INDICATORS: 456a0ffb263SMatthias Ringwald hfp_connection->state = HFP_W4_RETRIEVE_INDICATORS; 457a0ffb263SMatthias Ringwald hfp_hf_cmd_retrieve_indicators(hfp_connection->rfcomm_cid); 4583deb3ec6SMatthias Ringwald break; 4593deb3ec6SMatthias Ringwald case HFP_RETRIEVE_INDICATORS_STATUS: 460a0ffb263SMatthias Ringwald hfp_connection->state = HFP_W4_RETRIEVE_INDICATORS_STATUS; 461a0ffb263SMatthias Ringwald hfp_hf_cmd_retrieve_indicators_status(hfp_connection->rfcomm_cid); 4623deb3ec6SMatthias Ringwald break; 4633deb3ec6SMatthias Ringwald case HFP_ENABLE_INDICATORS_STATUS_UPDATE: 464a0ffb263SMatthias Ringwald hfp_connection->state = HFP_W4_ENABLE_INDICATORS_STATUS_UPDATE; 465a0ffb263SMatthias Ringwald hfp_hf_cmd_activate_status_update_for_all_ag_indicators(hfp_connection->rfcomm_cid, 1); 4663deb3ec6SMatthias Ringwald break; 4673deb3ec6SMatthias Ringwald case HFP_RETRIEVE_CAN_HOLD_CALL: 468a0ffb263SMatthias Ringwald hfp_connection->state = HFP_W4_RETRIEVE_CAN_HOLD_CALL; 469a0ffb263SMatthias Ringwald hfp_hf_cmd_retrieve_can_hold_call(hfp_connection->rfcomm_cid); 4703deb3ec6SMatthias Ringwald break; 4713deb3ec6SMatthias Ringwald case HFP_LIST_GENERIC_STATUS_INDICATORS: 472a0ffb263SMatthias Ringwald hfp_connection->state = HFP_W4_LIST_GENERIC_STATUS_INDICATORS; 473a0ffb263SMatthias Ringwald hfp_hf_cmd_list_supported_generic_status_indicators(hfp_connection->rfcomm_cid); 4743deb3ec6SMatthias Ringwald break; 4753deb3ec6SMatthias Ringwald case HFP_RETRIEVE_GENERIC_STATUS_INDICATORS: 476a0ffb263SMatthias Ringwald hfp_connection->state = HFP_W4_RETRIEVE_GENERIC_STATUS_INDICATORS; 477a0ffb263SMatthias Ringwald hfp_hf_cmd_retrieve_supported_generic_status_indicators(hfp_connection->rfcomm_cid); 4783deb3ec6SMatthias Ringwald break; 4793deb3ec6SMatthias Ringwald case HFP_RETRIEVE_INITITAL_STATE_GENERIC_STATUS_INDICATORS: 480a0ffb263SMatthias Ringwald hfp_connection->state = HFP_W4_RETRIEVE_INITITAL_STATE_GENERIC_STATUS_INDICATORS; 481a0ffb263SMatthias Ringwald hfp_hf_cmd_list_initital_supported_generic_status_indicators(hfp_connection->rfcomm_cid); 4823deb3ec6SMatthias Ringwald break; 4833deb3ec6SMatthias Ringwald default: 484aa4dd815SMatthias Ringwald done = 0; 4853deb3ec6SMatthias Ringwald break; 4863deb3ec6SMatthias Ringwald } 4873deb3ec6SMatthias Ringwald return done; 4883deb3ec6SMatthias Ringwald } 4893deb3ec6SMatthias Ringwald 490ce263fc8SMatthias Ringwald 491a0ffb263SMatthias Ringwald static int hfp_hf_run_for_context_service_level_connection_queries(hfp_connection_t * hfp_connection){ 492a0ffb263SMatthias Ringwald if (hfp_connection->state != HFP_SERVICE_LEVEL_CONNECTION_ESTABLISHED) return 0; 493498a8121SMilanka Ringwald if (hfp_connection->ok_pending){ 494498a8121SMilanka Ringwald return 0; 495498a8121SMilanka Ringwald } 496ce263fc8SMatthias Ringwald int done = 0; 497a0ffb263SMatthias Ringwald if (hfp_connection->enable_status_update_for_ag_indicators != 0xFF){ 498a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 499ce263fc8SMatthias Ringwald done = 1; 500a0ffb263SMatthias Ringwald hfp_hf_cmd_activate_status_update_for_all_ag_indicators(hfp_connection->rfcomm_cid, hfp_connection->enable_status_update_for_ag_indicators); 501ce263fc8SMatthias Ringwald return done; 502ce263fc8SMatthias Ringwald }; 503a0ffb263SMatthias Ringwald if (hfp_connection->change_status_update_for_individual_ag_indicators){ 504a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 505ce263fc8SMatthias Ringwald done = 1; 506a0ffb263SMatthias Ringwald hfp_hf_cmd_activate_status_update_for_ag_indicator(hfp_connection->rfcomm_cid, 507a0ffb263SMatthias Ringwald hfp_connection->ag_indicators_status_update_bitmap, 508a0ffb263SMatthias Ringwald hfp_connection->ag_indicators_nr); 509ce263fc8SMatthias Ringwald return done; 510ce263fc8SMatthias Ringwald } 511ce263fc8SMatthias Ringwald 512a0ffb263SMatthias Ringwald switch (hfp_connection->hf_query_operator_state){ 513ce263fc8SMatthias Ringwald case HFP_HF_QUERY_OPERATOR_SET_FORMAT: 514a0ffb263SMatthias Ringwald hfp_connection->hf_query_operator_state = HFP_HF_QUERY_OPERATOR_W4_SET_FORMAT_OK; 515a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 516a0ffb263SMatthias Ringwald hfp_hf_cmd_query_operator_name_format(hfp_connection->rfcomm_cid); 517ce263fc8SMatthias Ringwald return 1; 518ce263fc8SMatthias Ringwald case HFP_HF_QUERY_OPERATOR_SEND_QUERY: 519a0ffb263SMatthias Ringwald hfp_connection->hf_query_operator_state = HPF_HF_QUERY_OPERATOR_W4_RESULT; 520a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 521a0ffb263SMatthias Ringwald hfp_hf_cmd_query_operator_name(hfp_connection->rfcomm_cid); 522ce263fc8SMatthias Ringwald return 1; 523ce263fc8SMatthias Ringwald default: 524ce263fc8SMatthias Ringwald break; 525ce263fc8SMatthias Ringwald } 526ce263fc8SMatthias Ringwald 527a0ffb263SMatthias Ringwald if (hfp_connection->enable_extended_audio_gateway_error_report){ 528a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 529ce263fc8SMatthias Ringwald done = 1; 530a0ffb263SMatthias Ringwald hfp_hf_cmd_enable_extended_audio_gateway_error_report(hfp_connection->rfcomm_cid, hfp_connection->enable_extended_audio_gateway_error_report); 531ce263fc8SMatthias Ringwald return done; 532ce263fc8SMatthias Ringwald } 533ce263fc8SMatthias Ringwald 534ce263fc8SMatthias Ringwald return done; 535ce263fc8SMatthias Ringwald } 536ce263fc8SMatthias Ringwald 537af97579eSMilanka Ringwald static int hfp_hf_voice_recognition_state_machine(hfp_connection_t * hfp_connection){ 538be55a11dSMilanka Ringwald if (hfp_connection->state < HFP_SERVICE_LEVEL_CONNECTION_ESTABLISHED) { 539be55a11dSMilanka Ringwald return 0; 540be55a11dSMilanka Ringwald } 541be55a11dSMilanka Ringwald int done = 0; 542fd4151d1SMilanka Ringwald 5430b4debbfSMilanka Ringwald if (hfp_connection->ok_pending == 1){ 5440b4debbfSMilanka Ringwald return 0; 5450b4debbfSMilanka Ringwald } 5460b4debbfSMilanka Ringwald // voice recognition activated from AG 5470b4debbfSMilanka Ringwald if (hfp_connection->command == HFP_CMD_AG_ACTIVATE_VOICE_RECOGNITION){ 5480b4debbfSMilanka Ringwald switch(hfp_connection->vra_state_requested){ 5490b4debbfSMilanka Ringwald case HFP_VRA_W4_VOICE_RECOGNITION_ACTIVATED: 5500b4debbfSMilanka Ringwald case HFP_VRA_W4_VOICE_RECOGNITION_OFF: 551de9e0ea7SMilanka Ringwald case HFP_VRA_W4_ENHANCED_VOICE_RECOGNITION_READY_FOR_AUDIO: 5520b4debbfSMilanka Ringwald // ignore AG command, continue to wait for OK 5530b4debbfSMilanka Ringwald return 0; 554cf75be85SMilanka Ringwald 5550b4debbfSMilanka Ringwald default: 556b95cac54SMilanka Ringwald if (hfp_connection->ag_vra_msg_length > 0){ 557b95cac54SMilanka Ringwald hfp_hf_emit_enhanced_voice_recognition_text(hfp_connection); 558b95cac54SMilanka Ringwald hfp_connection->ag_vra_msg_length = 0; 559b95cac54SMilanka Ringwald break; 560b95cac54SMilanka Ringwald } 561cf75be85SMilanka Ringwald switch(hfp_connection->ag_vra_state){ 562cf75be85SMilanka Ringwald case HFP_VOICE_RECOGNITION_STATE_AG_READY: 563013cc750SMilanka Ringwald switch (hfp_connection->ag_vra_status){ 564013cc750SMilanka Ringwald case 0: 5650b4debbfSMilanka Ringwald hfp_connection->vra_state_requested = HFP_VRA_W4_VOICE_RECOGNITION_OFF; 566013cc750SMilanka Ringwald break; 567013cc750SMilanka Ringwald case 1: 568013cc750SMilanka Ringwald hfp_connection->vra_state_requested = HFP_VRA_W4_VOICE_RECOGNITION_ACTIVATED; 569013cc750SMilanka Ringwald break; 570013cc750SMilanka Ringwald case 2: 571013cc750SMilanka Ringwald hfp_connection->vra_state_requested = HFP_VRA_W4_ENHANCED_VOICE_RECOGNITION_READY_FOR_AUDIO; 572013cc750SMilanka Ringwald break; 573013cc750SMilanka Ringwald default: 574013cc750SMilanka Ringwald break; 5750b4debbfSMilanka Ringwald } 5760b4debbfSMilanka Ringwald break; 577cf75be85SMilanka Ringwald default: 578cf75be85SMilanka Ringwald // state messages from AG 579cf75be85SMilanka Ringwald hfp_emit_enhanced_voice_recognition_state_event(hfp_connection, ERROR_CODE_SUCCESS); 5808d5005b5SMilanka Ringwald hfp_connection->ag_vra_state = HFP_VOICE_RECOGNITION_STATE_AG_READY; 581cf75be85SMilanka Ringwald break; 582cf75be85SMilanka Ringwald } 583cf75be85SMilanka Ringwald break; 5840b4debbfSMilanka Ringwald } 5850b4debbfSMilanka Ringwald hfp_connection->command = HFP_CMD_NONE; 5860b4debbfSMilanka Ringwald } 5870b4debbfSMilanka Ringwald 5880b4debbfSMilanka Ringwald 589498a8121SMilanka Ringwald switch (hfp_connection->vra_state_requested){ 590fdda66c0SMilanka Ringwald case HFP_VRA_W2_SEND_VOICE_RECOGNITION_OFF: 591fdda66c0SMilanka Ringwald done = hfp_hf_set_voice_recognition_notification_cmd(hfp_connection->rfcomm_cid, 0); 592fdda66c0SMilanka Ringwald if (done != 0){ 593fdda66c0SMilanka Ringwald hfp_connection->vra_state_requested = HFP_VRA_W4_VOICE_RECOGNITION_OFF; 594498a8121SMilanka Ringwald hfp_connection->ok_pending = 1; 595498a8121SMilanka Ringwald } 596fd4151d1SMilanka Ringwald return 1; 597fd4151d1SMilanka Ringwald 598fd4151d1SMilanka Ringwald 599fdda66c0SMilanka Ringwald case HFP_VRA_W2_SEND_VOICE_RECOGNITION_ACTIVATED: 600fdda66c0SMilanka Ringwald done = hfp_hf_set_voice_recognition_notification_cmd(hfp_connection->rfcomm_cid, 1); 601fdda66c0SMilanka Ringwald if (done != 0){ 602fdda66c0SMilanka Ringwald hfp_connection->vra_state_requested = HFP_VRA_W4_VOICE_RECOGNITION_ACTIVATED; 603fd4151d1SMilanka Ringwald hfp_connection->ok_pending = 1; 604fd4151d1SMilanka Ringwald return 1; 6050b4debbfSMilanka Ringwald } 6060b4debbfSMilanka Ringwald break; 607013cc750SMilanka Ringwald 608de9e0ea7SMilanka Ringwald case HFP_VRA_W2_SEND_ENHANCED_VOICE_RECOGNITION_READY_FOR_AUDIO: 609de9e0ea7SMilanka Ringwald done = hfp_hf_set_voice_recognition_notification_cmd(hfp_connection->rfcomm_cid, 2); 610de9e0ea7SMilanka Ringwald if (done != 0){ 611de9e0ea7SMilanka Ringwald hfp_connection->vra_state_requested = HFP_VRA_W4_ENHANCED_VOICE_RECOGNITION_READY_FOR_AUDIO; 612de9e0ea7SMilanka Ringwald hfp_connection->ok_pending = 1; 613de9e0ea7SMilanka Ringwald return 1; 614de9e0ea7SMilanka Ringwald } 615de9e0ea7SMilanka Ringwald break; 616de9e0ea7SMilanka Ringwald 617de9e0ea7SMilanka Ringwald case HFP_VRA_W4_VOICE_RECOGNITION_OFF: 618de9e0ea7SMilanka Ringwald hfp_connection->vra_state = HFP_VRA_VOICE_RECOGNITION_OFF; 619de9e0ea7SMilanka Ringwald hfp_connection->vra_state_requested = hfp_connection->vra_state; 620de9e0ea7SMilanka Ringwald hfp_connection->activate_voice_recognition = false; 621de9e0ea7SMilanka Ringwald if (hfp_connection->activate_voice_recognition){ 622fcf4ede6SMilanka Ringwald hfp_connection->enhanced_voice_recognition_enabled = hfp_hf_enhanced_vra_flag_supported(hfp_connection); 623de9e0ea7SMilanka Ringwald hfp_hf_activate_voice_recognition(hfp_connection->acl_handle); 624de9e0ea7SMilanka Ringwald } else { 625553a4a56SMilanka Ringwald hfp_emit_voice_recognition_disabled(hfp_connection, ERROR_CODE_SUCCESS); 626de9e0ea7SMilanka Ringwald } 627de9e0ea7SMilanka Ringwald break; 628de9e0ea7SMilanka Ringwald 629be55a11dSMilanka Ringwald case HFP_VRA_W4_VOICE_RECOGNITION_ACTIVATED: 630498a8121SMilanka Ringwald hfp_connection->vra_state = HFP_VRA_VOICE_RECOGNITION_ACTIVATED; 631498a8121SMilanka Ringwald hfp_connection->vra_state_requested = hfp_connection->vra_state; 632de9e0ea7SMilanka Ringwald hfp_connection->activate_voice_recognition = false; 633de9e0ea7SMilanka Ringwald if (hfp_connection->deactivate_voice_recognition){ 634de9e0ea7SMilanka Ringwald hfp_hf_deactivate_voice_recognition(hfp_connection->acl_handle); 635de9e0ea7SMilanka Ringwald } else { 636fcf4ede6SMilanka Ringwald hfp_connection->enhanced_voice_recognition_enabled = hfp_hf_enhanced_vra_flag_supported(hfp_connection); 63784fb9ac1SMilanka Ringwald if (hfp_connection->state == HFP_AUDIO_CONNECTION_ESTABLISHED){ 638553a4a56SMilanka Ringwald hfp_emit_voice_recognition_enabled(hfp_connection, ERROR_CODE_SUCCESS); 63984fb9ac1SMilanka Ringwald } else { 64084fb9ac1SMilanka Ringwald // postpone VRA event to simplify application logic 64184fb9ac1SMilanka Ringwald hfp_connection->emit_vra_enabled_after_audio_established = true; 64284fb9ac1SMilanka Ringwald } 643de9e0ea7SMilanka Ringwald } 644be55a11dSMilanka Ringwald break; 645be55a11dSMilanka Ringwald 646de9e0ea7SMilanka Ringwald 647013cc750SMilanka Ringwald case HFP_VRA_W4_ENHANCED_VOICE_RECOGNITION_READY_FOR_AUDIO: 648013cc750SMilanka Ringwald hfp_connection->vra_state = HFP_VRA_ENHANCED_VOICE_RECOGNITION_READY_FOR_AUDIO; 649498a8121SMilanka Ringwald hfp_connection->vra_state_requested = hfp_connection->vra_state; 650de9e0ea7SMilanka Ringwald hfp_connection->activate_voice_recognition = false; 651de9e0ea7SMilanka Ringwald if (hfp_connection->deactivate_voice_recognition){ 652de9e0ea7SMilanka Ringwald hfp_hf_deactivate_voice_recognition(hfp_connection->acl_handle); 653de9e0ea7SMilanka Ringwald } else { 654de9e0ea7SMilanka Ringwald hfp_emit_enhanced_voice_recognition_hf_ready_for_audio_event(hfp_connection, ERROR_CODE_SUCCESS); 655de9e0ea7SMilanka Ringwald } 656be55a11dSMilanka Ringwald break; 657fd4151d1SMilanka Ringwald 658be55a11dSMilanka Ringwald default: 659be55a11dSMilanka Ringwald break; 660be55a11dSMilanka Ringwald } 661be55a11dSMilanka Ringwald return done; 662be55a11dSMilanka Ringwald } 663be55a11dSMilanka Ringwald 664be55a11dSMilanka Ringwald 665be55a11dSMilanka Ringwald static int codecs_exchange_state_machine(hfp_connection_t * hfp_connection){ 666a0ffb263SMatthias Ringwald if (hfp_connection->ok_pending) return 0; 667ce263fc8SMatthias Ringwald 668332ca98fSMatthias Ringwald if (hfp_connection->trigger_codec_exchange){ 669332ca98fSMatthias Ringwald hfp_connection->trigger_codec_exchange = 0; 670ce263fc8SMatthias Ringwald 671a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 672a0ffb263SMatthias Ringwald hfp_hf_cmd_trigger_codec_connection_setup(hfp_connection->rfcomm_cid); 673332ca98fSMatthias Ringwald return 1; 674332ca98fSMatthias Ringwald } 675332ca98fSMatthias Ringwald 6761cc65c4fSMatthias Ringwald if (hfp_connection->hf_send_codec_confirm){ 6771cc65c4fSMatthias Ringwald hfp_connection->hf_send_codec_confirm = false; 678ce263fc8SMatthias Ringwald 679a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 680fcb08cdbSMilanka Ringwald hfp_hf_cmd_confirm_codec(hfp_connection->rfcomm_cid, hfp_connection->codec_confirmed); 6811cc65c4fSMatthias Ringwald return 1; 6821cc65c4fSMatthias Ringwald } 6831cc65c4fSMatthias Ringwald 6841cc65c4fSMatthias Ringwald if (hfp_connection->hf_send_supported_codecs){ 6851cc65c4fSMatthias Ringwald hfp_connection->hf_send_supported_codecs = false; 6861cc65c4fSMatthias Ringwald 687a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 688a0ffb263SMatthias Ringwald hfp_hf_cmd_notify_on_codecs(hfp_connection->rfcomm_cid); 6891cc65c4fSMatthias Ringwald return 1; 6901cc65c4fSMatthias Ringwald } 691ce263fc8SMatthias Ringwald 692ce263fc8SMatthias Ringwald return 0; 693ce263fc8SMatthias Ringwald } 694ce263fc8SMatthias Ringwald 695a0ffb263SMatthias Ringwald static int hfp_hf_run_for_audio_connection(hfp_connection_t * hfp_connection){ 696505f1c30SMatthias Ringwald if ((hfp_connection->state < HFP_SERVICE_LEVEL_CONNECTION_ESTABLISHED) || 697505f1c30SMatthias Ringwald (hfp_connection->state > HFP_W2_DISCONNECT_SCO)) return 0; 698ce263fc8SMatthias Ringwald 69964f19dedSMilanka Ringwald if (hfp_connection->release_audio_connection){ 700a0ffb263SMatthias Ringwald hfp_connection->state = HFP_W4_SCO_DISCONNECTED; 701a0ffb263SMatthias Ringwald hfp_connection->release_audio_connection = 0; 702a0ffb263SMatthias Ringwald gap_disconnect(hfp_connection->sco_handle); 703ce263fc8SMatthias Ringwald return 1; 704ce263fc8SMatthias Ringwald } 705ce263fc8SMatthias Ringwald 706a0ffb263SMatthias Ringwald if (hfp_connection->state == HFP_AUDIO_CONNECTION_ESTABLISHED) return 0; 707ce263fc8SMatthias Ringwald 708ce263fc8SMatthias Ringwald // run codecs exchange 709a0ffb263SMatthias Ringwald int done = codecs_exchange_state_machine(hfp_connection); 710ce263fc8SMatthias Ringwald if (done) return 1; 711ce263fc8SMatthias Ringwald 71238200c1dSMilanka Ringwald if (hfp_connection->codecs_state != HFP_CODECS_EXCHANGED) return 0; 71338200c1dSMilanka Ringwald if (hfp_connection->establish_audio_connection){ 71438200c1dSMilanka Ringwald hfp_connection->state = HFP_W4_SCO_CONNECTED; 71538200c1dSMilanka Ringwald hfp_connection->establish_audio_connection = 0; 71638200c1dSMilanka Ringwald hfp_setup_synchronous_connection(hfp_connection); 71738200c1dSMilanka Ringwald return 1; 71838200c1dSMilanka Ringwald } 719ce263fc8SMatthias Ringwald return 0; 720ce263fc8SMatthias Ringwald } 721ce263fc8SMatthias Ringwald 72238200c1dSMilanka Ringwald 723a0ffb263SMatthias Ringwald static int call_setup_state_machine(hfp_connection_t * hfp_connection){ 724eaf2b0a1SMatthias Ringwald 725eaf2b0a1SMatthias Ringwald if (hfp_connection->ok_pending) return 0; 726eaf2b0a1SMatthias Ringwald 727a0ffb263SMatthias Ringwald if (hfp_connection->hf_answer_incoming_call){ 728a0ffb263SMatthias Ringwald hfp_hf_cmd_ata(hfp_connection->rfcomm_cid); 729a0ffb263SMatthias Ringwald hfp_connection->hf_answer_incoming_call = 0; 730ce263fc8SMatthias Ringwald return 1; 731ce263fc8SMatthias Ringwald } 732ce263fc8SMatthias Ringwald return 0; 733ce263fc8SMatthias Ringwald } 734ce263fc8SMatthias Ringwald 7351c6a0fc0SMatthias Ringwald static void hfp_hf_run_for_context(hfp_connection_t * hfp_connection){ 7367522e673SMatthias Ringwald 73776cc1527SMatthias Ringwald btstack_assert(hfp_connection != NULL); 73876cc1527SMatthias Ringwald btstack_assert(hfp_connection->local_role == HFP_ROLE_HF); 73976cc1527SMatthias Ringwald 74076cc1527SMatthias Ringwald // during SDP query, RFCOMM CID is not set 74176cc1527SMatthias Ringwald if (hfp_connection->rfcomm_cid == 0) return; 74222387625SMatthias Ringwald 74384fb9ac1SMilanka Ringwald // emit postponed VRA event 74484fb9ac1SMilanka Ringwald if (hfp_connection->state == HFP_AUDIO_CONNECTION_ESTABLISHED && hfp_connection->emit_vra_enabled_after_audio_established){ 74584fb9ac1SMilanka Ringwald hfp_connection->emit_vra_enabled_after_audio_established = false; 74684fb9ac1SMilanka Ringwald hfp_emit_voice_recognition_enabled(hfp_connection, ERROR_CODE_SUCCESS); 74784fb9ac1SMilanka Ringwald } 74884fb9ac1SMilanka Ringwald 7493721a235SMatthias Ringwald // assert command could be sent 7503721a235SMatthias Ringwald if (hci_can_send_command_packet_now() == 0) return; 7513721a235SMatthias Ringwald 7523721a235SMatthias Ringwald #ifdef ENABLE_CC256X_ASSISTED_HFP 7533721a235SMatthias Ringwald // WBS Disassociate 7543721a235SMatthias Ringwald if (hfp_connection->cc256x_send_wbs_disassociate){ 7553721a235SMatthias Ringwald hfp_connection->cc256x_send_wbs_disassociate = false; 7563721a235SMatthias Ringwald hci_send_cmd(&hci_ti_wbs_disassociate); 7573721a235SMatthias Ringwald return; 7583721a235SMatthias Ringwald } 7593721a235SMatthias Ringwald // Write Codec Config 7603721a235SMatthias Ringwald if (hfp_connection->cc256x_send_write_codec_config){ 7613721a235SMatthias Ringwald hfp_connection->cc256x_send_write_codec_config = false; 7623721a235SMatthias Ringwald hfp_cc256x_write_codec_config(hfp_connection); 7633721a235SMatthias Ringwald return; 7643721a235SMatthias Ringwald } 7653721a235SMatthias Ringwald // WBS Associate 7663721a235SMatthias Ringwald if (hfp_connection->cc256x_send_wbs_associate){ 7673721a235SMatthias Ringwald hfp_connection->cc256x_send_wbs_associate = false; 7683721a235SMatthias Ringwald hci_send_cmd(&hci_ti_wbs_associate, hfp_connection->acl_handle); 7693721a235SMatthias Ringwald return; 7703721a235SMatthias Ringwald } 7713721a235SMatthias Ringwald #endif 772689d4323SMatthias Ringwald #ifdef ENABLE_BCM_PCM_WBS 773689d4323SMatthias Ringwald // Enable WBS 774689d4323SMatthias Ringwald if (hfp_connection->bcm_send_enable_wbs){ 775689d4323SMatthias Ringwald hfp_connection->bcm_send_enable_wbs = false; 776689d4323SMatthias Ringwald hci_send_cmd(&hci_bcm_enable_wbs, 1, 2); 777689d4323SMatthias Ringwald return; 778689d4323SMatthias Ringwald } 779689d4323SMatthias Ringwald // Write I2S/PCM params 780689d4323SMatthias Ringwald if (hfp_connection->bcm_send_write_i2spcm_interface_param){ 781689d4323SMatthias Ringwald hfp_connection->bcm_send_write_i2spcm_interface_param = false; 782689d4323SMatthias Ringwald hfp_bcm_write_i2spcm_interface_param(hfp_connection); 783689d4323SMatthias Ringwald return; 784689d4323SMatthias Ringwald } 785689d4323SMatthias Ringwald // Disable WBS 786689d4323SMatthias Ringwald if (hfp_connection->bcm_send_disable_wbs){ 787689d4323SMatthias Ringwald hfp_connection->bcm_send_disable_wbs = false; 788689d4323SMatthias Ringwald hci_send_cmd(&hci_bcm_enable_wbs, 0, 2); 789689d4323SMatthias Ringwald return; 790689d4323SMatthias Ringwald } 791689d4323SMatthias Ringwald #endif 7922b5f92fdSMatthias Ringwald #ifdef ENABLE_RTK_PCM_WBS 7932b5f92fdSMatthias Ringwald if (hfp_connection->rtk_send_sco_config){ 7942b5f92fdSMatthias Ringwald hfp_connection->rtk_send_sco_config = false; 7952b5f92fdSMatthias Ringwald if (hfp_connection->negotiated_codec == HFP_CODEC_MSBC){ 7962b5f92fdSMatthias Ringwald log_info("RTK SCO: 16k + mSBC"); 7972b5f92fdSMatthias Ringwald hci_send_cmd(&hci_rtk_configure_sco_routing, 0x81, 0x90, 0x00, 0x00, 0x1a, 0x0c, 0x00, 0x00, 0x41); 7982b5f92fdSMatthias Ringwald } else { 7992b5f92fdSMatthias Ringwald log_info("RTK SCO: 16k + CVSD"); 8002b5f92fdSMatthias Ringwald hci_send_cmd(&hci_rtk_configure_sco_routing, 0x81, 0x90, 0x00, 0x00, 0x1a, 0x0c, 0x0c, 0x00, 0x01); 8012b5f92fdSMatthias Ringwald } 8022b5f92fdSMatthias Ringwald return; 8032b5f92fdSMatthias Ringwald } 8042b5f92fdSMatthias Ringwald #endif 80548e6eeeeSMatthias Ringwald #if defined (ENABLE_CC256X_ASSISTED_HFP) || defined (ENABLE_BCM_PCM_WBS) 80648e6eeeeSMatthias Ringwald if (hfp_connection->state == HFP_W4_WBS_SHUTDOWN){ 80748e6eeeeSMatthias Ringwald hfp_finalize_connection_context(hfp_connection); 80848e6eeeeSMatthias Ringwald return; 80948e6eeeeSMatthias Ringwald } 81048e6eeeeSMatthias Ringwald #endif 8113721a235SMatthias Ringwald 812cb81d35dSMatthias Ringwald if (hfp_connection->accept_sco){ 813cb81d35dSMatthias Ringwald bool incoming_eSCO = hfp_connection->accept_sco == 2; 814cb81d35dSMatthias Ringwald hfp_connection->accept_sco = 0; 8157522e673SMatthias Ringwald // notify about codec selection if not done already 8167522e673SMatthias Ringwald if (hfp_connection->negotiated_codec == 0){ 8177522e673SMatthias Ringwald hfp_connection->negotiated_codec = HFP_CODEC_CVSD; 8187522e673SMatthias Ringwald } 819cb81d35dSMatthias Ringwald hfp_accept_synchronous_connection(hfp_connection, incoming_eSCO); 8207522e673SMatthias Ringwald return; 8217522e673SMatthias Ringwald } 8227522e673SMatthias Ringwald 823d4dd47ffSMatthias Ringwald if (!rfcomm_can_send_packet_now(hfp_connection->rfcomm_cid)) { 824d4dd47ffSMatthias Ringwald rfcomm_request_can_send_now_event(hfp_connection->rfcomm_cid); 825d4dd47ffSMatthias Ringwald return; 826d4dd47ffSMatthias Ringwald } 827a0ffb263SMatthias Ringwald int done = hfp_hf_run_for_context_service_level_connection(hfp_connection); 828ce263fc8SMatthias Ringwald if (!done){ 829a0ffb263SMatthias Ringwald done = hfp_hf_run_for_context_service_level_connection_queries(hfp_connection); 830ce263fc8SMatthias Ringwald } 831ce263fc8SMatthias Ringwald if (!done){ 832c95b5b3cSMilanka Ringwald done = hfp_hf_run_for_audio_connection(hfp_connection); 833be55a11dSMilanka Ringwald } 834be55a11dSMilanka Ringwald if (!done){ 835c95b5b3cSMilanka Ringwald done = hfp_hf_voice_recognition_state_machine(hfp_connection); 836ce263fc8SMatthias Ringwald } 837ce263fc8SMatthias Ringwald if (!done){ 838a0ffb263SMatthias Ringwald done = call_setup_state_machine(hfp_connection); 839ce263fc8SMatthias Ringwald } 840ce263fc8SMatthias Ringwald 841e2c3b58dSMilanka Ringwald // don't send a new command while ok still pending or SLC is not established 842e2c3b58dSMilanka Ringwald if (hfp_connection->ok_pending || (hfp_connection->state < HFP_SERVICE_LEVEL_CONNECTION_ESTABLISHED)){ 843e2c3b58dSMilanka Ringwald return; 844e2c3b58dSMilanka Ringwald } 8451016a228SMatthias Ringwald 846a0ffb263SMatthias Ringwald if (hfp_connection->send_microphone_gain){ 847a0ffb263SMatthias Ringwald hfp_connection->send_microphone_gain = 0; 848a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 849a0ffb263SMatthias Ringwald hfp_hf_set_microphone_gain_cmd(hfp_connection->rfcomm_cid, hfp_connection->microphone_gain); 850ce263fc8SMatthias Ringwald return; 851ce263fc8SMatthias Ringwald } 852ce263fc8SMatthias Ringwald 853a0ffb263SMatthias Ringwald if (hfp_connection->send_speaker_gain){ 854a0ffb263SMatthias Ringwald hfp_connection->send_speaker_gain = 0; 855a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 856a0ffb263SMatthias Ringwald hfp_hf_set_speaker_gain_cmd(hfp_connection->rfcomm_cid, hfp_connection->speaker_gain); 857ce263fc8SMatthias Ringwald return; 858ce263fc8SMatthias Ringwald } 859ce263fc8SMatthias Ringwald 860a0ffb263SMatthias Ringwald if (hfp_connection->hf_deactivate_calling_line_notification){ 861a0ffb263SMatthias Ringwald hfp_connection->hf_deactivate_calling_line_notification = 0; 862a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 863a0ffb263SMatthias Ringwald hfp_hf_set_calling_line_notification_cmd(hfp_connection->rfcomm_cid, 0); 864ce263fc8SMatthias Ringwald return; 865ce263fc8SMatthias Ringwald } 866ce263fc8SMatthias Ringwald 867a0ffb263SMatthias Ringwald if (hfp_connection->hf_activate_calling_line_notification){ 868a0ffb263SMatthias Ringwald hfp_connection->hf_activate_calling_line_notification = 0; 869a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 870a0ffb263SMatthias Ringwald hfp_hf_set_calling_line_notification_cmd(hfp_connection->rfcomm_cid, 1); 871ce263fc8SMatthias Ringwald return; 872ce263fc8SMatthias Ringwald } 873ce263fc8SMatthias Ringwald 874a0ffb263SMatthias Ringwald if (hfp_connection->hf_deactivate_echo_canceling_and_noise_reduction){ 875a0ffb263SMatthias Ringwald hfp_connection->hf_deactivate_echo_canceling_and_noise_reduction = 0; 87699af1e28SMilanka Ringwald hfp_connection->response_pending_for_command = HFP_CMD_TURN_OFF_EC_AND_NR; 877a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 878e2c3b58dSMilanka Ringwald hfp_hf_send_cmd_with_int(hfp_connection->rfcomm_cid, HFP_TURN_OFF_EC_AND_NR, 0); 879ce263fc8SMatthias Ringwald return; 880ce263fc8SMatthias Ringwald } 881ce263fc8SMatthias Ringwald 882a0ffb263SMatthias Ringwald if (hfp_connection->hf_deactivate_call_waiting_notification){ 883a0ffb263SMatthias Ringwald hfp_connection->hf_deactivate_call_waiting_notification = 0; 884a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 885a0ffb263SMatthias Ringwald hfp_hf_set_call_waiting_notification_cmd(hfp_connection->rfcomm_cid, 0); 886ce263fc8SMatthias Ringwald return; 887ce263fc8SMatthias Ringwald } 888ce263fc8SMatthias Ringwald 889a0ffb263SMatthias Ringwald if (hfp_connection->hf_activate_call_waiting_notification){ 890a0ffb263SMatthias Ringwald hfp_connection->hf_activate_call_waiting_notification = 0; 891a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 892a0ffb263SMatthias Ringwald hfp_hf_set_call_waiting_notification_cmd(hfp_connection->rfcomm_cid, 1); 893ce263fc8SMatthias Ringwald return; 894ce263fc8SMatthias Ringwald } 895ce263fc8SMatthias Ringwald 896a0ffb263SMatthias Ringwald if (hfp_connection->hf_initiate_outgoing_call){ 897a0ffb263SMatthias Ringwald hfp_connection->hf_initiate_outgoing_call = 0; 898a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 899a0ffb263SMatthias Ringwald hfp_hf_initiate_outgoing_call_cmd(hfp_connection->rfcomm_cid); 900ce263fc8SMatthias Ringwald return; 901ce263fc8SMatthias Ringwald } 902ce263fc8SMatthias Ringwald 903a0ffb263SMatthias Ringwald if (hfp_connection->hf_initiate_memory_dialing){ 904a0ffb263SMatthias Ringwald hfp_connection->hf_initiate_memory_dialing = 0; 905a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 906a0ffb263SMatthias Ringwald hfp_hf_send_memory_dial_cmd(hfp_connection->rfcomm_cid, hfp_connection->memory_id); 907ce263fc8SMatthias Ringwald return; 908ce263fc8SMatthias Ringwald } 909ce263fc8SMatthias Ringwald 910a0ffb263SMatthias Ringwald if (hfp_connection->hf_initiate_redial_last_number){ 911a0ffb263SMatthias Ringwald hfp_connection->hf_initiate_redial_last_number = 0; 912a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 913a0ffb263SMatthias Ringwald hfp_hf_send_redial_last_number_cmd(hfp_connection->rfcomm_cid); 914ce263fc8SMatthias Ringwald return; 915ce263fc8SMatthias Ringwald } 916ce263fc8SMatthias Ringwald 917a0ffb263SMatthias Ringwald if (hfp_connection->hf_send_chup){ 918a0ffb263SMatthias Ringwald hfp_connection->hf_send_chup = 0; 919a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 920a0ffb263SMatthias Ringwald hfp_hf_send_chup(hfp_connection->rfcomm_cid); 921ce263fc8SMatthias Ringwald return; 922ce263fc8SMatthias Ringwald } 923ce263fc8SMatthias Ringwald 924a0ffb263SMatthias Ringwald if (hfp_connection->hf_send_chld_0){ 925a0ffb263SMatthias Ringwald hfp_connection->hf_send_chld_0 = 0; 926a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 927a0ffb263SMatthias Ringwald hfp_hf_send_chld(hfp_connection->rfcomm_cid, 0); 928ce263fc8SMatthias Ringwald return; 929ce263fc8SMatthias Ringwald } 930ce263fc8SMatthias Ringwald 931a0ffb263SMatthias Ringwald if (hfp_connection->hf_send_chld_1){ 932a0ffb263SMatthias Ringwald hfp_connection->hf_send_chld_1 = 0; 933a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 934a0ffb263SMatthias Ringwald hfp_hf_send_chld(hfp_connection->rfcomm_cid, 1); 935ce263fc8SMatthias Ringwald return; 936ce263fc8SMatthias Ringwald } 937ce263fc8SMatthias Ringwald 938a0ffb263SMatthias Ringwald if (hfp_connection->hf_send_chld_2){ 939a0ffb263SMatthias Ringwald hfp_connection->hf_send_chld_2 = 0; 940a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 941a0ffb263SMatthias Ringwald hfp_hf_send_chld(hfp_connection->rfcomm_cid, 2); 942ce263fc8SMatthias Ringwald return; 943ce263fc8SMatthias Ringwald } 944ce263fc8SMatthias Ringwald 945a0ffb263SMatthias Ringwald if (hfp_connection->hf_send_chld_3){ 946a0ffb263SMatthias Ringwald hfp_connection->hf_send_chld_3 = 0; 947a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 948a0ffb263SMatthias Ringwald hfp_hf_send_chld(hfp_connection->rfcomm_cid, 3); 949ce263fc8SMatthias Ringwald return; 950ce263fc8SMatthias Ringwald } 951ce263fc8SMatthias Ringwald 952a0ffb263SMatthias Ringwald if (hfp_connection->hf_send_chld_4){ 953a0ffb263SMatthias Ringwald hfp_connection->hf_send_chld_4 = 0; 954a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 955a0ffb263SMatthias Ringwald hfp_hf_send_chld(hfp_connection->rfcomm_cid, 4); 956ce263fc8SMatthias Ringwald return; 957ce263fc8SMatthias Ringwald } 958ce263fc8SMatthias Ringwald 959a0ffb263SMatthias Ringwald if (hfp_connection->hf_send_chld_x){ 960a0ffb263SMatthias Ringwald hfp_connection->hf_send_chld_x = 0; 961a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 962a0ffb263SMatthias Ringwald hfp_hf_send_chld(hfp_connection->rfcomm_cid, hfp_connection->hf_send_chld_x_index); 963667ec068SMatthias Ringwald return; 964667ec068SMatthias Ringwald } 965667ec068SMatthias Ringwald 966a0ffb263SMatthias Ringwald if (hfp_connection->hf_send_dtmf_code){ 967a0ffb263SMatthias Ringwald char code = hfp_connection->hf_send_dtmf_code; 968a0ffb263SMatthias Ringwald hfp_connection->hf_send_dtmf_code = 0; 969a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 970a0ffb263SMatthias Ringwald hfp_hf_send_dtmf(hfp_connection->rfcomm_cid, code); 971*c090e552SMatthias Ringwald // notify client that dtmf was sent 972*c090e552SMatthias Ringwald char buffer[2]; 973*c090e552SMatthias Ringwald buffer[0] = code; 974*c090e552SMatthias Ringwald buffer[1] = 0; 975*c090e552SMatthias Ringwald hfp_emit_string_event(hfp_connection, HFP_SUBEVENT_TRANSMIT_DTMF_CODES, buffer); 976ce263fc8SMatthias Ringwald return; 977ce263fc8SMatthias Ringwald } 978ce263fc8SMatthias Ringwald 979a0ffb263SMatthias Ringwald if (hfp_connection->hf_send_binp){ 980a0ffb263SMatthias Ringwald hfp_connection->hf_send_binp = 0; 981a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 982a0ffb263SMatthias Ringwald hfp_hf_send_binp(hfp_connection->rfcomm_cid); 983ce263fc8SMatthias Ringwald return; 984ce263fc8SMatthias Ringwald } 985ce263fc8SMatthias Ringwald 986a0ffb263SMatthias Ringwald if (hfp_connection->hf_send_clcc){ 987a0ffb263SMatthias Ringwald hfp_connection->hf_send_clcc = 0; 988a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 989a0ffb263SMatthias Ringwald hfp_hf_send_clcc(hfp_connection->rfcomm_cid); 990667ec068SMatthias Ringwald return; 991667ec068SMatthias Ringwald } 992667ec068SMatthias Ringwald 993a0ffb263SMatthias Ringwald if (hfp_connection->hf_send_rrh){ 994a0ffb263SMatthias Ringwald hfp_connection->hf_send_rrh = 0; 995667ec068SMatthias Ringwald char buffer[20]; 996a0ffb263SMatthias Ringwald switch (hfp_connection->hf_send_rrh_command){ 997667ec068SMatthias Ringwald case '?': 9981599fe57SMatthias Ringwald snprintf(buffer, sizeof(buffer), "AT%s?\r", 999ff7d6aeaSMatthias Ringwald HFP_RESPONSE_AND_HOLD); 1000ff7d6aeaSMatthias Ringwald buffer[sizeof(buffer) - 1] = 0; 1001a0ffb263SMatthias Ringwald send_str_over_rfcomm(hfp_connection->rfcomm_cid, buffer); 1002667ec068SMatthias Ringwald return; 1003667ec068SMatthias Ringwald case '0': 1004667ec068SMatthias Ringwald case '1': 1005667ec068SMatthias Ringwald case '2': 10061599fe57SMatthias Ringwald snprintf(buffer, sizeof(buffer), "AT%s=%c\r", 1007ff7d6aeaSMatthias Ringwald HFP_RESPONSE_AND_HOLD, 1008ff7d6aeaSMatthias Ringwald hfp_connection->hf_send_rrh_command); 1009ff7d6aeaSMatthias Ringwald buffer[sizeof(buffer) - 1] = 0; 1010a0ffb263SMatthias Ringwald send_str_over_rfcomm(hfp_connection->rfcomm_cid, buffer); 1011667ec068SMatthias Ringwald return; 1012667ec068SMatthias Ringwald default: 1013667ec068SMatthias Ringwald break; 1014667ec068SMatthias Ringwald } 1015667ec068SMatthias Ringwald return; 1016667ec068SMatthias Ringwald } 1017667ec068SMatthias Ringwald 1018a0ffb263SMatthias Ringwald if (hfp_connection->hf_send_cnum){ 1019a0ffb263SMatthias Ringwald hfp_connection->hf_send_cnum = 0; 1020667ec068SMatthias Ringwald char buffer[20]; 10211599fe57SMatthias Ringwald snprintf(buffer, sizeof(buffer), "AT%s\r", 1022ff7d6aeaSMatthias Ringwald HFP_SUBSCRIBER_NUMBER_INFORMATION); 1023ff7d6aeaSMatthias Ringwald buffer[sizeof(buffer) - 1] = 0; 1024a0ffb263SMatthias Ringwald send_str_over_rfcomm(hfp_connection->rfcomm_cid, buffer); 1025667ec068SMatthias Ringwald return; 1026667ec068SMatthias Ringwald } 1027667ec068SMatthias Ringwald 1028667ec068SMatthias Ringwald // update HF indicators 1029a0ffb263SMatthias Ringwald if (hfp_connection->generic_status_update_bitmap){ 1030667ec068SMatthias Ringwald int i; 1031aeb0f0feSMatthias Ringwald for (i=0; i < hfp_hf_indicators_nr; i++){ 1032a0ffb263SMatthias Ringwald if (get_bit(hfp_connection->generic_status_update_bitmap, i)){ 1033a0ffb263SMatthias Ringwald if (hfp_connection->generic_status_indicators[i].state){ 1034a0ffb263SMatthias Ringwald hfp_connection->ok_pending = 1; 1035a0ffb263SMatthias Ringwald hfp_connection->generic_status_update_bitmap = store_bit(hfp_connection->generic_status_update_bitmap, i, 0); 1036667ec068SMatthias Ringwald char buffer[30]; 10371599fe57SMatthias Ringwald snprintf(buffer, sizeof(buffer), "AT%s=%u,%u\r", 1038ff7d6aeaSMatthias Ringwald HFP_TRANSFER_HF_INDICATOR_STATUS, 1039aeb0f0feSMatthias Ringwald hfp_hf_indicators[i], 1040aeb0f0feSMatthias Ringwald (unsigned int)hfp_hf_indicators_value[i]); 1041ff7d6aeaSMatthias Ringwald buffer[sizeof(buffer) - 1] = 0; 1042a0ffb263SMatthias Ringwald send_str_over_rfcomm(hfp_connection->rfcomm_cid, buffer); 1043667ec068SMatthias Ringwald } else { 1044aeb0f0feSMatthias Ringwald log_info("Not sending HF indicator %u as it is disabled", hfp_hf_indicators[i]); 1045667ec068SMatthias Ringwald } 1046667ec068SMatthias Ringwald return; 1047667ec068SMatthias Ringwald } 1048667ec068SMatthias Ringwald } 1049667ec068SMatthias Ringwald } 1050667ec068SMatthias Ringwald 1051ce263fc8SMatthias Ringwald if (done) return; 1052ce263fc8SMatthias Ringwald // deal with disconnect 1053a0ffb263SMatthias Ringwald switch (hfp_connection->state){ 1054ce263fc8SMatthias Ringwald case HFP_W2_DISCONNECT_RFCOMM: 1055a0ffb263SMatthias Ringwald hfp_connection->state = HFP_W4_RFCOMM_DISCONNECTED; 1056a0ffb263SMatthias Ringwald rfcomm_disconnect(hfp_connection->rfcomm_cid); 1057ce263fc8SMatthias Ringwald break; 1058ce263fc8SMatthias Ringwald 1059ce263fc8SMatthias Ringwald default: 1060ce263fc8SMatthias Ringwald break; 1061ce263fc8SMatthias Ringwald } 1062ce263fc8SMatthias Ringwald } 1063ce263fc8SMatthias Ringwald 1064a0ffb263SMatthias Ringwald static void hfp_ag_slc_established(hfp_connection_t * hfp_connection){ 1065a0ffb263SMatthias Ringwald hfp_connection->state = HFP_SERVICE_LEVEL_CONNECTION_ESTABLISHED; 10666a7f44bdSMilanka Ringwald 10677095467fSMatthias Ringwald hfp_emit_slc_connection_event(hfp_connection->local_role, 0, hfp_connection->acl_handle, hfp_connection->remote_addr); 10687522e673SMatthias Ringwald 1069184a03edSMilanka Ringwald uint8_t i; 1070184a03edSMilanka Ringwald for (i = 0; i < hfp_connection->ag_indicators_nr; i++){ 10711ac1f60fSMilanka Ringwald hfp_emit_ag_indicator_mapping_event(hfp_connection, &hfp_connection->ag_indicators[i]); 1072184a03edSMilanka Ringwald } 1073722a85f3SMilanka Ringwald 1074722a85f3SMilanka Ringwald for (i = 0; i < hfp_connection->ag_indicators_nr; i++){ 1075722a85f3SMilanka Ringwald hfp_connection->ag_indicators[i].status_changed = 0; 1076722a85f3SMilanka Ringwald hfp_emit_ag_indicator_status_event(hfp_connection, &hfp_connection->ag_indicators[i]); 1077722a85f3SMilanka Ringwald } 1078722a85f3SMilanka Ringwald 1079667ec068SMatthias Ringwald // restore volume settings 1080a0ffb263SMatthias Ringwald hfp_connection->speaker_gain = hfp_hf_speaker_gain; 1081a0ffb263SMatthias Ringwald hfp_connection->send_speaker_gain = 1; 1082ca59be51SMatthias Ringwald hfp_emit_event(hfp_connection, HFP_SUBEVENT_SPEAKER_VOLUME, hfp_hf_speaker_gain); 1083a0ffb263SMatthias Ringwald hfp_connection->microphone_gain = hfp_hf_microphone_gain; 1084a0ffb263SMatthias Ringwald hfp_connection->send_microphone_gain = 1; 1085ca59be51SMatthias Ringwald hfp_emit_event(hfp_connection, HFP_SUBEVENT_MICROPHONE_VOLUME, hfp_hf_microphone_gain); 1086667ec068SMatthias Ringwald // enable all indicators 1087aeb0f0feSMatthias Ringwald for (i=0; i < hfp_hf_indicators_nr; i++){ 1088aeb0f0feSMatthias Ringwald hfp_connection->generic_status_indicators[i].uuid = hfp_hf_indicators[i]; 1089a0ffb263SMatthias Ringwald hfp_connection->generic_status_indicators[i].state = 1; 1090667ec068SMatthias Ringwald } 1091ce263fc8SMatthias Ringwald } 1092ce263fc8SMatthias Ringwald 10931cc65c4fSMatthias Ringwald static void hfp_hf_handle_suggested_codec(hfp_connection_t * hfp_connection){ 1094aeb0f0feSMatthias Ringwald if (hfp_supports_codec(hfp_connection->suggested_codec, hfp_hf_codecs_nr, hfp_hf_codecs)){ 10951cc65c4fSMatthias Ringwald // Codec supported, confirm 10961cc65c4fSMatthias Ringwald hfp_connection->negotiated_codec = hfp_connection->suggested_codec; 10971cc65c4fSMatthias Ringwald hfp_connection->codec_confirmed = hfp_connection->suggested_codec; 10981cc65c4fSMatthias Ringwald log_info("hfp: codec confirmed: %s", (hfp_connection->negotiated_codec == HFP_CODEC_MSBC) ? "mSBC" : "CVSD"); 10991cc65c4fSMatthias Ringwald hfp_connection->codecs_state = HFP_CODECS_HF_CONFIRMED_CODEC; 11001cc65c4fSMatthias Ringwald 11011cc65c4fSMatthias Ringwald hfp_connection->hf_send_codec_confirm = true; 11021cc65c4fSMatthias Ringwald } else { 11031cc65c4fSMatthias Ringwald // Codec not supported, send supported codecs 11041cc65c4fSMatthias Ringwald hfp_connection->codec_confirmed = 0; 11051cc65c4fSMatthias Ringwald hfp_connection->suggested_codec = 0; 11061cc65c4fSMatthias Ringwald hfp_connection->negotiated_codec = 0; 11071cc65c4fSMatthias Ringwald hfp_connection->codecs_state = HFP_CODECS_W4_AG_COMMON_CODEC; 11081cc65c4fSMatthias Ringwald 11091cc65c4fSMatthias Ringwald hfp_connection->hf_send_supported_codecs = true; 11101cc65c4fSMatthias Ringwald } 11111cc65c4fSMatthias Ringwald } 11121cc65c4fSMatthias Ringwald 1113e2c3b58dSMilanka Ringwald static bool hfp_hf_switch_on_ok_pending(hfp_connection_t *hfp_connection, uint8_t status){ 1114e2c3b58dSMilanka Ringwald bool event_emited = true; 1115e2c3b58dSMilanka Ringwald 111699af1e28SMilanka Ringwald switch (hfp_connection->response_pending_for_command){ 1117e2c3b58dSMilanka Ringwald case HFP_CMD_TURN_OFF_EC_AND_NR: 1118e2c3b58dSMilanka Ringwald hfp_emit_event(hfp_connection, HFP_SUBEVENT_ECHO_CANCELING_AND_NOISE_REDUCTION_DEACTIVATE, status); 1119e2c3b58dSMilanka Ringwald break; 1120e2c3b58dSMilanka Ringwald default: 1121e2c3b58dSMilanka Ringwald event_emited = false; 1122e2c3b58dSMilanka Ringwald 1123a0ffb263SMatthias Ringwald switch (hfp_connection->state){ 11243deb3ec6SMatthias Ringwald case HFP_W4_EXCHANGE_SUPPORTED_FEATURES: 1125a0ffb263SMatthias Ringwald if (has_codec_negotiation_feature(hfp_connection)){ 1126a0ffb263SMatthias Ringwald hfp_connection->state = HFP_NOTIFY_ON_CODECS; 11273deb3ec6SMatthias Ringwald break; 11283deb3ec6SMatthias Ringwald } 1129a0ffb263SMatthias Ringwald hfp_connection->state = HFP_RETRIEVE_INDICATORS; 11303deb3ec6SMatthias Ringwald break; 11313deb3ec6SMatthias Ringwald 11323deb3ec6SMatthias Ringwald case HFP_W4_NOTIFY_ON_CODECS: 1133a0ffb263SMatthias Ringwald hfp_connection->state = HFP_RETRIEVE_INDICATORS; 11343deb3ec6SMatthias Ringwald break; 11353deb3ec6SMatthias Ringwald 11363deb3ec6SMatthias Ringwald case HFP_W4_RETRIEVE_INDICATORS: 1137a0ffb263SMatthias Ringwald hfp_connection->state = HFP_RETRIEVE_INDICATORS_STATUS; 11383deb3ec6SMatthias Ringwald break; 11393deb3ec6SMatthias Ringwald 11403deb3ec6SMatthias Ringwald case HFP_W4_RETRIEVE_INDICATORS_STATUS: 1141a0ffb263SMatthias Ringwald hfp_connection->state = HFP_ENABLE_INDICATORS_STATUS_UPDATE; 11423deb3ec6SMatthias Ringwald break; 11433deb3ec6SMatthias Ringwald 11443deb3ec6SMatthias Ringwald case HFP_W4_ENABLE_INDICATORS_STATUS_UPDATE: 1145a0ffb263SMatthias Ringwald if (has_call_waiting_and_3way_calling_feature(hfp_connection)){ 1146a0ffb263SMatthias Ringwald hfp_connection->state = HFP_RETRIEVE_CAN_HOLD_CALL; 11473deb3ec6SMatthias Ringwald break; 11483deb3ec6SMatthias Ringwald } 1149a0ffb263SMatthias Ringwald if (has_hf_indicators_feature(hfp_connection)){ 1150a0ffb263SMatthias Ringwald hfp_connection->state = HFP_LIST_GENERIC_STATUS_INDICATORS; 11513deb3ec6SMatthias Ringwald break; 11523deb3ec6SMatthias Ringwald } 1153a0ffb263SMatthias Ringwald hfp_ag_slc_established(hfp_connection); 11543deb3ec6SMatthias Ringwald break; 11553deb3ec6SMatthias Ringwald 11563deb3ec6SMatthias Ringwald case HFP_W4_RETRIEVE_CAN_HOLD_CALL: 1157a0ffb263SMatthias Ringwald if (has_hf_indicators_feature(hfp_connection)){ 1158a0ffb263SMatthias Ringwald hfp_connection->state = HFP_LIST_GENERIC_STATUS_INDICATORS; 11593deb3ec6SMatthias Ringwald break; 11603deb3ec6SMatthias Ringwald } 1161a0ffb263SMatthias Ringwald hfp_ag_slc_established(hfp_connection); 11623deb3ec6SMatthias Ringwald break; 11633deb3ec6SMatthias Ringwald 11643deb3ec6SMatthias Ringwald case HFP_W4_LIST_GENERIC_STATUS_INDICATORS: 1165a0ffb263SMatthias Ringwald hfp_connection->state = HFP_RETRIEVE_GENERIC_STATUS_INDICATORS; 11663deb3ec6SMatthias Ringwald break; 11673deb3ec6SMatthias Ringwald 11683deb3ec6SMatthias Ringwald case HFP_W4_RETRIEVE_GENERIC_STATUS_INDICATORS: 1169a0ffb263SMatthias Ringwald hfp_connection->state = HFP_RETRIEVE_INITITAL_STATE_GENERIC_STATUS_INDICATORS; 11703deb3ec6SMatthias Ringwald break; 11713deb3ec6SMatthias Ringwald 11723deb3ec6SMatthias Ringwald case HFP_W4_RETRIEVE_INITITAL_STATE_GENERIC_STATUS_INDICATORS: 1173a0ffb263SMatthias Ringwald hfp_ag_slc_established(hfp_connection); 11743deb3ec6SMatthias Ringwald break; 1175ce263fc8SMatthias Ringwald case HFP_SERVICE_LEVEL_CONNECTION_ESTABLISHED: 1176a0ffb263SMatthias Ringwald if (hfp_connection->enable_status_update_for_ag_indicators != 0xFF){ 1177a0ffb263SMatthias Ringwald hfp_connection->enable_status_update_for_ag_indicators = 0xFF; 1178ca59be51SMatthias Ringwald hfp_emit_event(hfp_connection, HFP_SUBEVENT_COMPLETE, 0); 1179ce263fc8SMatthias Ringwald break; 1180ce263fc8SMatthias Ringwald } 11813deb3ec6SMatthias Ringwald 1182a0ffb263SMatthias Ringwald if (hfp_connection->change_status_update_for_individual_ag_indicators == 1){ 1183a0ffb263SMatthias Ringwald hfp_connection->change_status_update_for_individual_ag_indicators = 0; 1184ca59be51SMatthias Ringwald hfp_emit_event(hfp_connection, HFP_SUBEVENT_COMPLETE, 0); 1185ce263fc8SMatthias Ringwald break; 11863deb3ec6SMatthias Ringwald } 11873deb3ec6SMatthias Ringwald 1188a0ffb263SMatthias Ringwald switch (hfp_connection->hf_query_operator_state){ 1189ce263fc8SMatthias Ringwald case HFP_HF_QUERY_OPERATOR_W4_SET_FORMAT_OK: 1190a0ffb263SMatthias Ringwald hfp_connection->hf_query_operator_state = HFP_HF_QUERY_OPERATOR_SEND_QUERY; 1191ce263fc8SMatthias Ringwald break; 1192ce263fc8SMatthias Ringwald case HPF_HF_QUERY_OPERATOR_W4_RESULT: 1193a0ffb263SMatthias Ringwald hfp_connection->hf_query_operator_state = HFP_HF_QUERY_OPERATOR_FORMAT_SET; 1194a473a009SMatthias Ringwald hfp_emit_network_operator_event(hfp_connection); 1195ce263fc8SMatthias Ringwald break; 1196ce263fc8SMatthias Ringwald default: 1197ce263fc8SMatthias Ringwald break; 11983deb3ec6SMatthias Ringwald } 1199ce263fc8SMatthias Ringwald 1200a0ffb263SMatthias Ringwald if (hfp_connection->enable_extended_audio_gateway_error_report){ 1201a0ffb263SMatthias Ringwald hfp_connection->enable_extended_audio_gateway_error_report = 0; 1202ce263fc8SMatthias Ringwald break; 12033deb3ec6SMatthias Ringwald } 12043deb3ec6SMatthias Ringwald 1205a0ffb263SMatthias Ringwald switch (hfp_connection->codecs_state){ 1206aa4dd815SMatthias Ringwald case HFP_CODECS_RECEIVED_TRIGGER_CODEC_EXCHANGE: 1207a0ffb263SMatthias Ringwald hfp_connection->codecs_state = HFP_CODECS_W4_AG_COMMON_CODEC; 12083deb3ec6SMatthias Ringwald break; 1209ce263fc8SMatthias Ringwald case HFP_CODECS_HF_CONFIRMED_CODEC: 1210a0ffb263SMatthias Ringwald hfp_connection->codecs_state = HFP_CODECS_EXCHANGED; 1211ce263fc8SMatthias Ringwald break; 12123deb3ec6SMatthias Ringwald default: 12133deb3ec6SMatthias Ringwald break; 12143deb3ec6SMatthias Ringwald } 1215af97579eSMilanka Ringwald hfp_hf_voice_recognition_state_machine(hfp_connection); 1216be55a11dSMilanka Ringwald break; 1217be55a11dSMilanka Ringwald case HFP_AUDIO_CONNECTION_ESTABLISHED: 1218af97579eSMilanka Ringwald hfp_hf_voice_recognition_state_machine(hfp_connection); 12193deb3ec6SMatthias Ringwald break; 12203deb3ec6SMatthias Ringwald default: 12213deb3ec6SMatthias Ringwald break; 12223deb3ec6SMatthias Ringwald } 1223e2c3b58dSMilanka Ringwald break; 1224e2c3b58dSMilanka Ringwald } 12253deb3ec6SMatthias Ringwald 12263deb3ec6SMatthias Ringwald // done 122799af1e28SMilanka Ringwald hfp_connection->response_pending_for_command = HFP_CMD_NONE; 1228be55a11dSMilanka Ringwald hfp_connection->ok_pending = 0; 1229a0ffb263SMatthias Ringwald hfp_connection->command = HFP_CMD_NONE; 1230e2c3b58dSMilanka Ringwald return event_emited; 12313deb3ec6SMatthias Ringwald } 12323deb3ec6SMatthias Ringwald 1233a03dbc20SMilanka Ringwald static bool hfp_is_ringing(hfp_callsetup_status_t callsetup_status){ 1234a03dbc20SMilanka Ringwald switch (callsetup_status){ 1235a03dbc20SMilanka Ringwald case HFP_CALLSETUP_STATUS_INCOMING_CALL_SETUP_IN_PROGRESS: 1236a03dbc20SMilanka Ringwald case HFP_CALLSETUP_STATUS_OUTGOING_CALL_SETUP_IN_ALERTING_STATE: 1237a03dbc20SMilanka Ringwald return true; 1238a03dbc20SMilanka Ringwald default: 1239a03dbc20SMilanka Ringwald return false; 1240a03dbc20SMilanka Ringwald } 1241a03dbc20SMilanka Ringwald } 1242be55a11dSMilanka Ringwald 1243b08371a9SMilanka Ringwald static void hfp_hf_handle_transfer_ag_indicator_status(hfp_connection_t * hfp_connection) { 12444562e2a2SMatthias Ringwald uint16_t i; 1245a03dbc20SMilanka Ringwald 12464562e2a2SMatthias Ringwald for (i = 0; i < hfp_connection->ag_indicators_nr; i++){ 12474562e2a2SMatthias Ringwald if (hfp_connection->ag_indicators[i].status_changed) { 12484562e2a2SMatthias Ringwald if (strcmp(hfp_connection->ag_indicators[i].name, "callsetup") == 0){ 1249a03dbc20SMilanka Ringwald hfp_callsetup_status_t new_hf_callsetup_status = (hfp_callsetup_status_t) hfp_connection->ag_indicators[i].status; 1250a03dbc20SMilanka Ringwald bool ringing_old = hfp_is_ringing(hfp_hf_callsetup_status); 1251a03dbc20SMilanka Ringwald bool ringing_new = hfp_is_ringing(new_hf_callsetup_status); 1252a03dbc20SMilanka Ringwald if (ringing_old != ringing_new){ 1253a03dbc20SMilanka Ringwald if (ringing_new){ 1254a03dbc20SMilanka Ringwald hfp_emit_simple_event(hfp_connection, HFP_SUBEVENT_START_RINGING); 1255a03dbc20SMilanka Ringwald } else { 1256a03dbc20SMilanka Ringwald hfp_emit_simple_event(hfp_connection, HFP_SUBEVENT_STOP_RINGING); 1257a03dbc20SMilanka Ringwald } 1258a03dbc20SMilanka Ringwald } 1259a03dbc20SMilanka Ringwald hfp_hf_callsetup_status = new_hf_callsetup_status; 12604562e2a2SMatthias Ringwald } else if (strcmp(hfp_connection->ag_indicators[i].name, "callheld") == 0){ 1261aeb0f0feSMatthias Ringwald hfp_hf_callheld_status = (hfp_callheld_status_t) hfp_connection->ag_indicators[i].status; 12624562e2a2SMatthias Ringwald // avoid set but not used warning 1263aeb0f0feSMatthias Ringwald (void) hfp_hf_callheld_status; 12644562e2a2SMatthias Ringwald } else if (strcmp(hfp_connection->ag_indicators[i].name, "call") == 0){ 1265674ebed5SMilanka Ringwald hfp_call_status_t new_hf_call_status = (hfp_call_status_t) hfp_connection->ag_indicators[i].status; 1266674ebed5SMilanka Ringwald if (hfp_hf_call_status != new_hf_call_status){ 1267674ebed5SMilanka Ringwald if (new_hf_call_status == HFP_CALL_STATUS_NO_HELD_OR_ACTIVE_CALLS){ 1268674ebed5SMilanka Ringwald hfp_emit_simple_event(hfp_connection, HFP_SUBEVENT_CALL_TERMINATED); 1269674ebed5SMilanka Ringwald } else { 1270674ebed5SMilanka Ringwald hfp_emit_simple_event(hfp_connection, HFP_SUBEVENT_CALL_ANSWERED); 1271674ebed5SMilanka Ringwald } 1272674ebed5SMilanka Ringwald } 1273674ebed5SMilanka Ringwald hfp_hf_call_status = new_hf_call_status; 12744562e2a2SMatthias Ringwald } 12754562e2a2SMatthias Ringwald hfp_connection->ag_indicators[i].status_changed = 0; 1276ce3797e1SMilanka Ringwald hfp_emit_ag_indicator_status_event(hfp_connection, &hfp_connection->ag_indicators[i]); 12774562e2a2SMatthias Ringwald break; 12784562e2a2SMatthias Ringwald } 12794562e2a2SMatthias Ringwald } 12804562e2a2SMatthias Ringwald } 12814562e2a2SMatthias Ringwald 1282426f9988SMatthias Ringwald static void hfp_hf_handle_rfcomm_command(hfp_connection_t * hfp_connection){ 1283186dd3d2SMatthias Ringwald int value; 1284186dd3d2SMatthias Ringwald int i; 1285e2c3b58dSMilanka Ringwald bool event_emited; 1286e2c3b58dSMilanka Ringwald 1287125560b8SMatthias Ringwald // last argument is still in line_buffer 1288125560b8SMatthias Ringwald 1289a0ffb263SMatthias Ringwald switch (hfp_connection->command){ 1290667ec068SMatthias Ringwald case HFP_CMD_GET_SUBSCRIBER_NUMBER_INFORMATION: 1291a0ffb263SMatthias Ringwald hfp_connection->command = HFP_CMD_NONE; 1292a473a009SMatthias Ringwald hfp_hf_emit_subscriber_information(hfp_connection, 0); 1293667ec068SMatthias Ringwald break; 1294667ec068SMatthias Ringwald case HFP_CMD_RESPONSE_AND_HOLD_STATUS: 1295a0ffb263SMatthias Ringwald hfp_connection->command = HFP_CMD_NONE; 1296ca59be51SMatthias Ringwald hfp_emit_event(hfp_connection, HFP_SUBEVENT_RESPONSE_AND_HOLD_STATUS, btstack_atoi((char *)&hfp_connection->line_buffer[0])); 1297667ec068SMatthias Ringwald break; 1298667ec068SMatthias Ringwald case HFP_CMD_LIST_CURRENT_CALLS: 1299a0ffb263SMatthias Ringwald hfp_connection->command = HFP_CMD_NONE; 1300a473a009SMatthias Ringwald hfp_hf_emit_enhanced_call_status(hfp_connection); 1301667ec068SMatthias Ringwald break; 1302ce263fc8SMatthias Ringwald case HFP_CMD_SET_SPEAKER_GAIN: 1303a0ffb263SMatthias Ringwald hfp_connection->command = HFP_CMD_NONE; 13042308e108SMilanka Ringwald value = btstack_atoi((char*)hfp_connection->line_buffer); 1305667ec068SMatthias Ringwald hfp_hf_speaker_gain = value; 1306ca59be51SMatthias Ringwald hfp_emit_event(hfp_connection, HFP_SUBEVENT_SPEAKER_VOLUME, value); 1307ce263fc8SMatthias Ringwald break; 1308ce263fc8SMatthias Ringwald case HFP_CMD_SET_MICROPHONE_GAIN: 1309a0ffb263SMatthias Ringwald hfp_connection->command = HFP_CMD_NONE; 13102308e108SMilanka Ringwald value = btstack_atoi((char*)hfp_connection->line_buffer); 1311667ec068SMatthias Ringwald hfp_hf_microphone_gain = value; 1312ca59be51SMatthias Ringwald hfp_emit_event(hfp_connection, HFP_SUBEVENT_MICROPHONE_VOLUME, value); 1313ce263fc8SMatthias Ringwald break; 1314ce263fc8SMatthias Ringwald case HFP_CMD_AG_SENT_PHONE_NUMBER: 1315a0ffb263SMatthias Ringwald hfp_connection->command = HFP_CMD_NONE; 1316ca59be51SMatthias Ringwald hfp_emit_string_event(hfp_connection, HFP_SUBEVENT_NUMBER_FOR_VOICE_TAG, hfp_connection->bnip_number); 1317a0ffb263SMatthias Ringwald break; 1318a0ffb263SMatthias Ringwald case HFP_CMD_AG_SENT_CALL_WAITING_NOTIFICATION_UPDATE: 1319a0ffb263SMatthias Ringwald hfp_connection->command = HFP_CMD_NONE; 1320598d4936SMatthias Ringwald hfp_hf_emit_type_number_alpha(hfp_connection, HFP_SUBEVENT_CALL_WAITING_NOTIFICATION); 1321a0ffb263SMatthias Ringwald break; 1322a0ffb263SMatthias Ringwald case HFP_CMD_AG_SENT_CLIP_INFORMATION: 1323a0ffb263SMatthias Ringwald hfp_connection->command = HFP_CMD_NONE; 1324598d4936SMatthias Ringwald hfp_hf_emit_type_number_alpha(hfp_connection, HFP_SUBEVENT_CALLING_LINE_IDENTIFICATION_NOTIFICATION); 1325ce263fc8SMatthias Ringwald break; 1326ce263fc8SMatthias Ringwald case HFP_CMD_EXTENDED_AUDIO_GATEWAY_ERROR: 1327a0ffb263SMatthias Ringwald hfp_connection->command = HFP_CMD_NONE; 13285a4785c8SMatthias Ringwald hfp_connection->ok_pending = 0; 1329a0ffb263SMatthias Ringwald hfp_connection->extended_audio_gateway_error = 0; 1330ca59be51SMatthias Ringwald hfp_emit_event(hfp_connection, HFP_SUBEVENT_EXTENDED_AUDIO_GATEWAY_ERROR, hfp_connection->extended_audio_gateway_error_value); 1331ce263fc8SMatthias Ringwald break; 13320b4debbfSMilanka Ringwald case HFP_CMD_AG_ACTIVATE_VOICE_RECOGNITION: 13330b4debbfSMilanka Ringwald break; 1334fdda66c0SMilanka Ringwald case HFP_CMD_ERROR: 133590244c92SMilanka Ringwald switch (hfp_connection->state){ 133690244c92SMilanka Ringwald case HFP_SERVICE_LEVEL_CONNECTION_ESTABLISHED: 133790244c92SMilanka Ringwald switch (hfp_connection->codecs_state){ 133890244c92SMilanka Ringwald case HFP_CODECS_RECEIVED_TRIGGER_CODEC_EXCHANGE: 1339fdda66c0SMilanka Ringwald hfp_reset_context_flags(hfp_connection); 134090244c92SMilanka Ringwald hfp_emit_sco_event(hfp_connection, HFP_REMOTE_REJECTS_AUDIO_CONNECTION, 0, hfp_connection->remote_addr, hfp_connection->negotiated_codec); 134190244c92SMilanka Ringwald return; 134290244c92SMilanka Ringwald default: 134390244c92SMilanka Ringwald break; 134490244c92SMilanka Ringwald } 134556f1adacSMilanka Ringwald break; 134656f1adacSMilanka Ringwald default: 134756f1adacSMilanka Ringwald break; 134856f1adacSMilanka Ringwald } 1349e2c3b58dSMilanka Ringwald 1350fdda66c0SMilanka Ringwald // handle error response for voice activation (HF initiated) 13510b4debbfSMilanka Ringwald switch(hfp_connection->vra_state_requested){ 1352de9e0ea7SMilanka Ringwald case HFP_VRA_W4_ENHANCED_VOICE_RECOGNITION_READY_FOR_AUDIO: 1353de9e0ea7SMilanka Ringwald hfp_emit_enhanced_voice_recognition_hf_ready_for_audio_event(hfp_connection, ERROR_CODE_UNSPECIFIED_ERROR); 1354be55a11dSMilanka Ringwald break; 1355be55a11dSMilanka Ringwald default: 1356e2c3b58dSMilanka Ringwald if (hfp_connection->vra_state_requested == hfp_connection->vra_state){ 1357e2c3b58dSMilanka Ringwald break; 1358e2c3b58dSMilanka Ringwald } 13590b4debbfSMilanka Ringwald hfp_connection->vra_state_requested = hfp_connection->vra_state; 1360553a4a56SMilanka Ringwald hfp_emit_voice_recognition_enabled(hfp_connection, ERROR_CODE_UNSPECIFIED_ERROR); 13610b4debbfSMilanka Ringwald hfp_reset_context_flags(hfp_connection); 13620b4debbfSMilanka Ringwald return; 1363be55a11dSMilanka Ringwald } 1364e2c3b58dSMilanka Ringwald event_emited = hfp_hf_switch_on_ok_pending(hfp_connection, ERROR_CODE_UNSPECIFIED_ERROR); 1365e2c3b58dSMilanka Ringwald if (!event_emited){ 13660b4debbfSMilanka Ringwald hfp_emit_event(hfp_connection, HFP_SUBEVENT_COMPLETE, 1); 1367e2c3b58dSMilanka Ringwald } 1368fdda66c0SMilanka Ringwald hfp_reset_context_flags(hfp_connection); 1369ce263fc8SMatthias Ringwald break; 1370fdda66c0SMilanka Ringwald 1371ce263fc8SMatthias Ringwald case HFP_CMD_OK: 1372e2c3b58dSMilanka Ringwald hfp_hf_switch_on_ok_pending(hfp_connection, ERROR_CODE_SUCCESS); 1373ce263fc8SMatthias Ringwald break; 1374ce263fc8SMatthias Ringwald case HFP_CMD_RING: 13755a4785c8SMatthias Ringwald hfp_connection->command = HFP_CMD_NONE; 1376ca59be51SMatthias Ringwald hfp_emit_simple_event(hfp_connection, HFP_SUBEVENT_RING); 1377ce263fc8SMatthias Ringwald break; 1378ce263fc8SMatthias Ringwald case HFP_CMD_TRANSFER_AG_INDICATOR_STATUS: 13795a4785c8SMatthias Ringwald hfp_connection->command = HFP_CMD_NONE; 13804562e2a2SMatthias Ringwald hfp_hf_handle_transfer_ag_indicator_status(hfp_connection); 1381ce263fc8SMatthias Ringwald break; 1382c741b032SMilanka Ringwald case HFP_CMD_RETRIEVE_AG_INDICATORS_STATUS: 13835a4785c8SMatthias Ringwald hfp_connection->command = HFP_CMD_NONE; 1384184a03edSMilanka Ringwald // report status after SLC established 1385184a03edSMilanka Ringwald if (hfp_connection->state < HFP_SERVICE_LEVEL_CONNECTION_ESTABLISHED){ 1386184a03edSMilanka Ringwald break; 1387184a03edSMilanka Ringwald } 1388c741b032SMilanka Ringwald for (i = 0; i < hfp_connection->ag_indicators_nr; i++){ 13891ac1f60fSMilanka Ringwald hfp_emit_ag_indicator_mapping_event(hfp_connection, &hfp_connection->ag_indicators[i]); 1390c741b032SMilanka Ringwald } 1391c741b032SMilanka Ringwald break; 13921cc65c4fSMatthias Ringwald case HFP_CMD_AG_SUGGESTED_CODEC: 13931cc65c4fSMatthias Ringwald hfp_connection->command = HFP_CMD_NONE; 13945a4785c8SMatthias Ringwald hfp_hf_handle_suggested_codec(hfp_connection); 13951cc65c4fSMatthias Ringwald break; 1396eac56539SMilanka Ringwald case HFP_CMD_CHANGE_IN_BAND_RING_TONE_SETTING: 1397eac56539SMilanka Ringwald hfp_emit_event(hfp_connection, HFP_SUBEVENT_IN_BAND_RING_TONE, get_bit(hfp_connection->remote_supported_features, HFP_AGSF_IN_BAND_RING_TONE)); 1398ce263fc8SMatthias Ringwald default: 1399ce263fc8SMatthias Ringwald break; 14003deb3ec6SMatthias Ringwald } 14010cef86faSMatthias Ringwald } 1402426f9988SMatthias Ringwald 140376cc1527SMatthias Ringwald static int hfp_parser_is_end_of_line(uint8_t byte){ 140476cc1527SMatthias Ringwald return (byte == '\n') || (byte == '\r'); 140576cc1527SMatthias Ringwald } 140676cc1527SMatthias Ringwald 14070b4debbfSMilanka Ringwald static void hfp_hf_handle_rfcomm_data(hfp_connection_t * hfp_connection, uint8_t *packet, uint16_t size){ 1408426f9988SMatthias Ringwald // assertion: size >= 1 as rfcomm.c does not deliver empty packets 1409426f9988SMatthias Ringwald if (size < 1) return; 1410426f9988SMatthias Ringwald 1411426f9988SMatthias Ringwald hfp_log_rfcomm_message("HFP_HF_RX", packet, size); 1412e43d1938SMatthias Ringwald #ifdef ENABLE_HFP_AT_MESSAGES 1413e43d1938SMatthias Ringwald hfp_emit_string_event(hfp_connection, HFP_SUBEVENT_AT_MESSAGE_RECEIVED, (char *) packet); 1414e43d1938SMatthias Ringwald #endif 1415426f9988SMatthias Ringwald 1416426f9988SMatthias Ringwald // process messages byte-wise 1417a7ba78b0SMilanka Ringwald uint8_t pos; 1418426f9988SMatthias Ringwald for (pos = 0; pos < size; pos++){ 1419426f9988SMatthias Ringwald hfp_parse(hfp_connection, packet[pos], 1); 14201599fe57SMatthias Ringwald // parse until end of line "\r" or "\n" 1421426f9988SMatthias Ringwald if (!hfp_parser_is_end_of_line(packet[pos])) continue; 14220b4debbfSMilanka Ringwald hfp_hf_handle_rfcomm_command(hfp_connection); 14233deb3ec6SMatthias Ringwald } 1424a7ba78b0SMilanka Ringwald } 14253deb3ec6SMatthias Ringwald 14261c6a0fc0SMatthias Ringwald static void hfp_hf_run(void){ 1427665d90f2SMatthias Ringwald btstack_linked_list_iterator_t it; 1428665d90f2SMatthias Ringwald btstack_linked_list_iterator_init(&it, hfp_get_connections()); 1429665d90f2SMatthias Ringwald while (btstack_linked_list_iterator_has_next(&it)){ 1430a0ffb263SMatthias Ringwald hfp_connection_t * hfp_connection = (hfp_connection_t *)btstack_linked_list_iterator_next(&it); 143122387625SMatthias Ringwald if (hfp_connection->local_role != HFP_ROLE_HF) continue; 14321c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 14333deb3ec6SMatthias Ringwald } 14343deb3ec6SMatthias Ringwald } 14353deb3ec6SMatthias Ringwald 14361c6a0fc0SMatthias Ringwald static void hfp_hf_rfcomm_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){ 14370b4debbfSMilanka Ringwald hfp_connection_t * hfp_connection; 14383deb3ec6SMatthias Ringwald switch (packet_type){ 14393deb3ec6SMatthias Ringwald case RFCOMM_DATA_PACKET: 14400b4debbfSMilanka Ringwald hfp_connection = get_hfp_connection_context_for_rfcomm_cid(channel); 14410b4debbfSMilanka Ringwald if (!hfp_connection) return; 14420b4debbfSMilanka Ringwald hfp_hf_handle_rfcomm_data(hfp_connection, packet, size); 14430b4debbfSMilanka Ringwald hfp_hf_run_for_context(hfp_connection); 14440b4debbfSMilanka Ringwald return; 14453deb3ec6SMatthias Ringwald case HCI_EVENT_PACKET: 1446d4dd47ffSMatthias Ringwald if (packet[0] == RFCOMM_EVENT_CAN_SEND_NOW){ 1447d4dd47ffSMatthias Ringwald uint16_t rfcomm_cid = rfcomm_event_can_send_now_get_rfcomm_cid(packet); 14481c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(get_hfp_connection_context_for_rfcomm_cid(rfcomm_cid)); 1449d4dd47ffSMatthias Ringwald return; 1450d4dd47ffSMatthias Ringwald } 145127950165SMatthias Ringwald hfp_handle_rfcomm_event(packet_type, channel, packet, size, HFP_ROLE_HF); 1452202c8a4cSMatthias Ringwald break; 14533deb3ec6SMatthias Ringwald default: 14543deb3ec6SMatthias Ringwald break; 14553deb3ec6SMatthias Ringwald } 14561c6a0fc0SMatthias Ringwald hfp_hf_run(); 14573deb3ec6SMatthias Ringwald } 14583deb3ec6SMatthias Ringwald 14591c6a0fc0SMatthias Ringwald static void hfp_hf_hci_event_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){ 1460405014fbSMatthias Ringwald hfp_handle_hci_event(packet_type, channel, packet, size, HFP_ROLE_HF); 14611c6a0fc0SMatthias Ringwald hfp_hf_run(); 1462405014fbSMatthias Ringwald } 1463405014fbSMatthias Ringwald 1464aeb0f0feSMatthias Ringwald static void hfp_hf_set_defaults(void){ 1465aeb0f0feSMatthias Ringwald hfp_hf_supported_features = HFP_DEFAULT_HF_SUPPORTED_FEATURES; 1466aeb0f0feSMatthias Ringwald hfp_hf_call_status = HFP_CALL_STATUS_NO_HELD_OR_ACTIVE_CALLS; 1467aeb0f0feSMatthias Ringwald hfp_hf_callsetup_status = HFP_CALLSETUP_STATUS_NO_CALL_SETUP_IN_PROGRESS; 1468aeb0f0feSMatthias Ringwald hfp_hf_callheld_status= HFP_CALLHELD_STATUS_NO_CALLS_HELD; 1469aeb0f0feSMatthias Ringwald hfp_hf_codecs_nr = 0; 1470aeb0f0feSMatthias Ringwald hfp_hf_speaker_gain = 9; 1471aeb0f0feSMatthias Ringwald hfp_hf_microphone_gain = 9; 1472aeb0f0feSMatthias Ringwald hfp_hf_indicators_nr = 0; 1473aeb0f0feSMatthias Ringwald } 1474aeb0f0feSMatthias Ringwald 1475b4df8028SMilanka Ringwald uint8_t hfp_hf_init(uint16_t rfcomm_channel_nr){ 1476b4df8028SMilanka Ringwald uint8_t status = rfcomm_register_service(hfp_hf_rfcomm_packet_handler, rfcomm_channel_nr, 0xffff); 1477b4df8028SMilanka Ringwald if (status != ERROR_CODE_SUCCESS){ 1478b4df8028SMilanka Ringwald return status; 1479b4df8028SMilanka Ringwald } 1480b4df8028SMilanka Ringwald 1481520c92d5SMatthias Ringwald hfp_init(); 1482aeb0f0feSMatthias Ringwald hfp_hf_set_defaults(); 1483d63c37a1SMatthias Ringwald 14841c6a0fc0SMatthias Ringwald hfp_hf_hci_event_callback_registration.callback = &hfp_hf_hci_event_packet_handler; 14851c6a0fc0SMatthias Ringwald hci_add_event_handler(&hfp_hf_hci_event_callback_registration); 148627950165SMatthias Ringwald 148727950165SMatthias Ringwald // used to set packet handler for outgoing rfcomm connections - could be handled by emitting an event to us 14881c6a0fc0SMatthias Ringwald hfp_set_hf_rfcomm_packet_handler(&hfp_hf_rfcomm_packet_handler); 1489b4df8028SMilanka Ringwald return ERROR_CODE_SUCCESS; 149020b2edb6SMatthias Ringwald } 149127950165SMatthias Ringwald 149220b2edb6SMatthias Ringwald void hfp_hf_deinit(void){ 149320b2edb6SMatthias Ringwald hfp_deinit(); 1494aeb0f0feSMatthias Ringwald hfp_hf_set_defaults(); 1495aeb0f0feSMatthias Ringwald 1496aeb0f0feSMatthias Ringwald hfp_hf_callback = NULL; 149720b2edb6SMatthias Ringwald (void) memset(&hfp_hf_hci_event_callback_registration, 0, sizeof(btstack_packet_callback_registration_t)); 1498aeb0f0feSMatthias Ringwald (void) memset(hfp_hf_phone_number, 0, sizeof(hfp_hf_phone_number)); 1499a0ffb263SMatthias Ringwald } 1500a0ffb263SMatthias Ringwald 15017ca89cabSMatthias Ringwald void hfp_hf_init_codecs(int codecs_nr, const uint8_t * codecs){ 150268466199SMilanka Ringwald btstack_assert(codecs_nr < HFP_MAX_NUM_CODECS); 15033deb3ec6SMatthias Ringwald 1504aeb0f0feSMatthias Ringwald hfp_hf_codecs_nr = codecs_nr; 15053deb3ec6SMatthias Ringwald int i; 15063deb3ec6SMatthias Ringwald for (i=0; i<codecs_nr; i++){ 1507aeb0f0feSMatthias Ringwald hfp_hf_codecs[i] = codecs[i]; 15083deb3ec6SMatthias Ringwald } 15093deb3ec6SMatthias Ringwald } 15103deb3ec6SMatthias Ringwald 1511a0ffb263SMatthias Ringwald void hfp_hf_init_supported_features(uint32_t supported_features){ 1512aeb0f0feSMatthias Ringwald hfp_hf_supported_features = supported_features; 1513a0ffb263SMatthias Ringwald } 15143deb3ec6SMatthias Ringwald 15157ca89cabSMatthias Ringwald void hfp_hf_init_hf_indicators(int indicators_nr, const uint16_t * indicators){ 1516aeb0f0feSMatthias Ringwald btstack_assert(hfp_hf_indicators_nr < HFP_MAX_NUM_INDICATORS); 151768466199SMilanka Ringwald 1518aeb0f0feSMatthias Ringwald hfp_hf_indicators_nr = indicators_nr; 15193deb3ec6SMatthias Ringwald int i; 1520aeb0f0feSMatthias Ringwald for (i = 0; i < hfp_hf_indicators_nr ; i++){ 1521aeb0f0feSMatthias Ringwald hfp_hf_indicators[i] = indicators[i]; 15223deb3ec6SMatthias Ringwald } 15233deb3ec6SMatthias Ringwald } 15243deb3ec6SMatthias Ringwald 15254eb3f1d8SMilanka Ringwald uint8_t hfp_hf_establish_service_level_connection(bd_addr_t bd_addr){ 15264eb3f1d8SMilanka Ringwald return hfp_establish_service_level_connection(bd_addr, BLUETOOTH_SERVICE_CLASS_HANDSFREE_AUDIO_GATEWAY, HFP_ROLE_HF); 15273deb3ec6SMatthias Ringwald } 15283deb3ec6SMatthias Ringwald 1529657bc59fSMilanka Ringwald uint8_t hfp_hf_release_service_level_connection(hci_con_handle_t acl_handle){ 15309c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1531a33eb0c4SMilanka Ringwald if (!hfp_connection){ 1532657bc59fSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1533a33eb0c4SMilanka Ringwald } 15341ffa0dd9SMilanka Ringwald hfp_trigger_release_service_level_connection(hfp_connection); 15351c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 1536657bc59fSMilanka Ringwald return ERROR_CODE_SUCCESS; 15373deb3ec6SMatthias Ringwald } 15383deb3ec6SMatthias Ringwald 15393c65e705SMilanka Ringwald static uint8_t hfp_hf_set_status_update_for_all_ag_indicators(hci_con_handle_t acl_handle, uint8_t enable){ 15409c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1541a0ffb263SMatthias Ringwald if (!hfp_connection) { 15423c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 15433deb3ec6SMatthias Ringwald } 1544a0ffb263SMatthias Ringwald hfp_connection->enable_status_update_for_ag_indicators = enable; 15451c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 15463c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 15473deb3ec6SMatthias Ringwald } 15483deb3ec6SMatthias Ringwald 15493c65e705SMilanka Ringwald uint8_t hfp_hf_enable_status_update_for_all_ag_indicators(hci_con_handle_t acl_handle){ 15503c65e705SMilanka Ringwald return hfp_hf_set_status_update_for_all_ag_indicators(acl_handle, 1); 1551ce263fc8SMatthias Ringwald } 1552ce263fc8SMatthias Ringwald 15533c65e705SMilanka Ringwald uint8_t hfp_hf_disable_status_update_for_all_ag_indicators(hci_con_handle_t acl_handle){ 15543c65e705SMilanka Ringwald return hfp_hf_set_status_update_for_all_ag_indicators(acl_handle, 0); 1555ce263fc8SMatthias Ringwald } 1556ce263fc8SMatthias Ringwald 15573deb3ec6SMatthias Ringwald // TODO: returned ERROR - wrong format 15583c65e705SMilanka Ringwald uint8_t hfp_hf_set_status_update_for_individual_ag_indicators(hci_con_handle_t acl_handle, uint32_t indicators_status_bitmap){ 15599c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1560a0ffb263SMatthias Ringwald if (!hfp_connection) { 15613c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 15623deb3ec6SMatthias Ringwald } 1563a0ffb263SMatthias Ringwald hfp_connection->change_status_update_for_individual_ag_indicators = 1; 1564a0ffb263SMatthias Ringwald hfp_connection->ag_indicators_status_update_bitmap = indicators_status_bitmap; 15651c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 15663c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 15673deb3ec6SMatthias Ringwald } 15683deb3ec6SMatthias Ringwald 15693c65e705SMilanka Ringwald uint8_t hfp_hf_query_operator_selection(hci_con_handle_t acl_handle){ 15709c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1571a0ffb263SMatthias Ringwald if (!hfp_connection) { 15723c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 15733deb3ec6SMatthias Ringwald } 15743c65e705SMilanka Ringwald 1575a0ffb263SMatthias Ringwald switch (hfp_connection->hf_query_operator_state){ 1576ce263fc8SMatthias Ringwald case HFP_HF_QUERY_OPERATOR_FORMAT_NOT_SET: 1577a0ffb263SMatthias Ringwald hfp_connection->hf_query_operator_state = HFP_HF_QUERY_OPERATOR_SET_FORMAT; 1578ce263fc8SMatthias Ringwald break; 1579ce263fc8SMatthias Ringwald case HFP_HF_QUERY_OPERATOR_FORMAT_SET: 1580a0ffb263SMatthias Ringwald hfp_connection->hf_query_operator_state = HFP_HF_QUERY_OPERATOR_SEND_QUERY; 1581ce263fc8SMatthias Ringwald break; 1582ce263fc8SMatthias Ringwald default: 15833c65e705SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 1584ce263fc8SMatthias Ringwald } 15851c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 15863c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 15873deb3ec6SMatthias Ringwald } 15883deb3ec6SMatthias Ringwald 15893c65e705SMilanka Ringwald static uint8_t hfp_hf_set_report_extended_audio_gateway_error_result_code(hci_con_handle_t acl_handle, uint8_t enable){ 15909c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1591a0ffb263SMatthias Ringwald if (!hfp_connection) { 15923c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 15933deb3ec6SMatthias Ringwald } 1594a0ffb263SMatthias Ringwald hfp_connection->enable_extended_audio_gateway_error_report = enable; 15951c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 15963c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 15973deb3ec6SMatthias Ringwald } 15983deb3ec6SMatthias Ringwald 1599ce263fc8SMatthias Ringwald 16003c65e705SMilanka Ringwald uint8_t hfp_hf_enable_report_extended_audio_gateway_error_result_code(hci_con_handle_t acl_handle){ 16013c65e705SMilanka Ringwald return hfp_hf_set_report_extended_audio_gateway_error_result_code(acl_handle, 1); 1602ce263fc8SMatthias Ringwald } 1603ce263fc8SMatthias Ringwald 16043c65e705SMilanka Ringwald uint8_t hfp_hf_disable_report_extended_audio_gateway_error_result_code(hci_con_handle_t acl_handle){ 16053c65e705SMilanka Ringwald return hfp_hf_set_report_extended_audio_gateway_error_result_code(acl_handle, 0); 1606ce263fc8SMatthias Ringwald } 1607ce263fc8SMatthias Ringwald 160838200c1dSMilanka Ringwald static uint8_t hfp_hf_esco_s4_supported(hfp_connection_t * hfp_connection){ 1609aeb0f0feSMatthias Ringwald return (hfp_connection->remote_supported_features & (1<<HFP_AGSF_ESCO_S4)) && (hfp_hf_supported_features & (1 << HFP_HFSF_ESCO_S4)); 161038200c1dSMilanka Ringwald } 1611ce263fc8SMatthias Ringwald 16123c65e705SMilanka Ringwald uint8_t hfp_hf_establish_audio_connection(hci_con_handle_t acl_handle){ 16139c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1614a33eb0c4SMilanka Ringwald if (!hfp_connection) { 16153c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1616a33eb0c4SMilanka Ringwald } 1617ce263fc8SMatthias Ringwald 16183c65e705SMilanka Ringwald if (hfp_connection->state == HFP_AUDIO_CONNECTION_ESTABLISHED){ 16193c65e705SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 16203c65e705SMilanka Ringwald } 16213c65e705SMilanka Ringwald 16223c65e705SMilanka Ringwald if (hfp_connection->state >= HFP_W2_DISCONNECT_SCO){ 16233c65e705SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 16243c65e705SMilanka Ringwald } 1625f4412093SMatthias Ringwald if (has_codec_negotiation_feature(hfp_connection)) { 1626a0ffb263SMatthias Ringwald switch (hfp_connection->codecs_state) { 1627aa4dd815SMatthias Ringwald case HFP_CODECS_W4_AG_COMMON_CODEC: 1628aa4dd815SMatthias Ringwald break; 1629ec3bfc1aSMatthias Ringwald case HFP_CODECS_EXCHANGED: 1630ec3bfc1aSMatthias Ringwald hfp_connection->trigger_codec_exchange = 1; 1631ec3bfc1aSMatthias Ringwald break; 1632aa4dd815SMatthias Ringwald default: 16331cc65c4fSMatthias Ringwald hfp_connection->codec_confirmed = 0; 16341cc65c4fSMatthias Ringwald hfp_connection->suggested_codec = 0; 16351cc65c4fSMatthias Ringwald hfp_connection->negotiated_codec = 0; 16361cc65c4fSMatthias Ringwald hfp_connection->codecs_state = HFP_CODECS_RECEIVED_TRIGGER_CODEC_EXCHANGE; 163738200c1dSMilanka Ringwald hfp_connection->trigger_codec_exchange = 1; 1638aa4dd815SMatthias Ringwald break; 16393deb3ec6SMatthias Ringwald } 1640f4412093SMatthias Ringwald } else { 1641f4412093SMatthias Ringwald log_info("no codec negotiation feature, use CVSD"); 1642f4412093SMatthias Ringwald hfp_connection->codecs_state = HFP_CODECS_EXCHANGED; 1643f4412093SMatthias Ringwald hfp_connection->suggested_codec = HFP_CODEC_CVSD; 1644f4412093SMatthias Ringwald hfp_connection->codec_confirmed = hfp_connection->suggested_codec; 1645f4412093SMatthias Ringwald hfp_connection->negotiated_codec = hfp_connection->suggested_codec; 1646f4412093SMatthias Ringwald hfp_init_link_settings(hfp_connection, hfp_hf_esco_s4_supported(hfp_connection)); 1647f4412093SMatthias Ringwald hfp_connection->establish_audio_connection = 1; 1648f4412093SMatthias Ringwald hfp_connection->state = HFP_W4_SCO_CONNECTED; 1649ce263fc8SMatthias Ringwald } 1650ce263fc8SMatthias Ringwald 16511c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 16523c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 16533deb3ec6SMatthias Ringwald } 16543deb3ec6SMatthias Ringwald 16553c65e705SMilanka Ringwald uint8_t hfp_hf_release_audio_connection(hci_con_handle_t acl_handle){ 16569c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1657a33eb0c4SMilanka Ringwald if (!hfp_connection) { 16583c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1659a33eb0c4SMilanka Ringwald } 16600b4debbfSMilanka Ringwald if (hfp_connection->vra_state == HFP_VRA_VOICE_RECOGNITION_ACTIVATED){ 16610b4debbfSMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 16620b4debbfSMilanka Ringwald } 16630b4debbfSMilanka Ringwald uint8_t status = hfp_trigger_release_audio_connection(hfp_connection); 16640b4debbfSMilanka Ringwald if (status == ERROR_CODE_SUCCESS){ 16651c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 16660b4debbfSMilanka Ringwald } 16673c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 16683deb3ec6SMatthias Ringwald } 16693deb3ec6SMatthias Ringwald 16703c65e705SMilanka Ringwald uint8_t hfp_hf_answer_incoming_call(hci_con_handle_t acl_handle){ 16719c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1672a33eb0c4SMilanka Ringwald if (!hfp_connection) { 16733c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1674a33eb0c4SMilanka Ringwald } 1675ce263fc8SMatthias Ringwald 1676aeb0f0feSMatthias Ringwald if (hfp_hf_callsetup_status == HFP_CALLSETUP_STATUS_INCOMING_CALL_SETUP_IN_PROGRESS){ 1677a0ffb263SMatthias Ringwald hfp_connection->hf_answer_incoming_call = 1; 16781c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 1679ce263fc8SMatthias Ringwald } else { 1680aeb0f0feSMatthias Ringwald log_error("HFP HF: answering incoming call with wrong callsetup status %u", hfp_hf_callsetup_status); 16813c65e705SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 1682ce263fc8SMatthias Ringwald } 16833c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 1684ce263fc8SMatthias Ringwald } 1685ce263fc8SMatthias Ringwald 16863c65e705SMilanka Ringwald uint8_t hfp_hf_terminate_call(hci_con_handle_t acl_handle){ 16879c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1688a33eb0c4SMilanka Ringwald if (!hfp_connection) { 16893c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1690a33eb0c4SMilanka Ringwald } 1691a0ffb263SMatthias Ringwald hfp_connection->hf_send_chup = 1; 16921c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 16933c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 1694ce263fc8SMatthias Ringwald } 1695ce263fc8SMatthias Ringwald 16963c65e705SMilanka Ringwald uint8_t hfp_hf_reject_incoming_call(hci_con_handle_t acl_handle){ 16979c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1698a33eb0c4SMilanka Ringwald if (!hfp_connection) { 16993c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1700a33eb0c4SMilanka Ringwald } 1701ce263fc8SMatthias Ringwald 1702aeb0f0feSMatthias Ringwald if (hfp_hf_callsetup_status == HFP_CALLSETUP_STATUS_INCOMING_CALL_SETUP_IN_PROGRESS){ 1703a0ffb263SMatthias Ringwald hfp_connection->hf_send_chup = 1; 17041c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 1705ce263fc8SMatthias Ringwald } 17063c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 1707ce263fc8SMatthias Ringwald } 1708ce263fc8SMatthias Ringwald 17093c65e705SMilanka Ringwald uint8_t hfp_hf_user_busy(hci_con_handle_t acl_handle){ 17109c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1711a33eb0c4SMilanka Ringwald if (!hfp_connection) { 17123c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1713a33eb0c4SMilanka Ringwald } 1714ce263fc8SMatthias Ringwald 1715aeb0f0feSMatthias Ringwald if (hfp_hf_callsetup_status == HFP_CALLSETUP_STATUS_INCOMING_CALL_SETUP_IN_PROGRESS){ 1716a0ffb263SMatthias Ringwald hfp_connection->hf_send_chld_0 = 1; 17171c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 1718ce263fc8SMatthias Ringwald } 17193c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 1720ce263fc8SMatthias Ringwald } 1721ce263fc8SMatthias Ringwald 17223c65e705SMilanka Ringwald uint8_t hfp_hf_end_active_and_accept_other(hci_con_handle_t acl_handle){ 17239c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1724a33eb0c4SMilanka Ringwald if (!hfp_connection) { 17253c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1726a33eb0c4SMilanka Ringwald } 1727ce263fc8SMatthias Ringwald 1728aeb0f0feSMatthias Ringwald if ((hfp_hf_callsetup_status == HFP_CALLSETUP_STATUS_INCOMING_CALL_SETUP_IN_PROGRESS) || 1729aeb0f0feSMatthias Ringwald (hfp_hf_call_status == HFP_CALL_STATUS_ACTIVE_OR_HELD_CALL_IS_PRESENT)){ 1730a0ffb263SMatthias Ringwald hfp_connection->hf_send_chld_1 = 1; 17311c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 1732ce263fc8SMatthias Ringwald } 17333c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 1734ce263fc8SMatthias Ringwald } 1735ce263fc8SMatthias Ringwald 17363c65e705SMilanka Ringwald uint8_t hfp_hf_swap_calls(hci_con_handle_t acl_handle){ 17379c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1738a33eb0c4SMilanka Ringwald if (!hfp_connection) { 17393c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1740a33eb0c4SMilanka Ringwald } 1741ce263fc8SMatthias Ringwald 1742aeb0f0feSMatthias Ringwald if ((hfp_hf_callsetup_status == HFP_CALLSETUP_STATUS_INCOMING_CALL_SETUP_IN_PROGRESS) || 1743aeb0f0feSMatthias Ringwald (hfp_hf_call_status == HFP_CALL_STATUS_ACTIVE_OR_HELD_CALL_IS_PRESENT)){ 1744a0ffb263SMatthias Ringwald hfp_connection->hf_send_chld_2 = 1; 17451c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 1746ce263fc8SMatthias Ringwald } 17473c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 1748ce263fc8SMatthias Ringwald } 1749ce263fc8SMatthias Ringwald 17503c65e705SMilanka Ringwald uint8_t hfp_hf_join_held_call(hci_con_handle_t acl_handle){ 17519c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1752a33eb0c4SMilanka Ringwald if (!hfp_connection) { 17533c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1754a33eb0c4SMilanka Ringwald } 1755ce263fc8SMatthias Ringwald 1756aeb0f0feSMatthias Ringwald if ((hfp_hf_callsetup_status == HFP_CALLSETUP_STATUS_INCOMING_CALL_SETUP_IN_PROGRESS) || 1757aeb0f0feSMatthias Ringwald (hfp_hf_call_status == HFP_CALL_STATUS_ACTIVE_OR_HELD_CALL_IS_PRESENT)){ 1758a0ffb263SMatthias Ringwald hfp_connection->hf_send_chld_3 = 1; 17591c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 1760ce263fc8SMatthias Ringwald } 17613c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 1762ce263fc8SMatthias Ringwald } 1763ce263fc8SMatthias Ringwald 17643c65e705SMilanka Ringwald uint8_t hfp_hf_connect_calls(hci_con_handle_t acl_handle){ 17659c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1766a33eb0c4SMilanka Ringwald if (!hfp_connection) { 17673c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1768a33eb0c4SMilanka Ringwald } 1769ce263fc8SMatthias Ringwald 1770aeb0f0feSMatthias Ringwald if ((hfp_hf_callsetup_status == HFP_CALLSETUP_STATUS_INCOMING_CALL_SETUP_IN_PROGRESS) || 1771aeb0f0feSMatthias Ringwald (hfp_hf_call_status == HFP_CALL_STATUS_ACTIVE_OR_HELD_CALL_IS_PRESENT)){ 1772a0ffb263SMatthias Ringwald hfp_connection->hf_send_chld_4 = 1; 17731c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 1774ce263fc8SMatthias Ringwald } 17753c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 1776ce263fc8SMatthias Ringwald } 1777ce263fc8SMatthias Ringwald 17783c65e705SMilanka Ringwald uint8_t hfp_hf_release_call_with_index(hci_con_handle_t acl_handle, int index){ 17799c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1780a33eb0c4SMilanka Ringwald if (!hfp_connection) { 17813c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1782a33eb0c4SMilanka Ringwald } 1783667ec068SMatthias Ringwald 1784aeb0f0feSMatthias Ringwald if ((hfp_hf_callsetup_status == HFP_CALLSETUP_STATUS_INCOMING_CALL_SETUP_IN_PROGRESS) || 1785aeb0f0feSMatthias Ringwald (hfp_hf_call_status == HFP_CALL_STATUS_ACTIVE_OR_HELD_CALL_IS_PRESENT)){ 1786a0ffb263SMatthias Ringwald hfp_connection->hf_send_chld_x = 1; 1787a0ffb263SMatthias Ringwald hfp_connection->hf_send_chld_x_index = 10 + index; 17881c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 1789667ec068SMatthias Ringwald } 17903c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 1791667ec068SMatthias Ringwald } 1792667ec068SMatthias Ringwald 17933c65e705SMilanka Ringwald uint8_t hfp_hf_private_consultation_with_call(hci_con_handle_t acl_handle, int index){ 17949c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1795a33eb0c4SMilanka Ringwald if (!hfp_connection) { 17963c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1797a33eb0c4SMilanka Ringwald } 1798667ec068SMatthias Ringwald 1799aeb0f0feSMatthias Ringwald if ((hfp_hf_callsetup_status == HFP_CALLSETUP_STATUS_INCOMING_CALL_SETUP_IN_PROGRESS) || 1800aeb0f0feSMatthias Ringwald (hfp_hf_call_status == HFP_CALL_STATUS_ACTIVE_OR_HELD_CALL_IS_PRESENT)){ 1801a0ffb263SMatthias Ringwald hfp_connection->hf_send_chld_x = 1; 1802a0ffb263SMatthias Ringwald hfp_connection->hf_send_chld_x_index = 20 + index; 18031c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 1804667ec068SMatthias Ringwald } 18053c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 1806667ec068SMatthias Ringwald } 1807ce263fc8SMatthias Ringwald 18083c65e705SMilanka Ringwald uint8_t hfp_hf_dial_number(hci_con_handle_t acl_handle, char * number){ 18099c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1810a33eb0c4SMilanka Ringwald if (!hfp_connection) { 18113c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1812a33eb0c4SMilanka Ringwald } 1813ce263fc8SMatthias Ringwald 1814a0ffb263SMatthias Ringwald hfp_connection->hf_initiate_outgoing_call = 1; 1815aeb0f0feSMatthias Ringwald snprintf(hfp_hf_phone_number, sizeof(hfp_hf_phone_number), "%s", number); 18161c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 18173c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 1818ce263fc8SMatthias Ringwald } 1819ce263fc8SMatthias Ringwald 18203c65e705SMilanka Ringwald uint8_t hfp_hf_dial_memory(hci_con_handle_t acl_handle, int memory_id){ 18219c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1822a33eb0c4SMilanka Ringwald if (!hfp_connection) { 18233c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1824a33eb0c4SMilanka Ringwald } 1825ce263fc8SMatthias Ringwald 1826a0ffb263SMatthias Ringwald hfp_connection->hf_initiate_memory_dialing = 1; 1827a0ffb263SMatthias Ringwald hfp_connection->memory_id = memory_id; 1828a0ffb263SMatthias Ringwald 18291c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 18303c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 1831ce263fc8SMatthias Ringwald } 1832ce263fc8SMatthias Ringwald 18333c65e705SMilanka Ringwald uint8_t hfp_hf_redial_last_number(hci_con_handle_t acl_handle){ 18349c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1835a33eb0c4SMilanka Ringwald if (!hfp_connection) { 18363c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1837a33eb0c4SMilanka Ringwald } 1838ce263fc8SMatthias Ringwald 1839a0ffb263SMatthias Ringwald hfp_connection->hf_initiate_redial_last_number = 1; 18401c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 18413c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 1842ce263fc8SMatthias Ringwald } 1843ce263fc8SMatthias Ringwald 18443c65e705SMilanka Ringwald uint8_t hfp_hf_activate_call_waiting_notification(hci_con_handle_t acl_handle){ 18459c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1846a33eb0c4SMilanka Ringwald if (!hfp_connection) { 18473c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1848a33eb0c4SMilanka Ringwald } 1849ce263fc8SMatthias Ringwald 1850a0ffb263SMatthias Ringwald hfp_connection->hf_activate_call_waiting_notification = 1; 18511c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 18523c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 1853ce263fc8SMatthias Ringwald } 1854ce263fc8SMatthias Ringwald 1855ce263fc8SMatthias Ringwald 18563c65e705SMilanka Ringwald uint8_t hfp_hf_deactivate_call_waiting_notification(hci_con_handle_t acl_handle){ 18579c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1858a33eb0c4SMilanka Ringwald if (!hfp_connection) { 18593c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1860a33eb0c4SMilanka Ringwald } 1861ce263fc8SMatthias Ringwald 1862a0ffb263SMatthias Ringwald hfp_connection->hf_deactivate_call_waiting_notification = 1; 18631c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 18643c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 1865ce263fc8SMatthias Ringwald } 1866ce263fc8SMatthias Ringwald 1867ce263fc8SMatthias Ringwald 18683c65e705SMilanka Ringwald uint8_t hfp_hf_activate_calling_line_notification(hci_con_handle_t acl_handle){ 18699c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1870a33eb0c4SMilanka Ringwald if (!hfp_connection) { 18713c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1872a33eb0c4SMilanka Ringwald } 1873ce263fc8SMatthias Ringwald 1874a0ffb263SMatthias Ringwald hfp_connection->hf_activate_calling_line_notification = 1; 18751c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 18763c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 1877ce263fc8SMatthias Ringwald } 1878ce263fc8SMatthias Ringwald 18793c65e705SMilanka Ringwald uint8_t hfp_hf_deactivate_calling_line_notification(hci_con_handle_t acl_handle){ 18809c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1881a33eb0c4SMilanka Ringwald if (!hfp_connection) { 18823c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1883a33eb0c4SMilanka Ringwald } 1884ce263fc8SMatthias Ringwald 1885a0ffb263SMatthias Ringwald hfp_connection->hf_deactivate_calling_line_notification = 1; 18861c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 18873c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 1888ce263fc8SMatthias Ringwald } 1889ce263fc8SMatthias Ringwald 18906ba83b5eSMilanka Ringwald static bool hfp_hf_echo_canceling_and_noise_reduction_supported(hfp_connection_t * hfp_connection){ 18916ba83b5eSMilanka Ringwald int ag = get_bit(hfp_connection->remote_supported_features, HFP_AGSF_EC_NR_FUNCTION); 1892aeb0f0feSMatthias Ringwald int hf = get_bit(hfp_hf_supported_features, HFP_HFSF_EC_NR_FUNCTION); 18936ba83b5eSMilanka Ringwald return hf && ag; 1894ce263fc8SMatthias Ringwald } 1895ce263fc8SMatthias Ringwald 18963c65e705SMilanka Ringwald uint8_t hfp_hf_deactivate_echo_canceling_and_noise_reduction(hci_con_handle_t acl_handle){ 18979c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1898a33eb0c4SMilanka Ringwald if (!hfp_connection) { 18993c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1900a33eb0c4SMilanka Ringwald } 19016ba83b5eSMilanka Ringwald if (!hfp_hf_echo_canceling_and_noise_reduction_supported(hfp_connection)){ 19026ba83b5eSMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 19036ba83b5eSMilanka Ringwald } 1904ce263fc8SMatthias Ringwald 1905a0ffb263SMatthias Ringwald hfp_connection->hf_deactivate_echo_canceling_and_noise_reduction = 1; 19061c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 19073c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 1908ce263fc8SMatthias Ringwald } 1909ce263fc8SMatthias Ringwald 1910acd11d4aSMilanka Ringwald uint8_t hfp_hf_activate_voice_recognition(hci_con_handle_t acl_handle){ 1911fdda66c0SMilanka Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1912fdda66c0SMilanka Ringwald if (!hfp_connection) { 1913fdda66c0SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1914be55a11dSMilanka Ringwald } 1915013cc750SMilanka Ringwald if (hfp_connection->state < HFP_SERVICE_LEVEL_CONNECTION_ESTABLISHED || hfp_connection->state > HFP_AUDIO_CONNECTION_ESTABLISHED){ 1916013cc750SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 1917013cc750SMilanka Ringwald } 1918acd11d4aSMilanka Ringwald 1919acd11d4aSMilanka Ringwald bool enhanced_vra_supported = hfp_hf_enhanced_vra_flag_supported(hfp_connection); 1920acd11d4aSMilanka Ringwald bool legacy_vra_supported = hfp_hf_vra_flag_supported(hfp_connection); 1921acd11d4aSMilanka Ringwald if (!enhanced_vra_supported && !legacy_vra_supported){ 1922acd11d4aSMilanka Ringwald return ERROR_CODE_UNSUPPORTED_FEATURE_OR_PARAMETER_VALUE; 1923af97579eSMilanka Ringwald } 1924af97579eSMilanka Ringwald 1925498a8121SMilanka Ringwald switch (hfp_connection->vra_state){ 1926be55a11dSMilanka Ringwald case HFP_VRA_VOICE_RECOGNITION_OFF: 1927de9e0ea7SMilanka Ringwald case HFP_VRA_W2_SEND_VOICE_RECOGNITION_OFF: 1928fd4151d1SMilanka Ringwald hfp_connection->vra_state_requested = HFP_VRA_W2_SEND_VOICE_RECOGNITION_ACTIVATED; 1929acd11d4aSMilanka Ringwald hfp_connection->enhanced_voice_recognition_enabled = enhanced_vra_supported; 1930be55a11dSMilanka Ringwald break; 1931de9e0ea7SMilanka Ringwald case HFP_VRA_W4_VOICE_RECOGNITION_OFF: 1932de9e0ea7SMilanka Ringwald hfp_connection->activate_voice_recognition = true; 1933de9e0ea7SMilanka Ringwald break; 1934be55a11dSMilanka Ringwald default: 1935be55a11dSMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 1936be55a11dSMilanka Ringwald } 1937ce263fc8SMatthias Ringwald 1938af97579eSMilanka Ringwald hfp_hf_run_for_context(hfp_connection); 1939fdda66c0SMilanka Ringwald return ERROR_CODE_SUCCESS; 1940af97579eSMilanka Ringwald } 1941af97579eSMilanka Ringwald 1942acd11d4aSMilanka Ringwald uint8_t hfp_hf_enhanced_voice_recognition_report_ready_for_audio(hci_con_handle_t acl_handle){ 1943af97579eSMilanka Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1944af97579eSMilanka Ringwald if (!hfp_connection) { 1945af97579eSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1946af97579eSMilanka Ringwald } 194784fb9ac1SMilanka Ringwald 194884fb9ac1SMilanka Ringwald if (hfp_connection->emit_vra_enabled_after_audio_established){ 194984fb9ac1SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 195084fb9ac1SMilanka Ringwald } 195184fb9ac1SMilanka Ringwald 1952acd11d4aSMilanka Ringwald if (hfp_connection->state != HFP_AUDIO_CONNECTION_ESTABLISHED){ 195308a0b01cSMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 195408a0b01cSMilanka Ringwald } 1955acd11d4aSMilanka Ringwald 1956acd11d4aSMilanka Ringwald bool enhanced_vra_supported = hfp_hf_enhanced_vra_flag_supported(hfp_connection); 1957acd11d4aSMilanka Ringwald if (!enhanced_vra_supported){ 1958acd11d4aSMilanka Ringwald return ERROR_CODE_UNSUPPORTED_FEATURE_OR_PARAMETER_VALUE; 1959acd11d4aSMilanka Ringwald } 1960acd11d4aSMilanka Ringwald 1961acd11d4aSMilanka Ringwald switch (hfp_connection->vra_state){ 1962acd11d4aSMilanka Ringwald case HFP_VRA_VOICE_RECOGNITION_ACTIVATED: 1963acd11d4aSMilanka Ringwald case HFP_VRA_ENHANCED_VOICE_RECOGNITION_READY_FOR_AUDIO: 1964acd11d4aSMilanka Ringwald hfp_connection->vra_state_requested = HFP_VRA_W2_SEND_ENHANCED_VOICE_RECOGNITION_READY_FOR_AUDIO; 1965acd11d4aSMilanka Ringwald break; 1966acd11d4aSMilanka Ringwald default: 1967fdda66c0SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 1968af97579eSMilanka Ringwald } 1969013cc750SMilanka Ringwald 1970acd11d4aSMilanka Ringwald hfp_hf_run_for_context(hfp_connection); 1971acd11d4aSMilanka Ringwald return ERROR_CODE_SUCCESS; 1972acd11d4aSMilanka Ringwald } 1973acd11d4aSMilanka Ringwald 1974acd11d4aSMilanka Ringwald 1975acd11d4aSMilanka Ringwald uint8_t hfp_hf_deactivate_voice_recognition(hci_con_handle_t acl_handle){ 1976acd11d4aSMilanka Ringwald // return deactivate_voice_recognition(acl_handle, false); 1977acd11d4aSMilanka Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 1978acd11d4aSMilanka Ringwald if (!hfp_connection) { 1979acd11d4aSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 1980acd11d4aSMilanka Ringwald } 1981acd11d4aSMilanka Ringwald 198284fb9ac1SMilanka Ringwald if (hfp_connection->emit_vra_enabled_after_audio_established){ 198384fb9ac1SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 198484fb9ac1SMilanka Ringwald } 198584fb9ac1SMilanka Ringwald 1986acd11d4aSMilanka Ringwald if (hfp_connection->state < HFP_SERVICE_LEVEL_CONNECTION_ESTABLISHED || 1987acd11d4aSMilanka Ringwald hfp_connection->state > HFP_AUDIO_CONNECTION_ESTABLISHED){ 1988acd11d4aSMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 1989acd11d4aSMilanka Ringwald } 1990acd11d4aSMilanka Ringwald 1991acd11d4aSMilanka Ringwald bool enhanced_vra_supported = hfp_hf_enhanced_vra_flag_supported(hfp_connection); 1992acd11d4aSMilanka Ringwald bool legacy_vra_supported = hfp_hf_vra_flag_supported(hfp_connection); 1993acd11d4aSMilanka Ringwald if (!enhanced_vra_supported && !legacy_vra_supported){ 1994acd11d4aSMilanka Ringwald return ERROR_CODE_UNSUPPORTED_FEATURE_OR_PARAMETER_VALUE; 1995acd11d4aSMilanka Ringwald } 1996acd11d4aSMilanka Ringwald 1997fdda66c0SMilanka Ringwald switch (hfp_connection->vra_state){ 1998de9e0ea7SMilanka Ringwald case HFP_VRA_W2_SEND_VOICE_RECOGNITION_ACTIVATED: 1999fdda66c0SMilanka Ringwald case HFP_VRA_VOICE_RECOGNITION_ACTIVATED: 2000de9e0ea7SMilanka Ringwald case HFP_VRA_W2_SEND_ENHANCED_VOICE_RECOGNITION_READY_FOR_AUDIO: 2001de9e0ea7SMilanka Ringwald case HFP_VRA_ENHANCED_VOICE_RECOGNITION_READY_FOR_AUDIO: 2002fdda66c0SMilanka Ringwald hfp_connection->vra_state_requested = HFP_VRA_W2_SEND_VOICE_RECOGNITION_OFF; 2003fdda66c0SMilanka Ringwald break; 2004de9e0ea7SMilanka Ringwald 2005de9e0ea7SMilanka Ringwald case HFP_VRA_W4_VOICE_RECOGNITION_ACTIVATED: 2006de9e0ea7SMilanka Ringwald case HFP_VRA_W4_ENHANCED_VOICE_RECOGNITION_READY_FOR_AUDIO: 2007de9e0ea7SMilanka Ringwald hfp_connection->deactivate_voice_recognition = true; 2008de9e0ea7SMilanka Ringwald break; 2009de9e0ea7SMilanka Ringwald 2010de9e0ea7SMilanka Ringwald case HFP_VRA_VOICE_RECOGNITION_OFF: 2011de9e0ea7SMilanka Ringwald case HFP_VRA_W2_SEND_VOICE_RECOGNITION_OFF: 2012de9e0ea7SMilanka Ringwald case HFP_VRA_W4_VOICE_RECOGNITION_OFF: 2013fdda66c0SMilanka Ringwald default: 2014fdda66c0SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 2015fdda66c0SMilanka Ringwald } 2016fdda66c0SMilanka Ringwald 2017fdda66c0SMilanka Ringwald hfp_hf_run_for_context(hfp_connection); 2018fdda66c0SMilanka Ringwald return ERROR_CODE_SUCCESS; 2019af97579eSMilanka Ringwald } 2020af97579eSMilanka Ringwald 20213c65e705SMilanka Ringwald uint8_t hfp_hf_set_microphone_gain(hci_con_handle_t acl_handle, int gain){ 20229c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 2023a33eb0c4SMilanka Ringwald if (!hfp_connection) { 20243c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 2025a33eb0c4SMilanka Ringwald } 2026c8626498SMilanka Ringwald 20273c65e705SMilanka Ringwald if (hfp_connection->microphone_gain == gain) { 20283c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 20293c65e705SMilanka Ringwald } 20303c65e705SMilanka Ringwald 2031c1ab6cc1SMatthias Ringwald if ((gain < 0) || (gain > 15)){ 2032a0ffb263SMatthias Ringwald log_info("Valid range for a gain is [0..15]. Currently sent: %d", gain); 20333c65e705SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 2034a0ffb263SMatthias Ringwald } 20353c65e705SMilanka Ringwald 2036a0ffb263SMatthias Ringwald hfp_connection->microphone_gain = gain; 2037a0ffb263SMatthias Ringwald hfp_connection->send_microphone_gain = 1; 20381c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 20393c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 2040ce263fc8SMatthias Ringwald } 2041ce263fc8SMatthias Ringwald 20423c65e705SMilanka Ringwald uint8_t hfp_hf_set_speaker_gain(hci_con_handle_t acl_handle, int gain){ 20439c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 2044a33eb0c4SMilanka Ringwald if (!hfp_connection) { 20453c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 2046a33eb0c4SMilanka Ringwald } 2047c8626498SMilanka Ringwald 20483c65e705SMilanka Ringwald if (hfp_connection->speaker_gain == gain){ 20493c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 20503c65e705SMilanka Ringwald } 20513c65e705SMilanka Ringwald 2052c1ab6cc1SMatthias Ringwald if ((gain < 0) || (gain > 15)){ 2053a0ffb263SMatthias Ringwald log_info("Valid range for a gain is [0..15]. Currently sent: %d", gain); 20543c65e705SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 2055a0ffb263SMatthias Ringwald } 20563c65e705SMilanka Ringwald 2057a0ffb263SMatthias Ringwald hfp_connection->speaker_gain = gain; 2058a0ffb263SMatthias Ringwald hfp_connection->send_speaker_gain = 1; 20591c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 20603c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 2061ce263fc8SMatthias Ringwald } 2062ce263fc8SMatthias Ringwald 20633c65e705SMilanka Ringwald uint8_t hfp_hf_send_dtmf_code(hci_con_handle_t acl_handle, char code){ 20649c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 2065a33eb0c4SMilanka Ringwald if (!hfp_connection) { 20663c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 2067a33eb0c4SMilanka Ringwald } 2068a0ffb263SMatthias Ringwald hfp_connection->hf_send_dtmf_code = code; 20691c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 20703c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 2071ce263fc8SMatthias Ringwald } 2072ce263fc8SMatthias Ringwald 20733c65e705SMilanka Ringwald uint8_t hfp_hf_request_phone_number_for_voice_tag(hci_con_handle_t acl_handle){ 20749c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 2075a33eb0c4SMilanka Ringwald if (!hfp_connection) { 20763c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 2077a33eb0c4SMilanka Ringwald } 2078a0ffb263SMatthias Ringwald hfp_connection->hf_send_binp = 1; 20791c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 20803c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 2081ce263fc8SMatthias Ringwald } 20823deb3ec6SMatthias Ringwald 20833c65e705SMilanka Ringwald uint8_t hfp_hf_query_current_call_status(hci_con_handle_t acl_handle){ 20849c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 2085a33eb0c4SMilanka Ringwald if (!hfp_connection) { 20863c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 2087a33eb0c4SMilanka Ringwald } 2088a0ffb263SMatthias Ringwald hfp_connection->hf_send_clcc = 1; 20891c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 20903c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 2091667ec068SMatthias Ringwald } 2092667ec068SMatthias Ringwald 2093667ec068SMatthias Ringwald 20943c65e705SMilanka Ringwald uint8_t hfp_hf_rrh_query_status(hci_con_handle_t acl_handle){ 20959c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 2096a33eb0c4SMilanka Ringwald if (!hfp_connection) { 20973c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 2098a33eb0c4SMilanka Ringwald } 2099a0ffb263SMatthias Ringwald hfp_connection->hf_send_rrh = 1; 2100a0ffb263SMatthias Ringwald hfp_connection->hf_send_rrh_command = '?'; 21011c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 21023c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 2103667ec068SMatthias Ringwald } 2104667ec068SMatthias Ringwald 21053c65e705SMilanka Ringwald uint8_t hfp_hf_rrh_hold_call(hci_con_handle_t acl_handle){ 21069c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 2107a33eb0c4SMilanka Ringwald if (!hfp_connection) { 21083c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 2109a33eb0c4SMilanka Ringwald } 2110a0ffb263SMatthias Ringwald hfp_connection->hf_send_rrh = 1; 2111a0ffb263SMatthias Ringwald hfp_connection->hf_send_rrh_command = '0'; 21121c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 21133c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 2114667ec068SMatthias Ringwald } 2115667ec068SMatthias Ringwald 21163c65e705SMilanka Ringwald uint8_t hfp_hf_rrh_accept_held_call(hci_con_handle_t acl_handle){ 21179c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 2118a33eb0c4SMilanka Ringwald if (!hfp_connection) { 21193c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 2120a33eb0c4SMilanka Ringwald } 2121a0ffb263SMatthias Ringwald hfp_connection->hf_send_rrh = 1; 2122a0ffb263SMatthias Ringwald hfp_connection->hf_send_rrh_command = '1'; 21231c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 21243c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 2125667ec068SMatthias Ringwald } 2126667ec068SMatthias Ringwald 21273c65e705SMilanka Ringwald uint8_t hfp_hf_rrh_reject_held_call(hci_con_handle_t acl_handle){ 21289c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 2129a33eb0c4SMilanka Ringwald if (!hfp_connection) { 21303c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 2131a33eb0c4SMilanka Ringwald } 2132a0ffb263SMatthias Ringwald hfp_connection->hf_send_rrh = 1; 2133a0ffb263SMatthias Ringwald hfp_connection->hf_send_rrh_command = '2'; 21341c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 21353c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 2136667ec068SMatthias Ringwald } 2137667ec068SMatthias Ringwald 21383c65e705SMilanka Ringwald uint8_t hfp_hf_query_subscriber_number(hci_con_handle_t acl_handle){ 21399c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 2140a33eb0c4SMilanka Ringwald if (!hfp_connection) { 21413c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 2142a33eb0c4SMilanka Ringwald } 2143a0ffb263SMatthias Ringwald hfp_connection->hf_send_cnum = 1; 21441c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 21453c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 2146667ec068SMatthias Ringwald } 2147667ec068SMatthias Ringwald 21483c65e705SMilanka Ringwald uint8_t hfp_hf_set_hf_indicator(hci_con_handle_t acl_handle, int assigned_number, int value){ 21499c9c64c1SMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 2150a33eb0c4SMilanka Ringwald if (!hfp_connection) { 21513c65e705SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 2152a33eb0c4SMilanka Ringwald } 2153667ec068SMatthias Ringwald // find index for assigned number 2154667ec068SMatthias Ringwald int i; 2155aeb0f0feSMatthias Ringwald for (i = 0; i < hfp_hf_indicators_nr ; i++){ 2156aeb0f0feSMatthias Ringwald if (hfp_hf_indicators[i] == assigned_number){ 2157667ec068SMatthias Ringwald // set value 2158aeb0f0feSMatthias Ringwald hfp_hf_indicators_value[i] = value; 2159667ec068SMatthias Ringwald // mark for update 2160a0ffb263SMatthias Ringwald if (hfp_connection->state > HFP_LIST_GENERIC_STATUS_INDICATORS){ 2161a0ffb263SMatthias Ringwald hfp_connection->generic_status_update_bitmap |= (1<<i); 2162667ec068SMatthias Ringwald // send update 21631c6a0fc0SMatthias Ringwald hfp_hf_run_for_context(hfp_connection); 2164a0ffb263SMatthias Ringwald } 21653c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 2166667ec068SMatthias Ringwald } 2167667ec068SMatthias Ringwald } 21683c65e705SMilanka Ringwald return ERROR_CODE_SUCCESS; 2169667ec068SMatthias Ringwald } 2170667ec068SMatthias Ringwald 2171d7f6b5cbSMatthias Ringwald int hfp_hf_in_band_ringtone_active(hci_con_handle_t acl_handle){ 2172d7f6b5cbSMatthias Ringwald hfp_connection_t * hfp_connection = get_hfp_hf_connection_context_for_acl_handle(acl_handle); 2173d7f6b5cbSMatthias Ringwald if (!hfp_connection) { 2174d7f6b5cbSMatthias Ringwald return 0; 2175d7f6b5cbSMatthias Ringwald } 2176d7f6b5cbSMatthias Ringwald return get_bit(hfp_connection->remote_supported_features, HFP_AGSF_IN_BAND_RING_TONE); 2177d7f6b5cbSMatthias Ringwald } 217876cc1527SMatthias Ringwald 217976cc1527SMatthias Ringwald void hfp_hf_create_sdp_record(uint8_t * service, uint32_t service_record_handle, int rfcomm_channel_nr, const char * name, uint16_t supported_features, int wide_band_speech){ 218076cc1527SMatthias Ringwald if (!name){ 2181aeb0f0feSMatthias Ringwald name = hfp_hf_default_service_name; 218276cc1527SMatthias Ringwald } 218376cc1527SMatthias Ringwald hfp_create_sdp_record(service, service_record_handle, BLUETOOTH_SERVICE_CLASS_HANDSFREE, rfcomm_channel_nr, name); 218476cc1527SMatthias Ringwald 218576cc1527SMatthias Ringwald // Construct SupportedFeatures for SDP bitmap: 218676cc1527SMatthias Ringwald // 218776cc1527SMatthias Ringwald // "The values of the “SupportedFeatures” bitmap given in Table 5.4 shall be the same as the values 218876cc1527SMatthias Ringwald // of the Bits 0 to 4 of the unsolicited result code +BRSF" 218976cc1527SMatthias Ringwald // 219076cc1527SMatthias Ringwald // Wide band speech (bit 5) requires Codec negotiation 219176cc1527SMatthias Ringwald // 219276cc1527SMatthias Ringwald uint16_t sdp_features = supported_features & 0x1f; 2193ef3ae4ebSMilanka Ringwald if ( (wide_band_speech != 0) && (supported_features & (1 << HFP_HFSF_CODEC_NEGOTIATION))){ 219476cc1527SMatthias Ringwald sdp_features |= 1 << 5; 219576cc1527SMatthias Ringwald } 2196ef3ae4ebSMilanka Ringwald 2197ef3ae4ebSMilanka Ringwald if (supported_features & (1 << HFP_HFSF_ENHANCED_VOICE_RECOGNITION_STATUS)){ 219856f1adacSMilanka Ringwald sdp_features |= 1 << 6; 2199ef3ae4ebSMilanka Ringwald } 2200ef3ae4ebSMilanka Ringwald 2201ef3ae4ebSMilanka Ringwald if (supported_features & (1 << HFP_HFSF_VOICE_RECOGNITION_TEXT)){ 220256f1adacSMilanka Ringwald sdp_features |= 1 << 7; 2203ef3ae4ebSMilanka Ringwald } 2204ef3ae4ebSMilanka Ringwald 220576cc1527SMatthias Ringwald de_add_number(service, DE_UINT, DE_SIZE_16, 0x0311); // Hands-Free Profile - SupportedFeatures 220676cc1527SMatthias Ringwald de_add_number(service, DE_UINT, DE_SIZE_16, sdp_features); 220776cc1527SMatthias Ringwald } 220876cc1527SMatthias Ringwald 220976cc1527SMatthias Ringwald void hfp_hf_register_packet_handler(btstack_packet_handler_t callback){ 221068466199SMilanka Ringwald btstack_assert(callback != NULL); 221168466199SMilanka Ringwald 221276cc1527SMatthias Ringwald hfp_hf_callback = callback; 221376cc1527SMatthias Ringwald hfp_set_hf_callback(callback); 221476cc1527SMatthias Ringwald } 2215