1 /******************************************************************************
2 *
3 * Copyright (c) 2014 The Android Open Source Project
4 * Copyright 2009-2012 Broadcom Corporation
5 *
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at:
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
17 *
18 ******************************************************************************/
19
20 /*******************************************************************************
21 *
22 * Filename: btif_hf_client.c
23 *
24 * Description: Handsfree Profile (HF role) Bluetooth Interface
25 *
26 * Notes:
27 * a) Lifecycle of a control block
28 * Control block handles the lifecycle for a particular remote device's
29 * connection. The connection can go via the classic phases but more
30 * importantly there's only two messages from BTA that affect this.
31 * BTA_HF_CLIENT_OPEN_EVT and BTA_HF_CLIENT_CLOSE_EVT. Since the API between
32 * BTIF and BTA is controlled entirely by handles it's important to know where
33 * the handles are created and destroyed. Handles can be created at two
34 * locations:
35 * -- While connect() is called from BTIF. This is an outgoing connection
36 * -- While accepting an incoming connection (see BTA_HF_CLIENT_OPEN_EVT
37 * handling).
38 *
39 * The destruction or rather reuse of handles can be done when
40 * BTA_HF_CLIENT_CLOSE_EVT is called. Refer to the event handling for details
41 * of this.
42 *
43 ******************************************************************************/
44
45 #define LOG_TAG "bt_btif_hfc"
46
47 #include <stdlib.h>
48 #include <string.h>
49
50 #include <hardware/bluetooth.h>
51 #include <hardware/bt_hf_client.h>
52
53 #include "bt_utils.h"
54 #include "bta_hf_client_api.h"
55 #include "btif_common.h"
56 #include "btif_profile_queue.h"
57 #include "btif_util.h"
58 #include "osi/include/osi.h"
59 #include "osi/include/properties.h"
60
61 /*******************************************************************************
62 * Constants & Macros
63 ******************************************************************************/
64
65 #ifndef BTIF_HF_CLIENT_SERVICE_NAME
66 #define BTIF_HF_CLIENT_SERVICE_NAME ("Handsfree")
67 #endif
68
69 #ifndef BTIF_HF_CLIENT_SECURITY
70 #define BTIF_HF_CLIENT_SECURITY (BTA_SEC_AUTHENTICATE | BTA_SEC_ENCRYPT)
71 #endif
72
73 #ifndef BTIF_HF_CLIENT_FEATURES
74 #define BTIF_HF_CLIENT_FEATURES \
75 (BTA_HF_CLIENT_FEAT_ECNR | BTA_HF_CLIENT_FEAT_3WAY | \
76 BTA_HF_CLIENT_FEAT_CLI | BTA_HF_CLIENT_FEAT_VREC | BTA_HF_CLIENT_FEAT_VOL | \
77 BTA_HF_CLIENT_FEAT_ECS | BTA_HF_CLIENT_FEAT_ECC | BTA_HF_CLIENT_FEAT_CODEC)
78 #endif
79
80 /*******************************************************************************
81 * Local type definitions
82 ******************************************************************************/
83 /* BTIF-HF control block to map bdaddr to BTA handle */
84 typedef struct {
85 uint16_t handle; // Handle obtained frm the BTA
86 RawAddress peer_bda; // Device corresponding to handle
87 bthf_client_connection_state_t state; // State of current connection
88 tBTA_HF_CLIENT_PEER_FEAT peer_feat; // HF features
89 tBTA_HF_CLIENT_CHLD_FEAT chld_feat; // AT+CHLD=<> command features
90 } btif_hf_client_cb_t;
91
92 /* Max devices supported by BTIF (useful to match the value in BTA) */
93 #define HF_CLIENT_MAX_DEVICES 10
94 typedef struct {
95 btif_hf_client_cb_t cb[HF_CLIENT_MAX_DEVICES];
96 } btif_hf_client_cb_arr_t;
97
98 /******************************************************************************
99 * Local function declarations
100 ******************************************************************************/
101 btif_hf_client_cb_t* btif_hf_client_get_cb_by_handle(uint16_t handle);
102 btif_hf_client_cb_t* btif_hf_client_get_cb_by_bda(const RawAddress& addr);
103 bool is_connected(const btif_hf_client_cb_t* cb);
104
105 /*******************************************************************************
106 * Static variables
107 ******************************************************************************/
108 static bthf_client_callbacks_t* bt_hf_client_callbacks = NULL;
109
110 char btif_hf_client_version[PROPERTY_VALUE_MAX];
111
dump_hf_client_conn_state(uint16_t event)112 static const char* dump_hf_client_conn_state(uint16_t event) {
113 switch (event) {
114 CASE_RETURN_STR(BTHF_CLIENT_CONNECTION_STATE_DISCONNECTED)
115 CASE_RETURN_STR(BTHF_CLIENT_CONNECTION_STATE_CONNECTING)
116 CASE_RETURN_STR(BTHF_CLIENT_CONNECTION_STATE_CONNECTED)
117 CASE_RETURN_STR(BTHF_CLIENT_CONNECTION_STATE_SLC_CONNECTED)
118 CASE_RETURN_STR(BTHF_CLIENT_CONNECTION_STATE_DISCONNECTING)
119 default:
120 return "UNKNOWN MSG ID";
121 }
122 }
123
124 #define CHECK_BTHF_CLIENT_INIT() \
125 do { \
126 if (bt_hf_client_callbacks == NULL) { \
127 BTIF_TRACE_WARNING("BTHF CLIENT: %s: not initialized", __func__); \
128 return BT_STATUS_NOT_READY; \
129 } else { \
130 BTIF_TRACE_EVENT("BTHF CLIENT: %s", __func__); \
131 } \
132 } while (0)
133
134 #define CHECK_BTHF_CLIENT_SLC_CONNECTED(cb) \
135 do { \
136 if (bt_hf_client_callbacks == NULL) { \
137 BTIF_TRACE_WARNING("BTHF CLIENT: %s: not initialized", __func__); \
138 return BT_STATUS_NOT_READY; \
139 } else if ((cb)->state != BTHF_CLIENT_CONNECTION_STATE_SLC_CONNECTED) { \
140 BTIF_TRACE_WARNING("BTHF CLIENT: %s: SLC connection not up. state=%s", \
141 __func__, dump_hf_client_conn_state((cb)->state)); \
142 return BT_STATUS_NOT_READY; \
143 } else { \
144 BTIF_TRACE_EVENT("BTHF CLIENT: %s", __func__); \
145 } \
146 } while (0)
147
148 static btif_hf_client_cb_arr_t btif_hf_client_cb_arr;
149
150 /*******************************************************************************
151 * Static functions
152 ******************************************************************************/
153
154 /*******************************************************************************
155 *
156 * Function btif_in_hf_client_generic_evt
157 *
158 * Description Processes generic events to be sent to JNI that are not
159 * triggered from the BTA.
160 * Always runs in BTIF context
161 *
162 * Returns void
163 *
164 ******************************************************************************/
btif_in_hf_client_generic_evt(uint16_t event,char * p_param)165 static void btif_in_hf_client_generic_evt(uint16_t event, char* p_param) {
166 BTIF_TRACE_DEBUG("%s", __func__);
167 RawAddress* bd_addr = (RawAddress*)p_param;
168 btif_hf_client_cb_t* cb = btif_hf_client_get_cb_by_bda(*bd_addr);
169 if (cb == NULL || !is_connected(cb)) {
170 BTIF_TRACE_ERROR("%s: failed to find block for bda", __func__);
171 }
172
173 BTIF_TRACE_EVENT("%s: event=%d", __func__, event);
174 switch (event) {
175 case BTIF_HF_CLIENT_CB_AUDIO_CONNECTING: {
176 HAL_CBACK(bt_hf_client_callbacks, audio_state_cb, &cb->peer_bda,
177 (bthf_client_audio_state_t)BTHF_CLIENT_AUDIO_STATE_CONNECTING);
178 } break;
179 default: {
180 BTIF_TRACE_WARNING("%s: : Unknown event 0x%x", __func__, event);
181 } break;
182 }
183 }
184
185 /*******************************************************************************
186 * Functions
187 ******************************************************************************/
is_connected(const btif_hf_client_cb_t * cb)188 bool is_connected(const btif_hf_client_cb_t* cb) {
189 if ((cb->state == BTHF_CLIENT_CONNECTION_STATE_CONNECTED) ||
190 (cb->state == BTHF_CLIENT_CONNECTION_STATE_SLC_CONNECTED))
191 return true;
192
193 BTIF_TRACE_ERROR("%s: not connected!", __func__);
194 return false;
195 }
196
197 /*******************************************************************************
198 *
199 * Function btif_hf_client_get_cb_by_handle
200 *
201 * Description Get control block by handle
202 *
203 * Returns btif_hf_client_cb_t pointer if available NULL otherwise
204 *
205 ******************************************************************************/
btif_hf_client_get_cb_by_handle(uint16_t handle)206 btif_hf_client_cb_t* btif_hf_client_get_cb_by_handle(uint16_t handle) {
207 BTIF_TRACE_DEBUG("%s: cb by handle %d", __func__, handle);
208 for (int i = 0; i < HF_CLIENT_MAX_DEVICES; i++) {
209 // Block is valid only if it is allocated i.e. state is not DISCONNECTED
210 if (btif_hf_client_cb_arr.cb[i].state !=
211 BTHF_CLIENT_CONNECTION_STATE_DISCONNECTED &&
212 btif_hf_client_cb_arr.cb[i].handle == handle) {
213 return &btif_hf_client_cb_arr.cb[i];
214 }
215 }
216 BTIF_TRACE_ERROR("%s: could not find block for handle %d", __func__, handle);
217 return NULL;
218 }
219
220 /*******************************************************************************
221 *
222 * Function btif_hf_client_get_cb_by_bda
223 *
224 * Description Get control block by bda
225 *
226 * Returns btif_hf_client_cb_t pointer if available NULL otherwise
227 *
228 ******************************************************************************/
btif_hf_client_get_cb_by_bda(const RawAddress & bd_addr)229 btif_hf_client_cb_t* btif_hf_client_get_cb_by_bda(const RawAddress& bd_addr) {
230 VLOG(1) << __func__ << " incoming addr " << bd_addr;
231
232 for (int i = 0; i < HF_CLIENT_MAX_DEVICES; i++) {
233 // Block is valid only if it is allocated i.e. state is not DISCONNECTED
234 if (btif_hf_client_cb_arr.cb[i].state !=
235 BTHF_CLIENT_CONNECTION_STATE_DISCONNECTED &&
236 btif_hf_client_cb_arr.cb[i].peer_bda == bd_addr) {
237 return &btif_hf_client_cb_arr.cb[i];
238 }
239 }
240 BTIF_TRACE_ERROR("%s: could not find block for bdaddr", __func__);
241 return NULL;
242 }
243
244 /*******************************************************************************
245 *
246 * Function btif_hf_client_allocate_cb
247 *
248 * Description Get control block by bda
249 *
250 * Returns btif_hf_client_cb_t pointer if available NULL otherwise
251 *
252 ******************************************************************************/
btif_hf_client_allocate_cb()253 btif_hf_client_cb_t* btif_hf_client_allocate_cb() {
254 for (int i = 0; i < HF_CLIENT_MAX_DEVICES; i++) {
255 btif_hf_client_cb_t* cb = &btif_hf_client_cb_arr.cb[i];
256 if (cb->state == BTHF_CLIENT_CONNECTION_STATE_DISCONNECTED) {
257 return cb;
258 }
259 }
260 BTIF_TRACE_ERROR("%s: unable to allocate control block", __func__);
261 return NULL;
262 }
263
264 /*****************************************************************************
265 *
266 * btif hf api functions (no context switch)
267 *
268 ****************************************************************************/
269
270 /*******************************************************************************
271 *
272 * Function btif_hf_client_init
273 *
274 * Description initializes the hf interface
275 *
276 * Returns bt_status_t
277 *
278 ******************************************************************************/
init(bthf_client_callbacks_t * callbacks)279 static bt_status_t init(bthf_client_callbacks_t* callbacks) {
280 BTIF_TRACE_EVENT("%s", __func__);
281
282 bt_hf_client_callbacks = callbacks;
283
284 btif_enable_service(BTA_HFP_HS_SERVICE_ID);
285
286 memset(&btif_hf_client_cb_arr, 0, sizeof(btif_hf_client_cb_arr_t));
287
288 return BT_STATUS_SUCCESS;
289 }
290
291 /*******************************************************************************
292 *
293 * Function connect
294 *
295 * Description connect to audio gateway
296 *
297 * Returns bt_status_t
298 *
299 ******************************************************************************/
connect_int(RawAddress * bd_addr,uint16_t uuid)300 static bt_status_t connect_int(RawAddress* bd_addr, uint16_t uuid) {
301 btif_hf_client_cb_t* cb = btif_hf_client_allocate_cb();
302 if (cb == NULL) {
303 BTIF_TRACE_ERROR("%s: could not allocate block!", __func__);
304 return BT_STATUS_BUSY;
305 }
306
307 cb->peer_bda = *bd_addr;
308 if (is_connected(cb)) return BT_STATUS_BUSY;
309
310 cb->state = BTHF_CLIENT_CONNECTION_STATE_CONNECTING;
311 cb->peer_bda = *bd_addr;
312
313 /* Open HF connection to remote device and get the relevant handle.
314 * The handle is valid until we have called BTA_HfClientClose or the LL
315 * has notified us of channel close due to remote closing, error etc.
316 */
317 BTA_HfClientOpen(cb->peer_bda, BTIF_HF_CLIENT_SECURITY, &cb->handle);
318
319 return BT_STATUS_SUCCESS;
320 }
321
connect(RawAddress * bd_addr)322 static bt_status_t connect(RawAddress* bd_addr) {
323 BTIF_TRACE_EVENT("HFP Client version is %s", btif_hf_client_version);
324 CHECK_BTHF_CLIENT_INIT();
325 return btif_queue_connect(UUID_SERVCLASS_HF_HANDSFREE, bd_addr, connect_int);
326 }
327
328 /*******************************************************************************
329 *
330 * Function disconnect
331 *
332 * Description disconnect from audio gateway
333 *
334 * Returns bt_status_t
335 *
336 ******************************************************************************/
disconnect(const RawAddress * bd_addr)337 static bt_status_t disconnect(const RawAddress* bd_addr) {
338 CHECK_BTHF_CLIENT_INIT();
339
340 btif_hf_client_cb_t* cb = btif_hf_client_get_cb_by_bda(*bd_addr);
341 if (cb != NULL) {
342 BTA_HfClientClose(cb->handle);
343 return BT_STATUS_SUCCESS;
344 } else {
345 return BT_STATUS_BUSY;
346 }
347 }
348
349 /*******************************************************************************
350 *
351 * Function connect_audio
352 *
353 * Description create an audio connection
354 *
355 * Returns bt_status_t
356 *
357 ******************************************************************************/
connect_audio(const RawAddress * bd_addr)358 static bt_status_t connect_audio(const RawAddress* bd_addr) {
359 btif_hf_client_cb_t* cb = btif_hf_client_get_cb_by_bda(*bd_addr);
360 if (cb == NULL || !is_connected(cb)) return BT_STATUS_FAIL;
361
362 CHECK_BTHF_CLIENT_SLC_CONNECTED(cb);
363
364 if ((BTIF_HF_CLIENT_FEATURES & BTA_HF_CLIENT_FEAT_CODEC) &&
365 (cb->peer_feat & BTA_HF_CLIENT_PEER_CODEC)) {
366 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_BCC, 0, 0, NULL);
367 } else {
368 BTA_HfClientAudioOpen(cb->handle);
369 }
370
371 /* Inform the application that the audio connection has been initiated
372 * successfully */
373 btif_transfer_context(btif_in_hf_client_generic_evt,
374 BTIF_HF_CLIENT_CB_AUDIO_CONNECTING, (char*)bd_addr,
375 sizeof(RawAddress), NULL);
376 return BT_STATUS_SUCCESS;
377 }
378
379 /*******************************************************************************
380 *
381 * Function disconnect_audio
382 *
383 * Description close the audio connection
384 *
385 * Returns bt_status_t
386 *
387 ******************************************************************************/
disconnect_audio(const RawAddress * bd_addr)388 static bt_status_t disconnect_audio(const RawAddress* bd_addr) {
389 btif_hf_client_cb_t* cb = btif_hf_client_get_cb_by_bda(*bd_addr);
390 if (cb == NULL || !is_connected(cb)) return BT_STATUS_FAIL;
391
392 CHECK_BTHF_CLIENT_SLC_CONNECTED(cb);
393
394 BTA_HfClientAudioClose(cb->handle);
395 return BT_STATUS_SUCCESS;
396 }
397
398 /*******************************************************************************
399 *
400 * Function start_voice_recognition
401 *
402 * Description start voice recognition
403 *
404 * Returns bt_status_t
405 *
406 ******************************************************************************/
start_voice_recognition(const RawAddress * bd_addr)407 static bt_status_t start_voice_recognition(const RawAddress* bd_addr) {
408 btif_hf_client_cb_t* cb = btif_hf_client_get_cb_by_bda(*bd_addr);
409 if (cb == NULL || !is_connected(cb)) return BT_STATUS_FAIL;
410
411 CHECK_BTHF_CLIENT_SLC_CONNECTED(cb);
412
413 if (cb->peer_feat & BTA_HF_CLIENT_PEER_FEAT_VREC) {
414 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_BVRA, 1, 0, NULL);
415 return BT_STATUS_SUCCESS;
416 }
417 return BT_STATUS_UNSUPPORTED;
418 }
419
420 /*******************************************************************************
421 *
422 * Function stop_voice_recognition
423 *
424 * Description stop voice recognition
425 *
426 * Returns bt_status_t
427 *
428 ******************************************************************************/
stop_voice_recognition(const RawAddress * bd_addr)429 static bt_status_t stop_voice_recognition(const RawAddress* bd_addr) {
430 btif_hf_client_cb_t* cb = btif_hf_client_get_cb_by_bda(*bd_addr);
431 if (cb == NULL || !is_connected(cb)) return BT_STATUS_FAIL;
432
433 CHECK_BTHF_CLIENT_SLC_CONNECTED(cb);
434
435 if (cb->peer_feat & BTA_HF_CLIENT_PEER_FEAT_VREC) {
436 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_BVRA, 0, 0, NULL);
437 return BT_STATUS_SUCCESS;
438 }
439 return BT_STATUS_UNSUPPORTED;
440 }
441
442 /*******************************************************************************
443 *
444 * Function volume_control
445 *
446 * Description volume control
447 *
448 * Returns bt_status_t
449 *
450 ******************************************************************************/
volume_control(const RawAddress * bd_addr,bthf_client_volume_type_t type,int volume)451 static bt_status_t volume_control(const RawAddress* bd_addr,
452 bthf_client_volume_type_t type, int volume) {
453 btif_hf_client_cb_t* cb = btif_hf_client_get_cb_by_bda(*bd_addr);
454 if (cb == NULL || !is_connected(cb)) return BT_STATUS_FAIL;
455
456 CHECK_BTHF_CLIENT_SLC_CONNECTED(cb);
457
458 switch (type) {
459 case BTHF_CLIENT_VOLUME_TYPE_SPK:
460 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_VGS, volume, 0, NULL);
461 break;
462 case BTHF_CLIENT_VOLUME_TYPE_MIC:
463 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_VGM, volume, 0, NULL);
464 break;
465 default:
466 return BT_STATUS_UNSUPPORTED;
467 }
468
469 return BT_STATUS_SUCCESS;
470 }
471
472 /*******************************************************************************
473 *
474 * Function dial
475 *
476 * Description place a call
477 *
478 * Returns bt_status_t
479 *
480 ******************************************************************************/
dial(UNUSED_ATTR const RawAddress * bd_addr,const char * number)481 static bt_status_t dial(UNUSED_ATTR const RawAddress* bd_addr,
482 const char* number) {
483 btif_hf_client_cb_t* cb = btif_hf_client_get_cb_by_bda(*bd_addr);
484 if (cb == NULL || !is_connected(cb)) return BT_STATUS_FAIL;
485
486 CHECK_BTHF_CLIENT_SLC_CONNECTED(cb);
487
488 if (number) {
489 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_ATD, 0, 0, number);
490 } else {
491 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_BLDN, 0, 0, NULL);
492 }
493 return BT_STATUS_SUCCESS;
494 }
495
496 /*******************************************************************************
497 *
498 * Function dial_memory
499 *
500 * Description place a call with number specified by location (speed dial)
501 *
502 * Returns bt_status_t
503 *
504 ******************************************************************************/
dial_memory(const RawAddress * bd_addr,int location)505 static bt_status_t dial_memory(const RawAddress* bd_addr, int location) {
506 btif_hf_client_cb_t* cb = btif_hf_client_get_cb_by_bda(*bd_addr);
507 if (cb == NULL || !is_connected(cb)) return BT_STATUS_FAIL;
508
509 CHECK_BTHF_CLIENT_SLC_CONNECTED(cb);
510
511 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_ATD, location, 0, NULL);
512 return BT_STATUS_SUCCESS;
513 }
514
515 /*******************************************************************************
516 *
517 * Function handle_call_action
518 *
519 * Description handle specified call related action
520 *
521 * Returns bt_status_t
522 *
523 ******************************************************************************/
handle_call_action(const RawAddress * bd_addr,bthf_client_call_action_t action,int idx)524 static bt_status_t handle_call_action(const RawAddress* bd_addr,
525 bthf_client_call_action_t action,
526 int idx) {
527 btif_hf_client_cb_t* cb = btif_hf_client_get_cb_by_bda(*bd_addr);
528 if (cb == NULL || !is_connected(cb)) return BT_STATUS_FAIL;
529
530 CHECK_BTHF_CLIENT_SLC_CONNECTED(cb);
531
532 switch (action) {
533 case BTHF_CLIENT_CALL_ACTION_CHLD_0:
534 if (cb->chld_feat & BTA_HF_CLIENT_CHLD_REL) {
535 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_CHLD, 0, 0, NULL);
536 break;
537 }
538 return BT_STATUS_UNSUPPORTED;
539 case BTHF_CLIENT_CALL_ACTION_CHLD_1:
540 // CHLD 1 is mandatory for 3 way calling
541 if (cb->peer_feat & BTA_HF_CLIENT_PEER_FEAT_3WAY) {
542 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_CHLD, 1, 0, NULL);
543 break;
544 }
545 return BT_STATUS_UNSUPPORTED;
546 case BTHF_CLIENT_CALL_ACTION_CHLD_2:
547 // CHLD 2 is mandatory for 3 way calling
548 if (cb->peer_feat & BTA_HF_CLIENT_PEER_FEAT_3WAY) {
549 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_CHLD, 2, 0, NULL);
550 break;
551 }
552 return BT_STATUS_UNSUPPORTED;
553 case BTHF_CLIENT_CALL_ACTION_CHLD_3:
554 if (cb->chld_feat & BTA_HF_CLIENT_CHLD_MERGE) {
555 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_CHLD, 3, 0, NULL);
556 break;
557 }
558 return BT_STATUS_UNSUPPORTED;
559 case BTHF_CLIENT_CALL_ACTION_CHLD_4:
560 if (cb->chld_feat & BTA_HF_CLIENT_CHLD_MERGE_DETACH) {
561 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_CHLD, 4, 0, NULL);
562 break;
563 }
564 return BT_STATUS_UNSUPPORTED;
565 case BTHF_CLIENT_CALL_ACTION_CHLD_1x:
566 if (cb->peer_feat & BTA_HF_CLIENT_PEER_ECC) {
567 if (idx < 1) {
568 return BT_STATUS_FAIL;
569 }
570 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_CHLD, 1, idx, NULL);
571 break;
572 }
573 return BT_STATUS_UNSUPPORTED;
574 case BTHF_CLIENT_CALL_ACTION_CHLD_2x:
575 if (cb->peer_feat & BTA_HF_CLIENT_PEER_ECC) {
576 if (idx < 1) {
577 return BT_STATUS_FAIL;
578 }
579 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_CHLD, 2, idx, NULL);
580 break;
581 }
582 return BT_STATUS_UNSUPPORTED;
583 case BTHF_CLIENT_CALL_ACTION_ATA:
584 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_ATA, 0, 0, NULL);
585 break;
586 case BTHF_CLIENT_CALL_ACTION_CHUP:
587 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_CHUP, 0, 0, NULL);
588 break;
589 case BTHF_CLIENT_CALL_ACTION_BTRH_0:
590 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_BTRH, 0, 0, NULL);
591 break;
592 case BTHF_CLIENT_CALL_ACTION_BTRH_1:
593 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_BTRH, 1, 0, NULL);
594 break;
595 case BTHF_CLIENT_CALL_ACTION_BTRH_2:
596 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_BTRH, 2, 0, NULL);
597 break;
598 default:
599 return BT_STATUS_FAIL;
600 }
601
602 return BT_STATUS_SUCCESS;
603 }
604
605 /*******************************************************************************
606 *
607 * Function query_current_calls
608 *
609 * Description query list of current calls
610 *
611 * Returns bt_status_t
612 *
613 ******************************************************************************/
query_current_calls(UNUSED_ATTR const RawAddress * bd_addr)614 static bt_status_t query_current_calls(UNUSED_ATTR const RawAddress* bd_addr) {
615 btif_hf_client_cb_t* cb = btif_hf_client_get_cb_by_bda(*bd_addr);
616 if (cb == NULL || !is_connected(cb)) return BT_STATUS_FAIL;
617
618 CHECK_BTHF_CLIENT_SLC_CONNECTED(cb);
619
620 if (cb->peer_feat & BTA_HF_CLIENT_PEER_ECS) {
621 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_CLCC, 0, 0, NULL);
622 return BT_STATUS_SUCCESS;
623 }
624
625 return BT_STATUS_UNSUPPORTED;
626 }
627
628 /*******************************************************************************
629 *
630 * Function query_current_operator_name
631 *
632 * Description query current selected operator name
633 *
634 * Returns bt_status_t
635 *
636 ******************************************************************************/
query_current_operator_name(const RawAddress * bd_addr)637 static bt_status_t query_current_operator_name(const RawAddress* bd_addr) {
638 btif_hf_client_cb_t* cb = btif_hf_client_get_cb_by_bda(*bd_addr);
639 if (cb == NULL || !is_connected(cb)) return BT_STATUS_FAIL;
640
641 CHECK_BTHF_CLIENT_SLC_CONNECTED(cb);
642
643 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_COPS, 0, 0, NULL);
644 return BT_STATUS_SUCCESS;
645 }
646
647 /*******************************************************************************
648 *
649 * Function retieve_subscriber_info
650 *
651 * Description retrieve subscriber number information
652 *
653 * Returns bt_status_t
654 *
655 ******************************************************************************/
retrieve_subscriber_info(const RawAddress * bd_addr)656 static bt_status_t retrieve_subscriber_info(const RawAddress* bd_addr) {
657 btif_hf_client_cb_t* cb = btif_hf_client_get_cb_by_bda(*bd_addr);
658 if (cb == NULL || !is_connected(cb)) return BT_STATUS_FAIL;
659
660 CHECK_BTHF_CLIENT_SLC_CONNECTED(cb);
661
662 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_CNUM, 0, 0, NULL);
663 return BT_STATUS_SUCCESS;
664 }
665
666 /*******************************************************************************
667 *
668 * Function send_dtmf
669 *
670 * Description send dtmf
671 *
672 * Returns bt_status_t
673 *
674 ******************************************************************************/
send_dtmf(const RawAddress * bd_addr,char code)675 static bt_status_t send_dtmf(const RawAddress* bd_addr, char code) {
676 btif_hf_client_cb_t* cb = btif_hf_client_get_cb_by_bda(*bd_addr);
677 if (cb == NULL || !is_connected(cb)) return BT_STATUS_FAIL;
678
679 CHECK_BTHF_CLIENT_SLC_CONNECTED(cb);
680
681 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_VTS, code, 0, NULL);
682 return BT_STATUS_SUCCESS;
683 }
684
685 /*******************************************************************************
686 *
687 * Function request_last_voice_tag_number
688 *
689 * Description Request number from AG for VR purposes
690 *
691 * Returns bt_status_t
692 *
693 ******************************************************************************/
request_last_voice_tag_number(const RawAddress * bd_addr)694 static bt_status_t request_last_voice_tag_number(const RawAddress* bd_addr) {
695 btif_hf_client_cb_t* cb = btif_hf_client_get_cb_by_bda(*bd_addr);
696 if (cb == NULL || !is_connected(cb)) return BT_STATUS_FAIL;
697
698 CHECK_BTHF_CLIENT_SLC_CONNECTED(cb);
699
700 if (cb->peer_feat & BTA_HF_CLIENT_PEER_VTAG) {
701 BTA_HfClientSendAT(cb->handle, BTA_HF_CLIENT_AT_CMD_BINP, 1, 0, NULL);
702 return BT_STATUS_SUCCESS;
703 }
704 return BT_STATUS_UNSUPPORTED;
705 }
706
707 /*******************************************************************************
708 *
709 * Function cleanup
710 *
711 * Description Closes the HF interface
712 *
713 * Returns bt_status_t
714 *
715 ******************************************************************************/
cleanup(void)716 static void cleanup(void) {
717 BTIF_TRACE_EVENT("%s", __func__);
718
719 btif_queue_cleanup(UUID_SERVCLASS_HF_HANDSFREE);
720 if (bt_hf_client_callbacks) {
721 btif_disable_service(BTA_HFP_HS_SERVICE_ID);
722 bt_hf_client_callbacks = NULL;
723 }
724 }
725
726 /*******************************************************************************
727 *
728 * Function send_at_cmd
729 *
730 * Description Send requested AT command to rempte device.
731 *
732 * Returns bt_status_t
733 *
734 ******************************************************************************/
send_at_cmd(const RawAddress * bd_addr,int cmd,int val1,int val2,const char * arg)735 static bt_status_t send_at_cmd(const RawAddress* bd_addr, int cmd, int val1,
736 int val2, const char* arg) {
737 btif_hf_client_cb_t* cb = btif_hf_client_get_cb_by_bda(*bd_addr);
738 if (cb == NULL || !is_connected(cb)) return BT_STATUS_FAIL;
739
740 CHECK_BTHF_CLIENT_SLC_CONNECTED(cb);
741
742 BTIF_TRACE_EVENT("%s: Cmd %d val1 %d val2 %d arg %s", __func__, cmd, val1,
743 val2, (arg != NULL) ? arg : "<null>");
744 BTA_HfClientSendAT(cb->handle, cmd, val1, val2, arg);
745
746 return BT_STATUS_SUCCESS;
747 }
748
749 static const bthf_client_interface_t bthfClientInterface = {
750 sizeof(bthf_client_interface_t),
751 .init = init,
752 .connect = connect,
753 .disconnect = disconnect,
754 .connect_audio = connect_audio,
755 .disconnect_audio = disconnect_audio,
756 .start_voice_recognition = start_voice_recognition,
757 .stop_voice_recognition = stop_voice_recognition,
758 .volume_control = volume_control,
759 .dial = dial,
760 .dial_memory = dial_memory,
761 .handle_call_action = handle_call_action,
762 .query_current_calls = query_current_calls,
763 .query_current_operator_name = query_current_operator_name,
764 .retrieve_subscriber_info = retrieve_subscriber_info,
765 .send_dtmf = send_dtmf,
766 .request_last_voice_tag_number = request_last_voice_tag_number,
767 .cleanup = cleanup,
768 .send_at_cmd = send_at_cmd,
769 };
770
process_ind_evt(tBTA_HF_CLIENT_IND * ind)771 static void process_ind_evt(tBTA_HF_CLIENT_IND* ind) {
772 BTIF_TRACE_DEBUG("%s", __func__);
773
774 btif_hf_client_cb_t* cb = btif_hf_client_get_cb_by_bda(ind->bd_addr);
775 if (cb == NULL || !is_connected(cb)) return;
776
777 switch (ind->type) {
778 case BTA_HF_CLIENT_IND_CALL:
779 HAL_CBACK(bt_hf_client_callbacks, call_cb, &cb->peer_bda,
780 (bthf_client_call_t)ind->value);
781 break;
782
783 case BTA_HF_CLIENT_IND_CALLSETUP:
784 HAL_CBACK(bt_hf_client_callbacks, callsetup_cb, &cb->peer_bda,
785 (bthf_client_callsetup_t)ind->value);
786 break;
787 case BTA_HF_CLIENT_IND_CALLHELD:
788 HAL_CBACK(bt_hf_client_callbacks, callheld_cb, &cb->peer_bda,
789 (bthf_client_callheld_t)ind->value);
790 break;
791
792 case BTA_HF_CLIENT_IND_SERVICE:
793 HAL_CBACK(bt_hf_client_callbacks, network_state_cb, &cb->peer_bda,
794 (bthf_client_network_state_t)ind->value);
795 break;
796
797 case BTA_HF_CLIENT_IND_SIGNAL:
798 HAL_CBACK(bt_hf_client_callbacks, network_signal_cb, &cb->peer_bda,
799 ind->value);
800 break;
801
802 case BTA_HF_CLIENT_IND_ROAM:
803 HAL_CBACK(bt_hf_client_callbacks, network_roaming_cb, &cb->peer_bda,
804 (bthf_client_service_type_t)ind->value);
805 break;
806
807 case BTA_HF_CLIENT_IND_BATTCH:
808 HAL_CBACK(bt_hf_client_callbacks, battery_level_cb, &cb->peer_bda,
809 ind->value);
810 break;
811
812 default:
813 break;
814 }
815 }
816
817 /*******************************************************************************
818 *
819 * Function btif_hf_client_upstreams_evt
820 *
821 * Description Executes HF CLIENT UPSTREAMS events in btif context
822 *
823 * Returns void
824 *
825 ******************************************************************************/
btif_hf_client_upstreams_evt(uint16_t event,char * p_param)826 static void btif_hf_client_upstreams_evt(uint16_t event, char* p_param) {
827 tBTA_HF_CLIENT* p_data = (tBTA_HF_CLIENT*)p_param;
828
829 btif_hf_client_cb_t* cb = btif_hf_client_get_cb_by_bda(p_data->bd_addr);
830 if (cb == NULL && event == BTA_HF_CLIENT_OPEN_EVT) {
831 BTIF_TRACE_DEBUG("%s: event BTA_HF_CLIENT_OPEN_EVT allocating block",
832 __func__);
833 cb = btif_hf_client_allocate_cb();
834 cb->handle = p_data->open.handle;
835 cb->peer_bda = p_data->open.bd_addr;
836 } else if (cb == NULL) {
837 BTIF_TRACE_ERROR("%s: event %d but not allocating block: cb not found",
838 __func__, event);
839 return;
840 }
841
842 BTIF_TRACE_DEBUG("%s: event=%s (%u)", __func__, dump_hf_client_event(event),
843 event);
844
845 switch (event) {
846 case BTA_HF_CLIENT_OPEN_EVT:
847 if (p_data->open.status == BTA_HF_CLIENT_SUCCESS) {
848 cb->state = BTHF_CLIENT_CONNECTION_STATE_CONNECTED;
849 cb->peer_feat = 0;
850 cb->chld_feat = 0;
851 } else if (cb->state == BTHF_CLIENT_CONNECTION_STATE_CONNECTING) {
852 cb->state = BTHF_CLIENT_CONNECTION_STATE_DISCONNECTED;
853 } else {
854 BTIF_TRACE_WARNING(
855 "%s: HF CLient open failed, but another device connected. "
856 "status=%d state=%d connected device=%s",
857 __func__, p_data->open.status, cb->state,
858 cb->peer_bda.ToString().c_str());
859 break;
860 }
861
862 HAL_CBACK(bt_hf_client_callbacks, connection_state_cb, &cb->peer_bda,
863 cb->state, 0, /* peer feat */
864 0 /* AT+CHLD feat */);
865
866 if (cb->state == BTHF_CLIENT_CONNECTION_STATE_DISCONNECTED)
867 cb->peer_bda = RawAddress::kAny;
868
869 if (p_data->open.status != BTA_HF_CLIENT_SUCCESS) btif_queue_advance();
870 break;
871
872 case BTA_HF_CLIENT_CONN_EVT:
873 cb->peer_feat = p_data->conn.peer_feat;
874 cb->chld_feat = p_data->conn.chld_feat;
875 cb->state = BTHF_CLIENT_CONNECTION_STATE_SLC_CONNECTED;
876
877 HAL_CBACK(bt_hf_client_callbacks, connection_state_cb, &cb->peer_bda,
878 cb->state, cb->peer_feat, cb->chld_feat);
879
880 /* Inform the application about in-band ringtone */
881 if (cb->peer_feat & BTA_HF_CLIENT_PEER_INBAND) {
882 HAL_CBACK(bt_hf_client_callbacks, in_band_ring_tone_cb, &cb->peer_bda,
883 BTHF_CLIENT_IN_BAND_RINGTONE_PROVIDED);
884 }
885
886 btif_queue_advance();
887 break;
888
889 case BTA_HF_CLIENT_CLOSE_EVT:
890 cb->state = BTHF_CLIENT_CONNECTION_STATE_DISCONNECTED;
891 HAL_CBACK(bt_hf_client_callbacks, connection_state_cb, &cb->peer_bda,
892 cb->state, 0, 0);
893 cb->peer_bda = RawAddress::kAny;
894 cb->peer_feat = 0;
895 cb->chld_feat = 0;
896 btif_queue_advance();
897 break;
898
899 case BTA_HF_CLIENT_IND_EVT:
900 process_ind_evt(&p_data->ind);
901 break;
902
903 case BTA_HF_CLIENT_MIC_EVT:
904 HAL_CBACK(bt_hf_client_callbacks, volume_change_cb, &cb->peer_bda,
905 BTHF_CLIENT_VOLUME_TYPE_MIC, p_data->val.value);
906 break;
907
908 case BTA_HF_CLIENT_SPK_EVT:
909 HAL_CBACK(bt_hf_client_callbacks, volume_change_cb, &cb->peer_bda,
910 BTHF_CLIENT_VOLUME_TYPE_SPK, p_data->val.value);
911 break;
912
913 case BTA_HF_CLIENT_VOICE_REC_EVT:
914 HAL_CBACK(bt_hf_client_callbacks, vr_cmd_cb, &cb->peer_bda,
915 (bthf_client_vr_state_t)p_data->val.value);
916 break;
917
918 case BTA_HF_CLIENT_OPERATOR_NAME_EVT:
919 HAL_CBACK(bt_hf_client_callbacks, current_operator_cb, &cb->peer_bda,
920 p_data->operator_name.name);
921 break;
922
923 case BTA_HF_CLIENT_CLIP_EVT:
924 HAL_CBACK(bt_hf_client_callbacks, clip_cb, &cb->peer_bda,
925 p_data->number.number);
926 break;
927
928 case BTA_HF_CLIENT_BINP_EVT:
929 HAL_CBACK(bt_hf_client_callbacks, last_voice_tag_number_callback,
930 &cb->peer_bda, p_data->number.number);
931 break;
932
933 case BTA_HF_CLIENT_CCWA_EVT:
934 HAL_CBACK(bt_hf_client_callbacks, call_waiting_cb, &cb->peer_bda,
935 p_data->number.number);
936 break;
937
938 case BTA_HF_CLIENT_AT_RESULT_EVT:
939 HAL_CBACK(bt_hf_client_callbacks, cmd_complete_cb, &cb->peer_bda,
940 (bthf_client_cmd_complete_t)p_data->result.type,
941 p_data->result.cme);
942 break;
943
944 case BTA_HF_CLIENT_CLCC_EVT:
945 HAL_CBACK(bt_hf_client_callbacks, current_calls_cb, &cb->peer_bda,
946 p_data->clcc.idx,
947 p_data->clcc.inc ? BTHF_CLIENT_CALL_DIRECTION_INCOMING
948 : BTHF_CLIENT_CALL_DIRECTION_OUTGOING,
949 (bthf_client_call_state_t)p_data->clcc.status,
950 p_data->clcc.mpty ? BTHF_CLIENT_CALL_MPTY_TYPE_MULTI
951 : BTHF_CLIENT_CALL_MPTY_TYPE_SINGLE,
952 p_data->clcc.number_present ? p_data->clcc.number : NULL);
953 break;
954
955 case BTA_HF_CLIENT_CNUM_EVT:
956 if (p_data->cnum.service == 4) {
957 HAL_CBACK(bt_hf_client_callbacks, subscriber_info_cb, &cb->peer_bda,
958 p_data->cnum.number, BTHF_CLIENT_SERVICE_VOICE);
959 } else if (p_data->cnum.service == 5) {
960 HAL_CBACK(bt_hf_client_callbacks, subscriber_info_cb, &cb->peer_bda,
961 p_data->cnum.number, BTHF_CLIENT_SERVICE_FAX);
962 } else {
963 HAL_CBACK(bt_hf_client_callbacks, subscriber_info_cb, &cb->peer_bda,
964 p_data->cnum.number, BTHF_CLIENT_SERVICE_UNKNOWN);
965 }
966 break;
967
968 case BTA_HF_CLIENT_BTRH_EVT:
969 if (p_data->val.value <= BTRH_CLIENT_RESP_AND_HOLD_REJECT) {
970 HAL_CBACK(bt_hf_client_callbacks, resp_and_hold_cb, &cb->peer_bda,
971 (bthf_client_resp_and_hold_t)p_data->val.value);
972 }
973 break;
974
975 case BTA_HF_CLIENT_BSIR_EVT:
976 if (p_data->val.value != 0) {
977 HAL_CBACK(bt_hf_client_callbacks, in_band_ring_tone_cb, &cb->peer_bda,
978 BTHF_CLIENT_IN_BAND_RINGTONE_PROVIDED);
979 } else {
980 HAL_CBACK(bt_hf_client_callbacks, in_band_ring_tone_cb, &cb->peer_bda,
981 BTHF_CLIENT_IN_BAND_RINGTONE_NOT_PROVIDED);
982 }
983 break;
984
985 case BTA_HF_CLIENT_AUDIO_OPEN_EVT:
986 HAL_CBACK(bt_hf_client_callbacks, audio_state_cb, &cb->peer_bda,
987 BTHF_CLIENT_AUDIO_STATE_CONNECTED);
988 break;
989
990 case BTA_HF_CLIENT_AUDIO_MSBC_OPEN_EVT:
991 HAL_CBACK(bt_hf_client_callbacks, audio_state_cb, &cb->peer_bda,
992 BTHF_CLIENT_AUDIO_STATE_CONNECTED_MSBC);
993 break;
994
995 case BTA_HF_CLIENT_AUDIO_CLOSE_EVT:
996 HAL_CBACK(bt_hf_client_callbacks, audio_state_cb, &cb->peer_bda,
997 BTHF_CLIENT_AUDIO_STATE_DISCONNECTED);
998 break;
999 case BTA_HF_CLIENT_RING_INDICATION:
1000 HAL_CBACK(bt_hf_client_callbacks, ring_indication_cb, &cb->peer_bda);
1001 break;
1002 default:
1003 BTIF_TRACE_WARNING("%s: Unhandled event: %d", __func__, event);
1004 break;
1005 }
1006 }
1007
1008 /*******************************************************************************
1009 *
1010 * Function bta_hf_client_evt
1011 *
1012 * Description Switches context from BTA to BTIF for all HF Client events
1013 *
1014 * Returns void
1015 *
1016 ******************************************************************************/
1017
bta_hf_client_evt(tBTA_HF_CLIENT_EVT event,tBTA_HF_CLIENT * p_data)1018 static void bta_hf_client_evt(tBTA_HF_CLIENT_EVT event,
1019 tBTA_HF_CLIENT* p_data) {
1020 bt_status_t status;
1021
1022 /* switch context to btif task context (copy full union size for convenience)
1023 */
1024 status = btif_transfer_context(btif_hf_client_upstreams_evt, (uint16_t)event,
1025 (char*)p_data, sizeof(*p_data), NULL);
1026
1027 /* catch any failed context transfers */
1028 ASSERTC(status == BT_STATUS_SUCCESS, "context transfer failed", status);
1029 }
1030
1031 /*******************************************************************************
1032 *
1033 * Function btif_hf_client_execute_service
1034 *
1035 * Description Initializes/Shuts down the service
1036 *
1037 * Returns BT_STATUS_SUCCESS on success, BT_STATUS_FAIL otherwise
1038 *
1039 ******************************************************************************/
btif_hf_client_execute_service(bool b_enable)1040 bt_status_t btif_hf_client_execute_service(bool b_enable) {
1041 BTIF_TRACE_EVENT("%s: enable: %d", __func__, b_enable);
1042
1043 if (b_enable) {
1044 /* Enable and register with BTA-HFClient */
1045 BTIF_TRACE_EVENT("%s: support codec negotiation %d ", __func__,
1046 BTIF_HF_CLIENT_FEATURES);
1047 BTA_HfClientEnable(bta_hf_client_evt, BTIF_HF_CLIENT_SECURITY,
1048 BTIF_HF_CLIENT_FEATURES, BTIF_HF_CLIENT_SERVICE_NAME);
1049 } else {
1050 BTA_HfClientDisable();
1051 }
1052 return BT_STATUS_SUCCESS;
1053 }
1054
1055 /*******************************************************************************
1056 *
1057 * Function btif_hf_get_interface
1058 *
1059 * Description Get the hf callback interface
1060 *
1061 * Returns bthf_interface_t
1062 *
1063 ******************************************************************************/
btif_hf_client_get_interface(void)1064 const bthf_client_interface_t* btif_hf_client_get_interface(void) {
1065 BTIF_TRACE_EVENT("%s", __func__);
1066 return &bthfClientInterface;
1067 }
1068