1 /******************************************************************************
2 *
3 * Copyright (c) 2014 The Android Open Source Project
4 * Copyright (C) 2003-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 #define LOG_TAG "bt_hf_client"
21
22 #include <errno.h>
23 #include <stdio.h>
24 #include <string.h>
25
26 #include "bta_hf_client_api.h"
27 #include "bta_hf_client_int.h"
28 #include "osi/include/log.h"
29 #include "osi/include/osi.h"
30 #include "port_api.h"
31
32 /* Uncomment to enable AT traffic dumping */
33 /* #define BTA_HF_CLIENT_AT_DUMP 1 */
34
35 /* minimum length of AT event */
36 #define BTA_HF_CLIENT_AT_EVENT_MIN_LEN 3
37
38 /* timeout (in milliseconds) for AT response */
39 #define BTA_HF_CLIENT_AT_TIMEOUT 29989
40
41 /* timeout (in milliseconds) for AT hold timer */
42 #define BTA_HF_CLIENT_AT_HOLD_TIMEOUT 41
43
44 /******************************************************************************
45 * SUPPORTED EVENT MESSAGES
46 ******************************************************************************/
47
48 /* CIND: supported indicator names */
49 #define BTA_HF_CLIENT_INDICATOR_BATTERYCHG "battchg"
50 #define BTA_HF_CLIENT_INDICATOR_SIGNAL "signal"
51 #define BTA_HF_CLIENT_INDICATOR_SERVICE "service"
52 #define BTA_HF_CLIENT_INDICATOR_CALL "call"
53 #define BTA_HF_CLIENT_INDICATOR_ROAM "roam"
54 #define BTA_HF_CLIENT_INDICATOR_CALLSETUP "callsetup"
55 #define BTA_HF_CLIENT_INDICATOR_CALLHELD "callheld"
56
57 #define MIN(a, b) \
58 ({ \
59 __typeof__(a) _a = (a); \
60 __typeof__(b) _b = (b); \
61 (_a < _b) ? _a : _b; \
62 })
63
64 /* CIND: represents each indicators boundaries */
65 typedef struct {
66 const char* name;
67 uint8_t min;
68 uint8_t max;
69 uint8_t namelen;
70 } tBTA_HF_CLIENT_INDICATOR;
71
72 #define BTA_HF_CLIENT_AT_SUPPORTED_INDICATOR_COUNT 7
73
74 /* CIND: storage room for indicators value range and their statuses */
75 static const tBTA_HF_CLIENT_INDICATOR
76 bta_hf_client_indicators[BTA_HF_CLIENT_AT_SUPPORTED_INDICATOR_COUNT] = {
77 /* name | min | max | name length -
78 used by parser */
79 {BTA_HF_CLIENT_INDICATOR_BATTERYCHG, 0, 5,
80 sizeof(BTA_HF_CLIENT_INDICATOR_BATTERYCHG)},
81 {BTA_HF_CLIENT_INDICATOR_SIGNAL, 0, 5,
82 sizeof(BTA_HF_CLIENT_INDICATOR_SIGNAL)},
83 {BTA_HF_CLIENT_INDICATOR_SERVICE, 0, 1,
84 sizeof(BTA_HF_CLIENT_INDICATOR_SERVICE)},
85 {BTA_HF_CLIENT_INDICATOR_CALL, 0, 1,
86 sizeof(BTA_HF_CLIENT_INDICATOR_CALL)},
87 {BTA_HF_CLIENT_INDICATOR_ROAM, 0, 1,
88 sizeof(BTA_HF_CLIENT_INDICATOR_ROAM)},
89 {BTA_HF_CLIENT_INDICATOR_CALLSETUP, 0, 3,
90 sizeof(BTA_HF_CLIENT_INDICATOR_CALLSETUP)},
91 {BTA_HF_CLIENT_INDICATOR_CALLHELD, 0, 2,
92 sizeof(BTA_HF_CLIENT_INDICATOR_CALLHELD)}};
93
94 /* +VGM/+VGS - gain min/max values */
95 #define BTA_HF_CLIENT_VGS_MIN 0
96 #define BTA_HF_CLIENT_VGS_MAX 15
97 #define BTA_HF_CLIENT_VGM_MIN 0
98 #define BTA_HF_CLIENT_VGM_MAX 15
99
100 uint32_t service_index = 0;
101 bool service_availability = true;
102 /* helper functions for handling AT commands queueing */
103
104 static void bta_hf_client_handle_ok(tBTA_HF_CLIENT_CB* client_cb);
105
bta_hf_client_clear_queued_at(tBTA_HF_CLIENT_CB * client_cb)106 static void bta_hf_client_clear_queued_at(tBTA_HF_CLIENT_CB* client_cb) {
107 tBTA_HF_CLIENT_AT_QCMD* cur = client_cb->at_cb.queued_cmd;
108 tBTA_HF_CLIENT_AT_QCMD* next;
109
110 while (cur != NULL) {
111 next = cur->next;
112 osi_free(cur);
113 cur = next;
114 }
115
116 client_cb->at_cb.queued_cmd = NULL;
117 }
118
bta_hf_client_queue_at(tBTA_HF_CLIENT_CB * client_cb,tBTA_HF_CLIENT_AT_CMD cmd,const char * buf,uint16_t buf_len)119 static void bta_hf_client_queue_at(tBTA_HF_CLIENT_CB* client_cb,
120 tBTA_HF_CLIENT_AT_CMD cmd, const char* buf,
121 uint16_t buf_len) {
122 tBTA_HF_CLIENT_AT_QCMD* new_cmd =
123 (tBTA_HF_CLIENT_AT_QCMD*)osi_malloc(sizeof(tBTA_HF_CLIENT_AT_QCMD));
124
125 APPL_TRACE_DEBUG("%s", __func__);
126
127 new_cmd->cmd = cmd;
128 new_cmd->buf_len = buf_len;
129 new_cmd->next = NULL;
130 memcpy(new_cmd->buf, buf, buf_len);
131
132 if (client_cb->at_cb.queued_cmd != NULL) {
133 tBTA_HF_CLIENT_AT_QCMD* qcmd = client_cb->at_cb.queued_cmd;
134
135 while (qcmd->next != NULL) qcmd = qcmd->next;
136
137 qcmd->next = new_cmd;
138 } else {
139 client_cb->at_cb.queued_cmd = new_cmd;
140 }
141 }
142
bta_hf_client_at_resp_timer_cback(void * data)143 static void bta_hf_client_at_resp_timer_cback(void* data) {
144 tBTA_HF_CLIENT_CB* client_cb = (tBTA_HF_CLIENT_CB*)data;
145 if (client_cb->at_cb.current_cmd == BTA_HF_CLIENT_AT_CNUM) {
146 LOG_INFO(LOG_TAG,
147 "%s: timed out waiting for AT+CNUM response; spoofing OK.",
148 __func__);
149 bta_hf_client_handle_ok(client_cb);
150 } else {
151 APPL_TRACE_ERROR("HFPClient: AT response timeout, disconnecting");
152
153 tBTA_HF_CLIENT_DATA msg;
154 msg.hdr.layer_specific = client_cb->handle;
155 bta_hf_client_sm_execute(BTA_HF_CLIENT_API_CLOSE_EVT, &msg);
156 }
157 }
158
bta_hf_client_start_at_resp_timer(tBTA_HF_CLIENT_CB * client_cb)159 static void bta_hf_client_start_at_resp_timer(tBTA_HF_CLIENT_CB* client_cb) {
160 alarm_set_on_mloop(client_cb->at_cb.resp_timer, BTA_HF_CLIENT_AT_TIMEOUT,
161 bta_hf_client_at_resp_timer_cback, (void*)client_cb);
162 }
163
bta_hf_client_stop_at_resp_timer(tBTA_HF_CLIENT_CB * client_cb)164 static void bta_hf_client_stop_at_resp_timer(tBTA_HF_CLIENT_CB* client_cb) {
165 alarm_cancel(client_cb->at_cb.resp_timer);
166 }
167
bta_hf_client_send_at(tBTA_HF_CLIENT_CB * client_cb,tBTA_HF_CLIENT_AT_CMD cmd,const char * buf,uint16_t buf_len)168 static void bta_hf_client_send_at(tBTA_HF_CLIENT_CB* client_cb,
169 tBTA_HF_CLIENT_AT_CMD cmd, const char* buf,
170 uint16_t buf_len) {
171 APPL_TRACE_DEBUG("%s", __func__);
172 if ((client_cb->at_cb.current_cmd == BTA_HF_CLIENT_AT_NONE ||
173 client_cb->svc_conn == false) &&
174 !alarm_is_scheduled(client_cb->at_cb.hold_timer)) {
175 uint16_t len;
176
177 #ifdef BTA_HF_CLIENT_AT_DUMP
178 APPL_TRACE_DEBUG("%s: %.*s", __func__, buf_len - 1, buf);
179 #endif
180
181 client_cb->at_cb.current_cmd = cmd;
182 /* Generate fake responses for these because they won't reliably work */
183 if (!service_availability &&
184 (cmd == BTA_HF_CLIENT_AT_CNUM || cmd == BTA_HF_CLIENT_AT_COPS)) {
185 APPL_TRACE_WARNING("%s: No service, skipping %d command", __func__, cmd);
186 bta_hf_client_handle_ok(client_cb);
187 return;
188 }
189
190 APPL_TRACE_DEBUG("%s: writing port data to %d", __func__,
191 client_cb->conn_handle);
192 PORT_WriteData(client_cb->conn_handle, buf, buf_len, &len);
193
194 bta_hf_client_start_at_resp_timer(client_cb);
195
196 return;
197 }
198
199 bta_hf_client_queue_at(client_cb, cmd, buf, buf_len);
200 }
201
bta_hf_client_send_queued_at(tBTA_HF_CLIENT_CB * client_cb)202 static void bta_hf_client_send_queued_at(tBTA_HF_CLIENT_CB* client_cb) {
203 tBTA_HF_CLIENT_AT_QCMD* cur = client_cb->at_cb.queued_cmd;
204
205 APPL_TRACE_DEBUG("%s", __func__);
206
207 if (cur != NULL) {
208 client_cb->at_cb.queued_cmd = cur->next;
209
210 bta_hf_client_send_at(client_cb, cur->cmd, cur->buf, cur->buf_len);
211
212 osi_free(cur);
213 }
214 }
215
bta_hf_client_at_hold_timer_cback(void * data)216 static void bta_hf_client_at_hold_timer_cback(void* data) {
217 tBTA_HF_CLIENT_CB* client_cb = (tBTA_HF_CLIENT_CB*)data;
218 APPL_TRACE_DEBUG("%s", __func__);
219 bta_hf_client_send_queued_at(client_cb);
220 }
221
bta_hf_client_stop_at_hold_timer(tBTA_HF_CLIENT_CB * client_cb)222 static void bta_hf_client_stop_at_hold_timer(tBTA_HF_CLIENT_CB* client_cb) {
223 APPL_TRACE_DEBUG("%s", __func__);
224 alarm_cancel(client_cb->at_cb.hold_timer);
225 }
226
bta_hf_client_start_at_hold_timer(tBTA_HF_CLIENT_CB * client_cb)227 static void bta_hf_client_start_at_hold_timer(tBTA_HF_CLIENT_CB* client_cb) {
228 APPL_TRACE_DEBUG("%s", __func__);
229 alarm_set_on_mloop(client_cb->at_cb.hold_timer, BTA_HF_CLIENT_AT_HOLD_TIMEOUT,
230 bta_hf_client_at_hold_timer_cback, (void*)client_cb);
231 }
232
233 /******************************************************************************
234 *
235 * COMMON AT EVENT HANDLING funcS
236 *
237 * Receives data (strings, ints, etc.) from the parser and processes this
238 * data. No buffer parsing is being done here.
239 ******************************************************************************/
240
bta_hf_client_handle_ok(tBTA_HF_CLIENT_CB * client_cb)241 static void bta_hf_client_handle_ok(tBTA_HF_CLIENT_CB* client_cb) {
242 APPL_TRACE_DEBUG("%s", __func__);
243
244 bta_hf_client_stop_at_resp_timer(client_cb);
245
246 if (!client_cb->svc_conn) {
247 bta_hf_client_slc_seq(client_cb, false);
248 return;
249 }
250
251 switch (client_cb->at_cb.current_cmd) {
252 case BTA_HF_CLIENT_AT_BIA:
253 case BTA_HF_CLIENT_AT_BCC:
254 break;
255 case BTA_HF_CLIENT_AT_BCS:
256 bta_hf_client_start_at_hold_timer(client_cb);
257 client_cb->at_cb.current_cmd = BTA_HF_CLIENT_AT_NONE;
258 return;
259 case BTA_HF_CLIENT_AT_CLIP: // last cmd is post slc seq
260 if (client_cb->send_at_reply == false) {
261 client_cb->send_at_reply = true;
262 }
263 break;
264 case BTA_HF_CLIENT_AT_NONE:
265 bta_hf_client_stop_at_hold_timer(client_cb);
266 break;
267 default:
268 if (client_cb->send_at_reply) {
269 bta_hf_client_at_result(client_cb, BTA_HF_CLIENT_AT_RESULT_OK, 0);
270 }
271 break;
272 }
273
274 client_cb->at_cb.current_cmd = BTA_HF_CLIENT_AT_NONE;
275
276 bta_hf_client_send_queued_at(client_cb);
277 }
278
bta_hf_client_handle_error(tBTA_HF_CLIENT_CB * client_cb,tBTA_HF_CLIENT_AT_RESULT_TYPE type,uint16_t cme)279 static void bta_hf_client_handle_error(tBTA_HF_CLIENT_CB* client_cb,
280 tBTA_HF_CLIENT_AT_RESULT_TYPE type,
281 uint16_t cme) {
282 APPL_TRACE_DEBUG("%s: %u %u", __func__, type, cme);
283
284 bta_hf_client_stop_at_resp_timer(client_cb);
285
286 if (!client_cb->svc_conn) {
287 bta_hf_client_slc_seq(client_cb, true);
288 return;
289 }
290
291 switch (client_cb->at_cb.current_cmd) {
292 case BTA_HF_CLIENT_AT_BIA:
293 break;
294 case BTA_HF_CLIENT_AT_BCC:
295 case BTA_HF_CLIENT_AT_BCS:
296 bta_hf_client_cback_sco(client_cb, BTA_HF_CLIENT_AUDIO_CLOSE_EVT);
297 break;
298 case BTA_HF_CLIENT_AT_CLIP: // last cmd is post slc seq
299 if (client_cb->send_at_reply == false) {
300 client_cb->send_at_reply = true;
301 }
302 break;
303 default:
304 if (client_cb->send_at_reply) {
305 bta_hf_client_at_result(client_cb, type, cme);
306 }
307 break;
308 }
309
310 client_cb->at_cb.current_cmd = BTA_HF_CLIENT_AT_NONE;
311
312 bta_hf_client_send_queued_at(client_cb);
313 }
314
bta_hf_client_handle_ring(tBTA_HF_CLIENT_CB * client_cb)315 static void bta_hf_client_handle_ring(tBTA_HF_CLIENT_CB* client_cb) {
316 APPL_TRACE_DEBUG("%s", __func__);
317 bta_hf_client_evt_val(client_cb, BTA_HF_CLIENT_RING_INDICATION, 0);
318 }
319
bta_hf_client_handle_brsf(tBTA_HF_CLIENT_CB * client_cb,uint32_t value)320 static void bta_hf_client_handle_brsf(tBTA_HF_CLIENT_CB* client_cb,
321 uint32_t value) {
322 APPL_TRACE_DEBUG("%s: 0x%x", __func__, value);
323 client_cb->peer_features = value;
324 }
325
326 /* handles a single indicator descriptor - registers it for value changing
327 * events */
bta_hf_client_handle_cind_list_item(tBTA_HF_CLIENT_CB * client_cb,char * name,uint32_t min,uint32_t max,uint32_t index)328 static void bta_hf_client_handle_cind_list_item(tBTA_HF_CLIENT_CB* client_cb,
329 char* name, uint32_t min,
330 uint32_t max, uint32_t index) {
331 uint8_t i = 0;
332
333 APPL_TRACE_DEBUG("%s: %lu.%s <%lu:%lu>", __func__, index, name, min, max);
334
335 /* look for a matching indicator on list of supported ones */
336 for (i = 0; i < BTA_HF_CLIENT_AT_SUPPORTED_INDICATOR_COUNT; i++) {
337 if (strcmp(name, BTA_HF_CLIENT_INDICATOR_SERVICE) == 0) {
338 service_index = index;
339 }
340 /* look for a match - search one sign further than indicators name to check
341 * for string end */
342 /* It will distinguish 'callheld' which could be matched by strncmp as
343 * 'call'. */
344 if (strncmp(name, bta_hf_client_indicators[i].name,
345 bta_hf_client_indicators[i].namelen) != 0)
346 continue;
347
348 /* index - enumerates value position in the incoming sequence */
349 /* if name matches one of the known indicators, add its incoming position */
350 /* to lookup table for easy value->indicator matching later, when only
351 * values come */
352 client_cb->at_cb.indicator_lookup[index] = i;
353
354 return;
355 }
356 }
357
bta_hf_client_handle_cind_value(tBTA_HF_CLIENT_CB * client_cb,uint32_t index,uint32_t value)358 static void bta_hf_client_handle_cind_value(tBTA_HF_CLIENT_CB* client_cb,
359 uint32_t index, uint32_t value) {
360 APPL_TRACE_DEBUG("%s: index: %u value: %u", __func__, index, value);
361
362 if (index >= BTA_HF_CLIENT_AT_INDICATOR_COUNT) {
363 return;
364 }
365
366 if (service_index == index) {
367 if (value == 0) {
368 service_availability = false;
369 } else {
370 service_availability = true;
371 }
372 }
373 if (client_cb->at_cb.indicator_lookup[index] == -1) {
374 return;
375 }
376
377 /* get the real array index from lookup table */
378 index = client_cb->at_cb.indicator_lookup[index];
379
380 /* Ignore out of range values */
381 if (value > bta_hf_client_indicators[index].max ||
382 value < bta_hf_client_indicators[index].min) {
383 return;
384 }
385
386 /* tBTA_HF_CLIENT_IND_TYPE match index in bta_hf_client_indicators */
387 bta_hf_client_ind(client_cb, index, value);
388 }
389
bta_hf_client_handle_chld(tBTA_HF_CLIENT_CB * client_cb,uint32_t mask)390 static void bta_hf_client_handle_chld(tBTA_HF_CLIENT_CB* client_cb,
391 uint32_t mask) {
392 APPL_TRACE_DEBUG("%s: 0x%x", __func__, mask);
393
394 client_cb->chld_features |= mask;
395 }
396
bta_hf_client_handle_ciev(tBTA_HF_CLIENT_CB * client_cb,uint32_t index,uint32_t value)397 static void bta_hf_client_handle_ciev(tBTA_HF_CLIENT_CB* client_cb,
398 uint32_t index, uint32_t value) {
399 int8_t realind = -1;
400
401 APPL_TRACE_DEBUG("%s: index: %u value: %u", __func__, index, value);
402
403 if (index == 0 || index > BTA_HF_CLIENT_AT_INDICATOR_COUNT) {
404 return;
405 }
406
407 if (service_index == index - 1) {
408 service_availability = value == 0 ? false : true;
409 }
410
411 realind = client_cb->at_cb.indicator_lookup[index - 1];
412
413 if (realind >= 0 && realind < BTA_HF_CLIENT_AT_SUPPORTED_INDICATOR_COUNT) {
414 /* get the real in-array index from lookup table by index it comes at */
415 /* if there is no bug it should automatically be correctly calculated */
416 if (value > bta_hf_client_indicators[realind].max ||
417 value < bta_hf_client_indicators[realind].min) {
418 return;
419 }
420
421 /* update service availability on +ciev from AG. */
422 if (service_index == (index - 1)) {
423 if (value == 1) {
424 service_availability = true;
425 } else {
426 service_availability = false;
427 }
428 }
429
430 /* tBTA_HF_CLIENT_IND_TYPE match index in bta_hf_client_indicators */
431 bta_hf_client_ind(client_cb, realind, value);
432 }
433 }
434
bta_hf_client_handle_bcs(tBTA_HF_CLIENT_CB * client_cb,uint32_t codec)435 static void bta_hf_client_handle_bcs(tBTA_HF_CLIENT_CB* client_cb,
436 uint32_t codec) {
437 APPL_TRACE_DEBUG("%s: codec: %u sco listen state: %d", __func__, codec,
438 client_cb->sco_state);
439 if (codec == BTM_SCO_CODEC_CVSD || codec == BTM_SCO_CODEC_MSBC) {
440 client_cb->negotiated_codec = codec;
441 bta_hf_client_send_at_bcs(client_cb, codec);
442 } else {
443 client_cb->negotiated_codec = BTM_SCO_CODEC_CVSD;
444 bta_hf_client_send_at_bac(client_cb);
445 }
446 }
447
bta_hf_client_handle_bsir(tBTA_HF_CLIENT_CB * client_cb,uint32_t provided)448 static void bta_hf_client_handle_bsir(tBTA_HF_CLIENT_CB* client_cb,
449 uint32_t provided) {
450 APPL_TRACE_DEBUG("%s: %u", __func__, provided);
451
452 bta_hf_client_evt_val(client_cb, BTA_HF_CLIENT_BSIR_EVT, provided);
453 }
454
bta_hf_client_handle_cmeerror(tBTA_HF_CLIENT_CB * client_cb,uint32_t code)455 static void bta_hf_client_handle_cmeerror(tBTA_HF_CLIENT_CB* client_cb,
456 uint32_t code) {
457 bta_hf_client_handle_error(client_cb, BTA_HF_CLIENT_AT_RESULT_CME, code);
458 }
459
bta_hf_client_handle_vgm(tBTA_HF_CLIENT_CB * client_cb,uint32_t value)460 static void bta_hf_client_handle_vgm(tBTA_HF_CLIENT_CB* client_cb,
461 uint32_t value) {
462 APPL_TRACE_DEBUG("%s: %lu", __func__, value);
463
464 if (value <= BTA_HF_CLIENT_VGM_MAX) {
465 bta_hf_client_evt_val(client_cb, BTA_HF_CLIENT_MIC_EVT, value);
466 }
467 }
468
bta_hf_client_handle_vgs(tBTA_HF_CLIENT_CB * client_cb,uint32_t value)469 static void bta_hf_client_handle_vgs(tBTA_HF_CLIENT_CB* client_cb,
470 uint32_t value) {
471 APPL_TRACE_DEBUG("%s: %lu", __func__, value);
472
473 if (value <= BTA_HF_CLIENT_VGS_MAX) {
474 bta_hf_client_evt_val(client_cb, BTA_HF_CLIENT_SPK_EVT, value);
475 }
476 }
477
bta_hf_client_handle_bvra(tBTA_HF_CLIENT_CB * client_cb,uint32_t value)478 static void bta_hf_client_handle_bvra(tBTA_HF_CLIENT_CB* client_cb,
479 uint32_t value) {
480 APPL_TRACE_DEBUG("%s: %lu", __func__, value);
481
482 if (value > 1) {
483 return;
484 }
485
486 bta_hf_client_evt_val(client_cb, BTA_HF_CLIENT_VOICE_REC_EVT, value);
487 }
488
bta_hf_client_handle_clip(tBTA_HF_CLIENT_CB * client_cb,char * numstr,uint32_t type)489 static void bta_hf_client_handle_clip(tBTA_HF_CLIENT_CB* client_cb,
490 char* numstr, uint32_t type) {
491 APPL_TRACE_DEBUG("%s: %u %s", __func__, type, numstr);
492
493 bta_hf_client_clip(client_cb, numstr);
494 }
495
bta_hf_client_handle_ccwa(tBTA_HF_CLIENT_CB * client_cb,char * numstr,uint32_t type)496 static void bta_hf_client_handle_ccwa(tBTA_HF_CLIENT_CB* client_cb,
497 char* numstr, uint32_t type) {
498 APPL_TRACE_DEBUG("%s: %u %s", __func__, type, numstr);
499
500 bta_hf_client_ccwa(client_cb, numstr);
501 }
502
bta_hf_client_handle_cops(tBTA_HF_CLIENT_CB * client_cb,char * opstr,uint32_t mode)503 static void bta_hf_client_handle_cops(tBTA_HF_CLIENT_CB* client_cb, char* opstr,
504 uint32_t mode) {
505 APPL_TRACE_DEBUG("%s: %u %s", __func__, mode, opstr);
506
507 bta_hf_client_operator_name(client_cb, opstr);
508 }
509
bta_hf_client_handle_binp(tBTA_HF_CLIENT_CB * client_cb,char * numstr)510 static void bta_hf_client_handle_binp(tBTA_HF_CLIENT_CB* client_cb,
511 char* numstr) {
512 APPL_TRACE_DEBUG("%s: %s", __func__, numstr);
513
514 bta_hf_client_binp(client_cb, numstr);
515 }
516
bta_hf_client_handle_clcc(tBTA_HF_CLIENT_CB * client_cb,uint16_t idx,uint16_t dir,uint16_t status,uint16_t mode,uint16_t mpty,char * numstr,uint16_t type)517 static void bta_hf_client_handle_clcc(tBTA_HF_CLIENT_CB* client_cb,
518 uint16_t idx, uint16_t dir,
519 uint16_t status, uint16_t mode,
520 uint16_t mpty, char* numstr,
521 uint16_t type) {
522 APPL_TRACE_DEBUG("%s: idx: %u dir: %u status: %u mode: %u mpty: %u", __func__,
523 idx, dir, status, mode, mpty);
524
525 if (numstr) {
526 APPL_TRACE_DEBUG("%s: number: %s type: %u", __func__, numstr, type);
527 }
528
529 bta_hf_client_clcc(client_cb, idx, dir, status, mpty, numstr);
530 }
531
bta_hf_client_handle_cnum(tBTA_HF_CLIENT_CB * client_cb,char * numstr,uint16_t type,uint16_t service)532 static void bta_hf_client_handle_cnum(tBTA_HF_CLIENT_CB* client_cb,
533 char* numstr, uint16_t type,
534 uint16_t service) {
535 APPL_TRACE_DEBUG("%s: number: %s type: %u service: %u", __func__, numstr,
536 type, service);
537
538 /* TODO: should number be modified according to type? */
539 bta_hf_client_cnum(client_cb, numstr, service);
540 }
541
bta_hf_client_handle_btrh(tBTA_HF_CLIENT_CB * client_cb,uint16_t code)542 static void bta_hf_client_handle_btrh(tBTA_HF_CLIENT_CB* client_cb,
543 uint16_t code) {
544 APPL_TRACE_DEBUG("%s: %lu", __func__, code);
545
546 bta_hf_client_evt_val(client_cb, BTA_HF_CLIENT_BTRH_EVT, code);
547 }
548
549 /*******************************************************************************
550 *
551 * Function bta_hf_client_cback_ind
552 *
553 * Description Send indicator callback event to application.
554 *
555 * Returns void
556 *
557 ******************************************************************************/
bta_hf_client_ind(tBTA_HF_CLIENT_CB * client_cb,tBTA_HF_CLIENT_IND_TYPE type,uint16_t value)558 void bta_hf_client_ind(tBTA_HF_CLIENT_CB* client_cb,
559 tBTA_HF_CLIENT_IND_TYPE type, uint16_t value) {
560 tBTA_HF_CLIENT evt;
561
562 memset(&evt, 0, sizeof(evt));
563
564 evt.ind.type = type;
565 evt.ind.value = value;
566
567 evt.ind.bd_addr = client_cb->peer_addr;
568 bta_hf_client_app_callback(BTA_HF_CLIENT_IND_EVT, &evt);
569 }
570
571 /*******************************************************************************
572 *
573 * Function bta_hf_client_evt_val
574 *
575 * Description Send event to application.
576 * This is a generic helper for events with common data.
577 *
578 *
579 * Returns void
580 *
581 ******************************************************************************/
bta_hf_client_evt_val(tBTA_HF_CLIENT_CB * client_cb,tBTA_HF_CLIENT_EVT type,uint16_t value)582 void bta_hf_client_evt_val(tBTA_HF_CLIENT_CB* client_cb,
583 tBTA_HF_CLIENT_EVT type, uint16_t value) {
584 tBTA_HF_CLIENT evt;
585
586 memset(&evt, 0, sizeof(evt));
587
588 evt.val.bd_addr = client_cb->peer_addr;
589 evt.val.value = value;
590
591 bta_hf_client_app_callback(type, &evt);
592 }
593
594 /*******************************************************************************
595 *
596 * Function bta_hf_client_operator_name
597 *
598 * Description Send operator name event to application.
599 *
600 *
601 * Returns void
602 *
603 ******************************************************************************/
bta_hf_client_operator_name(tBTA_HF_CLIENT_CB * client_cb,char * name)604 void bta_hf_client_operator_name(tBTA_HF_CLIENT_CB* client_cb, char* name) {
605 tBTA_HF_CLIENT evt;
606
607 memset(&evt, 0, sizeof(evt));
608
609 strlcpy(evt.operator_name.name, name, BTA_HF_CLIENT_OPERATOR_NAME_LEN + 1);
610 evt.operator_name.name[BTA_HF_CLIENT_OPERATOR_NAME_LEN] = '\0';
611
612 evt.operator_name.bd_addr = client_cb->peer_addr;
613 bta_hf_client_app_callback(BTA_HF_CLIENT_OPERATOR_NAME_EVT, &evt);
614 }
615
616 /*******************************************************************************
617 *
618 * Function bta_hf_client_clip
619 *
620 * Description Send CLIP event to application.
621 *
622 *
623 * Returns void
624 *
625 ******************************************************************************/
bta_hf_client_clip(tBTA_HF_CLIENT_CB * client_cb,char * number)626 void bta_hf_client_clip(tBTA_HF_CLIENT_CB* client_cb, char* number) {
627 tBTA_HF_CLIENT evt;
628
629 memset(&evt, 0, sizeof(evt));
630
631 strlcpy(evt.number.number, number, BTA_HF_CLIENT_NUMBER_LEN + 1);
632 evt.number.number[BTA_HF_CLIENT_NUMBER_LEN] = '\0';
633
634 evt.number.bd_addr = client_cb->peer_addr;
635 bta_hf_client_app_callback(BTA_HF_CLIENT_CLIP_EVT, &evt);
636 }
637
638 /*******************************************************************************
639 *
640 * Function bta_hf_client_ccwa
641 *
642 * Description Send CLIP event to application.
643 *
644 *
645 * Returns void
646 *
647 ******************************************************************************/
bta_hf_client_ccwa(tBTA_HF_CLIENT_CB * client_cb,char * number)648 void bta_hf_client_ccwa(tBTA_HF_CLIENT_CB* client_cb, char* number) {
649 tBTA_HF_CLIENT evt;
650
651 memset(&evt, 0, sizeof(evt));
652
653 strlcpy(evt.number.number, number, BTA_HF_CLIENT_NUMBER_LEN + 1);
654 evt.number.number[BTA_HF_CLIENT_NUMBER_LEN] = '\0';
655
656 evt.number.bd_addr = client_cb->peer_addr;
657 bta_hf_client_app_callback(BTA_HF_CLIENT_CCWA_EVT, &evt);
658 }
659
660 /*******************************************************************************
661 *
662 * Function bta_hf_client_at_result
663 *
664 * Description Send AT result event to application.
665 *
666 *
667 * Returns void
668 *
669 ******************************************************************************/
bta_hf_client_at_result(tBTA_HF_CLIENT_CB * client_cb,tBTA_HF_CLIENT_AT_RESULT_TYPE type,uint16_t cme)670 void bta_hf_client_at_result(tBTA_HF_CLIENT_CB* client_cb,
671 tBTA_HF_CLIENT_AT_RESULT_TYPE type, uint16_t cme) {
672 tBTA_HF_CLIENT evt;
673
674 memset(&evt, 0, sizeof(evt));
675
676 evt.result.type = type;
677 evt.result.cme = cme;
678
679 evt.result.bd_addr = client_cb->peer_addr;
680 bta_hf_client_app_callback(BTA_HF_CLIENT_AT_RESULT_EVT, &evt);
681 }
682
683 /*******************************************************************************
684 *
685 * Function bta_hf_client_clcc
686 *
687 * Description Send clcc event to application.
688 *
689 *
690 * Returns void
691 *
692 ******************************************************************************/
bta_hf_client_clcc(tBTA_HF_CLIENT_CB * client_cb,uint32_t idx,bool incoming,uint8_t status,bool mpty,char * number)693 void bta_hf_client_clcc(tBTA_HF_CLIENT_CB* client_cb, uint32_t idx,
694 bool incoming, uint8_t status, bool mpty,
695 char* number) {
696 tBTA_HF_CLIENT evt;
697
698 memset(&evt, 0, sizeof(evt));
699
700 evt.clcc.idx = idx;
701 evt.clcc.inc = incoming;
702 evt.clcc.status = status;
703 evt.clcc.mpty = mpty;
704
705 if (number) {
706 evt.clcc.number_present = true;
707 strlcpy(evt.clcc.number, number, BTA_HF_CLIENT_NUMBER_LEN + 1);
708 evt.clcc.number[BTA_HF_CLIENT_NUMBER_LEN] = '\0';
709 }
710
711 evt.clcc.bd_addr = client_cb->peer_addr;
712 bta_hf_client_app_callback(BTA_HF_CLIENT_CLCC_EVT, &evt);
713 }
714
715 /*******************************************************************************
716 *
717 * Function bta_hf_client_cnum
718 *
719 * Description Send cnum event to application.
720 *
721 *
722 * Returns void
723 *
724 ******************************************************************************/
bta_hf_client_cnum(tBTA_HF_CLIENT_CB * client_cb,char * number,uint16_t service)725 void bta_hf_client_cnum(tBTA_HF_CLIENT_CB* client_cb, char* number,
726 uint16_t service) {
727 tBTA_HF_CLIENT evt;
728
729 memset(&evt, 0, sizeof(evt));
730
731 evt.cnum.service = service;
732 strlcpy(evt.cnum.number, number, BTA_HF_CLIENT_NUMBER_LEN + 1);
733 evt.cnum.number[BTA_HF_CLIENT_NUMBER_LEN] = '\0';
734
735 evt.cnum.bd_addr = client_cb->peer_addr;
736 bta_hf_client_app_callback(BTA_HF_CLIENT_CNUM_EVT, &evt);
737 }
738
739 /*******************************************************************************
740 *
741 * Function bta_hf_client_binp
742 *
743 * Description Send BINP event to application.
744 *
745 *
746 * Returns void
747 *
748 ******************************************************************************/
bta_hf_client_binp(tBTA_HF_CLIENT_CB * client_cb,char * number)749 void bta_hf_client_binp(tBTA_HF_CLIENT_CB* client_cb, char* number) {
750 tBTA_HF_CLIENT evt;
751
752 memset(&evt, 0, sizeof(evt));
753
754 strlcpy(evt.number.number, number, BTA_HF_CLIENT_NUMBER_LEN + 1);
755 evt.number.number[BTA_HF_CLIENT_NUMBER_LEN] = '\0';
756
757 evt.number.bd_addr = client_cb->peer_addr;
758 bta_hf_client_app_callback(BTA_HF_CLIENT_BINP_EVT, &evt);
759 }
760
761 /******************************************************************************
762 *
763 * COMMON AT EVENTS PARSING FUNCTIONS
764 *
765 ******************************************************************************/
766
767 /* Check if prefix match and skip spaces if any */
768 #define AT_CHECK_EVENT(buf, event) \
769 do { \
770 if (strncmp("\r\n" event, buf, sizeof("\r\n" event) - 1) != 0) return buf; \
771 (buf) += sizeof("\r\n" event) - 1; \
772 while (*(buf) == ' ') (buf)++; \
773 } while (0)
774
775 /* check for <cr><lf> and forward buffer if match */
776 #define AT_CHECK_RN(buf) \
777 do { \
778 if (strncmp("\r\n", buf, sizeof("\r\n") - 1) != 0) { \
779 APPL_TRACE_DEBUG("%s: missing end <cr><lf>", __func__); \
780 return NULL; \
781 } \
782 (buf) += sizeof("\r\n") - 1; \
783 } while (0)
784
785 /* skip rest of AT string up to <cr> */
786 #define AT_SKIP_REST(buf) \
787 do { \
788 while (*(buf) != '\r') (buf)++; \
789 } while (0)
790
bta_hf_client_parse_ok(tBTA_HF_CLIENT_CB * client_cb,char * buffer)791 static char* bta_hf_client_parse_ok(tBTA_HF_CLIENT_CB* client_cb,
792 char* buffer) {
793 AT_CHECK_EVENT(buffer, "OK");
794 AT_CHECK_RN(buffer);
795
796 bta_hf_client_handle_ok(client_cb);
797
798 return buffer;
799 }
800
bta_hf_client_parse_error(tBTA_HF_CLIENT_CB * client_cb,char * buffer)801 static char* bta_hf_client_parse_error(tBTA_HF_CLIENT_CB* client_cb,
802 char* buffer) {
803 AT_CHECK_EVENT(buffer, "ERROR");
804 AT_CHECK_RN(buffer);
805
806 bta_hf_client_handle_error(client_cb, BTA_HF_CLIENT_AT_RESULT_ERROR, 0);
807
808 return buffer;
809 }
810
bta_hf_client_parse_ring(tBTA_HF_CLIENT_CB * client_cb,char * buffer)811 static char* bta_hf_client_parse_ring(tBTA_HF_CLIENT_CB* client_cb,
812 char* buffer) {
813 AT_CHECK_EVENT(buffer, "RING");
814 AT_CHECK_RN(buffer);
815
816 bta_hf_client_handle_ring(client_cb);
817
818 return buffer;
819 }
820
821 /* generic uint32 parser */
bta_hf_client_parse_uint32(tBTA_HF_CLIENT_CB * client_cb,char * buffer,void (* handler_callback)(tBTA_HF_CLIENT_CB *,uint32_t))822 static char* bta_hf_client_parse_uint32(
823 tBTA_HF_CLIENT_CB* client_cb, char* buffer,
824 void (*handler_callback)(tBTA_HF_CLIENT_CB*, uint32_t)) {
825 uint32_t value;
826 int res;
827 int offset;
828
829 res = sscanf(buffer, "%u%n", &value, &offset);
830 if (res < 1) {
831 return NULL;
832 }
833
834 buffer += offset;
835
836 AT_CHECK_RN(buffer);
837
838 handler_callback(client_cb, value);
839 return buffer;
840 }
841
bta_hf_client_parse_brsf(tBTA_HF_CLIENT_CB * client_cb,char * buffer)842 static char* bta_hf_client_parse_brsf(tBTA_HF_CLIENT_CB* client_cb,
843 char* buffer) {
844 AT_CHECK_EVENT(buffer, "+BRSF:");
845
846 return bta_hf_client_parse_uint32(client_cb, buffer,
847 bta_hf_client_handle_brsf);
848 }
849
bta_hf_client_parse_cind_values(tBTA_HF_CLIENT_CB * client_cb,char * buffer)850 static char* bta_hf_client_parse_cind_values(tBTA_HF_CLIENT_CB* client_cb,
851 char* buffer) {
852 /* value and its position */
853 uint16_t index = 0;
854 uint32_t value = 0;
855
856 int offset;
857 int res;
858
859 while ((res = sscanf(buffer, "%u%n", &value, &offset)) > 0) {
860 /* decides if its valid index and value, if yes stores it */
861 bta_hf_client_handle_cind_value(client_cb, index, value);
862
863 buffer += offset;
864
865 /* check if more values are present */
866 if (*buffer != ',') {
867 break;
868 }
869
870 index++;
871 buffer++;
872 }
873
874 if (res > 0) {
875 AT_CHECK_RN(buffer);
876 return buffer;
877 }
878
879 return NULL;
880 }
881
bta_hf_client_parse_cind_list(tBTA_HF_CLIENT_CB * client_cb,char * buffer)882 static char* bta_hf_client_parse_cind_list(tBTA_HF_CLIENT_CB* client_cb,
883 char* buffer) {
884 int offset = 0;
885 char name[129];
886 uint32_t min, max;
887 uint32_t index = 0;
888 int res;
889
890 while ((res = sscanf(buffer, "(\"%128[^\"]\",(%u%*[-,]%u))%n", name, &min,
891 &max, &offset)) > 2) {
892 bta_hf_client_handle_cind_list_item(client_cb, name, min, max, index);
893 if (offset == 0) {
894 APPL_TRACE_ERROR("%s: Format Error %s", __func__, buffer);
895 return NULL;
896 }
897
898 buffer += offset;
899 index++;
900
901 if (*buffer != ',') {
902 break;
903 }
904
905 buffer++;
906 }
907
908 if (res > 2) {
909 AT_CHECK_RN(buffer);
910 return buffer;
911 }
912
913 return NULL;
914 }
915
bta_hf_client_parse_cind(tBTA_HF_CLIENT_CB * client_cb,char * buffer)916 static char* bta_hf_client_parse_cind(tBTA_HF_CLIENT_CB* client_cb,
917 char* buffer) {
918 AT_CHECK_EVENT(buffer, "+CIND:");
919
920 if (*buffer == '(') return bta_hf_client_parse_cind_list(client_cb, buffer);
921
922 return bta_hf_client_parse_cind_values(client_cb, buffer);
923 }
924
bta_hf_client_parse_chld(tBTA_HF_CLIENT_CB * client_cb,char * buffer)925 static char* bta_hf_client_parse_chld(tBTA_HF_CLIENT_CB* client_cb,
926 char* buffer) {
927 AT_CHECK_EVENT(buffer, "+CHLD:");
928
929 if (*buffer != '(') {
930 return NULL;
931 }
932
933 buffer++;
934
935 while (*buffer != '\0') {
936 if (strncmp("0", buffer, 1) == 0) {
937 bta_hf_client_handle_chld(client_cb, BTA_HF_CLIENT_CHLD_REL);
938 buffer++;
939 } else if (strncmp("1x", buffer, 2) == 0) {
940 bta_hf_client_handle_chld(client_cb, BTA_HF_CLIENT_CHLD_REL_X);
941 buffer += 2;
942 } else if (strncmp("1", buffer, 1) == 0) {
943 bta_hf_client_handle_chld(client_cb, BTA_HF_CLIENT_CHLD_REL_ACC);
944 buffer++;
945 } else if (strncmp("2x", buffer, 2) == 0) {
946 bta_hf_client_handle_chld(client_cb, BTA_HF_CLIENT_CHLD_PRIV_X);
947 buffer += 2;
948 } else if (strncmp("2", buffer, 1) == 0) {
949 bta_hf_client_handle_chld(client_cb, BTA_HF_CLIENT_CHLD_HOLD_ACC);
950 buffer++;
951 } else if (strncmp("3", buffer, 1) == 0) {
952 bta_hf_client_handle_chld(client_cb, BTA_HF_CLIENT_CHLD_MERGE);
953 buffer++;
954 } else if (strncmp("4", buffer, 1) == 0) {
955 bta_hf_client_handle_chld(client_cb, BTA_HF_CLIENT_CHLD_MERGE_DETACH);
956 buffer++;
957 } else {
958 return NULL;
959 }
960
961 if (*buffer == ',') {
962 buffer++;
963 continue;
964 }
965
966 if (*buffer == ')') {
967 buffer++;
968 break;
969 }
970
971 return NULL;
972 }
973
974 AT_CHECK_RN(buffer);
975
976 return buffer;
977 }
978
bta_hf_client_parse_ciev(tBTA_HF_CLIENT_CB * client_cb,char * buffer)979 static char* bta_hf_client_parse_ciev(tBTA_HF_CLIENT_CB* client_cb,
980 char* buffer) {
981 uint32_t index, value;
982 int res;
983 int offset = 0;
984
985 AT_CHECK_EVENT(buffer, "+CIEV:");
986
987 res = sscanf(buffer, "%u,%u%n", &index, &value, &offset);
988 if (res < 2) {
989 return NULL;
990 }
991
992 if (offset == 0) {
993 APPL_TRACE_ERROR("%s: Format Error %s", __func__, buffer);
994 return NULL;
995 }
996
997 buffer += offset;
998
999 AT_CHECK_RN(buffer);
1000
1001 bta_hf_client_handle_ciev(client_cb, index, value);
1002 return buffer;
1003 }
1004
bta_hf_client_parse_bcs(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1005 static char* bta_hf_client_parse_bcs(tBTA_HF_CLIENT_CB* client_cb,
1006 char* buffer) {
1007 AT_CHECK_EVENT(buffer, "+BCS:");
1008
1009 return bta_hf_client_parse_uint32(client_cb, buffer,
1010 bta_hf_client_handle_bcs);
1011 }
1012
bta_hf_client_parse_bsir(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1013 static char* bta_hf_client_parse_bsir(tBTA_HF_CLIENT_CB* client_cb,
1014 char* buffer) {
1015 AT_CHECK_EVENT(buffer, "+BSIR:");
1016
1017 return bta_hf_client_parse_uint32(client_cb, buffer,
1018 bta_hf_client_handle_bsir);
1019 }
1020
bta_hf_client_parse_cmeerror(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1021 static char* bta_hf_client_parse_cmeerror(tBTA_HF_CLIENT_CB* client_cb,
1022 char* buffer) {
1023 AT_CHECK_EVENT(buffer, "+CME ERROR:");
1024
1025 return bta_hf_client_parse_uint32(client_cb, buffer,
1026 bta_hf_client_handle_cmeerror);
1027 }
1028
bta_hf_client_parse_vgm(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1029 static char* bta_hf_client_parse_vgm(tBTA_HF_CLIENT_CB* client_cb,
1030 char* buffer) {
1031 AT_CHECK_EVENT(buffer, "+VGM:");
1032
1033 return bta_hf_client_parse_uint32(client_cb, buffer,
1034 bta_hf_client_handle_vgm);
1035 }
1036
bta_hf_client_parse_vgme(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1037 static char* bta_hf_client_parse_vgme(tBTA_HF_CLIENT_CB* client_cb,
1038 char* buffer) {
1039 AT_CHECK_EVENT(buffer, "+VGM=");
1040
1041 return bta_hf_client_parse_uint32(client_cb, buffer,
1042 bta_hf_client_handle_vgm);
1043 }
1044
bta_hf_client_parse_vgs(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1045 static char* bta_hf_client_parse_vgs(tBTA_HF_CLIENT_CB* client_cb,
1046 char* buffer) {
1047 AT_CHECK_EVENT(buffer, "+VGS:");
1048
1049 return bta_hf_client_parse_uint32(client_cb, buffer,
1050 bta_hf_client_handle_vgs);
1051 }
1052
bta_hf_client_parse_vgse(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1053 static char* bta_hf_client_parse_vgse(tBTA_HF_CLIENT_CB* client_cb,
1054 char* buffer) {
1055 AT_CHECK_EVENT(buffer, "+VGS=");
1056
1057 return bta_hf_client_parse_uint32(client_cb, buffer,
1058 bta_hf_client_handle_vgs);
1059 }
1060
bta_hf_client_parse_bvra(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1061 static char* bta_hf_client_parse_bvra(tBTA_HF_CLIENT_CB* client_cb,
1062 char* buffer) {
1063 AT_CHECK_EVENT(buffer, "+BVRA:");
1064
1065 return bta_hf_client_parse_uint32(client_cb, buffer,
1066 bta_hf_client_handle_bvra);
1067 }
1068
bta_hf_client_parse_clip(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1069 static char* bta_hf_client_parse_clip(tBTA_HF_CLIENT_CB* client_cb,
1070 char* buffer) {
1071 /* spec forces 32 chars, plus \0 here */
1072 char number[33];
1073 uint32_t type = 0;
1074 int res;
1075 int offset = 0;
1076
1077 AT_CHECK_EVENT(buffer, "+CLIP:");
1078
1079 /* there might be something more after %lu but HFP doesn't care */
1080 res = sscanf(buffer, "\"%32[^\"]\",%u%n", number, &type, &offset);
1081 if (res < 2) {
1082 return NULL;
1083 }
1084
1085 if (offset == 0) {
1086 APPL_TRACE_ERROR("%s: Format Error %s", __func__, buffer);
1087 return NULL;
1088 }
1089
1090 buffer += offset;
1091
1092 AT_SKIP_REST(buffer);
1093
1094 AT_CHECK_RN(buffer);
1095
1096 bta_hf_client_handle_clip(client_cb, number, type);
1097 return buffer;
1098 }
1099
1100 /* in HFP context there is no difference between ccwa and clip */
bta_hf_client_parse_ccwa(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1101 static char* bta_hf_client_parse_ccwa(tBTA_HF_CLIENT_CB* client_cb,
1102 char* buffer) {
1103 /* ac to spec 32 chars max, plus \0 here */
1104 char number[33];
1105 uint32_t type = 0;
1106 int res;
1107 int offset = 0;
1108
1109 AT_CHECK_EVENT(buffer, "+CCWA:");
1110
1111 /* there might be something more after %lu but HFP doesn't care */
1112 res = sscanf(buffer, "\"%32[^\"]\",%u%n", number, &type, &offset);
1113 if (res < 2) {
1114 return NULL;
1115 }
1116
1117 if (offset == 0) {
1118 APPL_TRACE_ERROR("%s: Format Error %s", __func__, buffer);
1119 return NULL;
1120 }
1121
1122 buffer += offset;
1123
1124 AT_SKIP_REST(buffer);
1125
1126 AT_CHECK_RN(buffer);
1127
1128 bta_hf_client_handle_ccwa(client_cb, number, type);
1129 return buffer;
1130 }
1131
bta_hf_client_parse_cops(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1132 static char* bta_hf_client_parse_cops(tBTA_HF_CLIENT_CB* client_cb,
1133 char* buffer) {
1134 uint8_t mode;
1135 /* spec forces 16 chars max, plus \0 here */
1136 char opstr[17];
1137 int res;
1138 int offset = 0;
1139
1140 AT_CHECK_EVENT(buffer, "+COPS:");
1141
1142 /* TODO: Not sure if operator string actually can contain escaped " char
1143 * inside */
1144 res = sscanf(buffer, "%hhi,0,\"%16[^\"]\"%n", &mode, opstr, &offset);
1145 if (res < 2) {
1146 return NULL;
1147 }
1148 /* Abort in case offset not set because of format error */
1149 if (offset == 0) {
1150 APPL_TRACE_ERROR("%s: Format Error %s", __func__, buffer);
1151 return NULL;
1152 }
1153
1154 buffer += offset;
1155
1156 AT_SKIP_REST(buffer);
1157
1158 AT_CHECK_RN(buffer);
1159
1160 bta_hf_client_handle_cops(client_cb, opstr, mode);
1161 // check for OK Response in end
1162 AT_CHECK_EVENT(buffer, "OK");
1163 AT_CHECK_RN(buffer);
1164
1165 bta_hf_client_handle_ok(client_cb);
1166
1167 return buffer;
1168 }
1169
bta_hf_client_parse_binp(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1170 static char* bta_hf_client_parse_binp(tBTA_HF_CLIENT_CB* client_cb,
1171 char* buffer) {
1172 /* HFP only supports phone number as BINP data */
1173 /* phone number is 32 chars plus one for \0*/
1174 char numstr[33];
1175 int res;
1176 int offset = 0;
1177
1178 AT_CHECK_EVENT(buffer, "+BINP:");
1179
1180 res = sscanf(buffer, "\"%32[^\"]\"\r\n%n", numstr, &offset);
1181 if (res < 1) {
1182 return NULL;
1183 }
1184
1185 /* Abort in case offset not set because of format error */
1186 if (offset == 0) {
1187 APPL_TRACE_ERROR("%s: Format Error %s", __func__, buffer);
1188 return NULL;
1189 }
1190
1191 buffer += offset;
1192
1193 /* some phones might sent type as well, just skip it */
1194 AT_SKIP_REST(buffer);
1195
1196 AT_CHECK_RN(buffer);
1197
1198 bta_hf_client_handle_binp(client_cb, numstr);
1199
1200 // check for OK response in end
1201 AT_CHECK_EVENT(buffer, "OK");
1202 AT_CHECK_RN(buffer);
1203
1204 bta_hf_client_handle_ok(client_cb);
1205
1206 return buffer;
1207 }
1208
bta_hf_client_parse_clcc(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1209 static char* bta_hf_client_parse_clcc(tBTA_HF_CLIENT_CB* client_cb,
1210 char* buffer) {
1211 uint16_t idx, dir, status, mode, mpty;
1212 char numstr[33]; /* spec forces 32 chars, plus one for \0*/
1213 uint16_t type;
1214 int res;
1215 int offset = 0;
1216
1217 AT_CHECK_EVENT(buffer, "+CLCC:");
1218
1219 res = sscanf(buffer, "%hu,%hu,%hu,%hu,%hu%n", &idx, &dir, &status, &mode,
1220 &mpty, &offset);
1221 if (res < 5) {
1222 return NULL;
1223 }
1224
1225 /* Abort in case offset not set because of format error */
1226 if (offset == 0) {
1227 APPL_TRACE_ERROR("%s: Format Error %s", __func__, buffer);
1228 return NULL;
1229 }
1230
1231 buffer += offset;
1232 offset = 0;
1233
1234 /* check optional part */
1235 if (*buffer == ',') {
1236 int res2 = sscanf(buffer, ",\"%32[^\"]\",%hu%n", numstr, &type, &offset);
1237 if (res2 < 0) return NULL;
1238
1239 if (res2 == 0) {
1240 res2 = sscanf(buffer, ",\"\",%hu%n", &type, &offset);
1241 if (res2 < 0) return NULL;
1242
1243 /* numstr is not matched in second attempt, correct this */
1244 res2++;
1245 numstr[0] = '\0';
1246 }
1247
1248 if (res2 >= 2) {
1249 res += res2;
1250 /* Abort in case offset not set because of format error */
1251 if (offset == 0) {
1252 APPL_TRACE_ERROR("%s: Format Error %s", __func__, buffer);
1253 return NULL;
1254 }
1255
1256 buffer += offset;
1257 }
1258 }
1259
1260 /* Skip any remaing param,as they are not defined by BT HFP spec */
1261 AT_SKIP_REST(buffer);
1262 AT_CHECK_RN(buffer);
1263
1264 if (res > 6) {
1265 /* we also have last two optional parameters */
1266 bta_hf_client_handle_clcc(client_cb, idx, dir, status, mode, mpty, numstr,
1267 type);
1268 } else {
1269 /* we didn't get the last two parameters */
1270 bta_hf_client_handle_clcc(client_cb, idx, dir, status, mode, mpty, NULL, 0);
1271 }
1272
1273 // check for OK response in end
1274 AT_CHECK_EVENT(buffer, "OK");
1275 AT_CHECK_RN(buffer);
1276
1277 bta_hf_client_handle_ok(client_cb);
1278 return buffer;
1279 }
1280
bta_hf_client_parse_cnum(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1281 static char* bta_hf_client_parse_cnum(tBTA_HF_CLIENT_CB* client_cb,
1282 char* buffer) {
1283 char numstr[33]; /* spec forces 32 chars, plus one for \0*/
1284 uint16_t type;
1285 uint16_t service =
1286 0; /* 0 in case this optional parameter is not being sent */
1287 int res;
1288 int offset = 0;
1289
1290 AT_CHECK_EVENT(buffer, "+CNUM:");
1291
1292 res = sscanf(buffer, ",\"%32[^\"]\",%hu,,%hu%n", numstr, &type, &service,
1293 &offset);
1294 if (res < 0) {
1295 return NULL;
1296 }
1297
1298 if (res == 0) {
1299 res = sscanf(buffer, ",\"\",%hu,,%hu%n", &type, &service, &offset);
1300 if (res < 0) {
1301 return NULL;
1302 }
1303
1304 /* numstr is not matched in second attempt, correct this */
1305 res++;
1306 numstr[0] = '\0';
1307 }
1308
1309 if (res < 3) {
1310 return NULL;
1311 }
1312
1313 /* Abort in case offset not set because of format error */
1314 if (offset == 0) {
1315 APPL_TRACE_ERROR("%s: Format Error %s", __func__, buffer);
1316 return NULL;
1317 }
1318
1319 buffer += offset;
1320
1321 AT_CHECK_RN(buffer);
1322
1323 /* service is optional */
1324 if (res == 2) {
1325 bta_hf_client_handle_cnum(client_cb, numstr, type, service);
1326 return buffer;
1327 }
1328
1329 if (service != 4 && service != 5) {
1330 return NULL;
1331 }
1332
1333 bta_hf_client_handle_cnum(client_cb, numstr, type, service);
1334
1335 // check for OK response in end
1336 AT_CHECK_EVENT(buffer, "OK");
1337 AT_CHECK_RN(buffer);
1338
1339 bta_hf_client_handle_ok(client_cb);
1340 return buffer;
1341 }
1342
bta_hf_client_parse_btrh(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1343 static char* bta_hf_client_parse_btrh(tBTA_HF_CLIENT_CB* client_cb,
1344 char* buffer) {
1345 uint16_t code = 0;
1346 int res;
1347 int offset;
1348
1349 AT_CHECK_EVENT(buffer, "+BTRH:");
1350
1351 res = sscanf(buffer, "%hu%n", &code, &offset);
1352 if (res < 1) {
1353 return NULL;
1354 }
1355
1356 buffer += offset;
1357
1358 AT_CHECK_RN(buffer);
1359
1360 bta_hf_client_handle_btrh(client_cb, code);
1361 return buffer;
1362 }
1363
bta_hf_client_parse_busy(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1364 static char* bta_hf_client_parse_busy(tBTA_HF_CLIENT_CB* client_cb,
1365 char* buffer) {
1366 AT_CHECK_EVENT(buffer, "BUSY");
1367 AT_CHECK_RN(buffer);
1368
1369 bta_hf_client_handle_error(client_cb, BTA_HF_CLIENT_AT_RESULT_BUSY, 0);
1370
1371 return buffer;
1372 }
1373
bta_hf_client_parse_delayed(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1374 static char* bta_hf_client_parse_delayed(tBTA_HF_CLIENT_CB* client_cb,
1375 char* buffer) {
1376 AT_CHECK_EVENT(buffer, "DELAYED");
1377 AT_CHECK_RN(buffer);
1378
1379 bta_hf_client_handle_error(client_cb, BTA_HF_CLIENT_AT_RESULT_DELAY, 0);
1380
1381 return buffer;
1382 }
1383
bta_hf_client_parse_no_carrier(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1384 static char* bta_hf_client_parse_no_carrier(tBTA_HF_CLIENT_CB* client_cb,
1385 char* buffer) {
1386 AT_CHECK_EVENT(buffer, "NO CARRIER");
1387 AT_CHECK_RN(buffer);
1388
1389 bta_hf_client_handle_error(client_cb, BTA_HF_CLIENT_AT_RESULT_NO_CARRIER, 0);
1390
1391 return buffer;
1392 }
1393
bta_hf_client_parse_no_answer(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1394 static char* bta_hf_client_parse_no_answer(tBTA_HF_CLIENT_CB* client_cb,
1395 char* buffer) {
1396 AT_CHECK_EVENT(buffer, "NO ANSWER");
1397 AT_CHECK_RN(buffer);
1398
1399 bta_hf_client_handle_error(client_cb, BTA_HF_CLIENT_AT_RESULT_NO_ANSWER, 0);
1400
1401 return buffer;
1402 }
1403
bta_hf_client_parse_blacklisted(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1404 static char* bta_hf_client_parse_blacklisted(tBTA_HF_CLIENT_CB* client_cb,
1405 char* buffer) {
1406 AT_CHECK_EVENT(buffer, "BLACKLISTED");
1407 AT_CHECK_RN(buffer);
1408
1409 bta_hf_client_handle_error(client_cb, BTA_HF_CLIENT_AT_RESULT_BLACKLISTED, 0);
1410
1411 return buffer;
1412 }
1413
bta_hf_client_skip_unknown(tBTA_HF_CLIENT_CB * client_cb,char * buffer)1414 static char* bta_hf_client_skip_unknown(tBTA_HF_CLIENT_CB* client_cb,
1415 char* buffer) {
1416 char* start;
1417 char* tmp;
1418
1419 tmp = strstr(buffer, "\r\n");
1420 if (tmp == NULL) {
1421 return NULL;
1422 }
1423
1424 buffer += 2;
1425 start = buffer;
1426
1427 tmp = strstr(buffer, "\r\n");
1428 if (tmp == NULL) {
1429 return NULL;
1430 }
1431
1432 buffer = tmp + 2;
1433
1434 APPL_TRACE_DEBUG("%s: %.*s", __func__, buffer - start - 2, start);
1435
1436 return buffer;
1437 }
1438
1439 /******************************************************************************
1440 * SUPPORTED EVENT MESSAGES
1441 ******************************************************************************/
1442
1443 /* returned values are as follow:
1444 * != NULL && != buf : match and parsed ok
1445 * == NULL : match but parse failed
1446 * != NULL && == buf : no match
1447 */
1448 typedef char* (*tBTA_HF_CLIENT_PARSER_CALLBACK)(tBTA_HF_CLIENT_CB*, char*);
1449
1450 static const tBTA_HF_CLIENT_PARSER_CALLBACK bta_hf_client_parser_cb[] = {
1451 bta_hf_client_parse_ok, bta_hf_client_parse_error,
1452 bta_hf_client_parse_ring, bta_hf_client_parse_brsf,
1453 bta_hf_client_parse_cind, bta_hf_client_parse_ciev,
1454 bta_hf_client_parse_chld, bta_hf_client_parse_bcs,
1455 bta_hf_client_parse_bsir, bta_hf_client_parse_cmeerror,
1456 bta_hf_client_parse_vgm, bta_hf_client_parse_vgme,
1457 bta_hf_client_parse_vgs, bta_hf_client_parse_vgse,
1458 bta_hf_client_parse_bvra, bta_hf_client_parse_clip,
1459 bta_hf_client_parse_ccwa, bta_hf_client_parse_cops,
1460 bta_hf_client_parse_binp, bta_hf_client_parse_clcc,
1461 bta_hf_client_parse_cnum, bta_hf_client_parse_btrh,
1462 bta_hf_client_parse_busy, bta_hf_client_parse_delayed,
1463 bta_hf_client_parse_no_carrier, bta_hf_client_parse_no_answer,
1464 bta_hf_client_parse_blacklisted, bta_hf_client_skip_unknown};
1465
1466 /* calculate supported event list length */
1467 static const uint16_t bta_hf_client_parser_cb_count =
1468 sizeof(bta_hf_client_parser_cb) / sizeof(bta_hf_client_parser_cb[0]);
1469
1470 #ifdef BTA_HF_CLIENT_AT_DUMP
bta_hf_client_dump_at(tBTA_HF_CLIENT_CB * client_cb)1471 static void bta_hf_client_dump_at(tBTA_HF_CLIENT_CB* client_cb) {
1472 char dump[(4 * BTA_HF_CLIENT_AT_PARSER_MAX_LEN) + 1];
1473 char *p1, *p2;
1474
1475 p1 = client_cb->at_cb.buf;
1476 p2 = dump;
1477
1478 while (*p1 != '\0') {
1479 if (*p1 == '\r') {
1480 strlcpy(p2, "<cr>", 4);
1481 p2 += 4;
1482 } else if (*p1 == '\n') {
1483 strlcpy(p2, "<lf>", 4);
1484 p2 += 4;
1485 } else {
1486 *p2 = *p1;
1487 p2++;
1488 }
1489 p1++;
1490 }
1491
1492 *p2 = '\0';
1493
1494 APPL_TRACE_DEBUG("%s: %s", __func__, dump);
1495 }
1496 #endif
1497
bta_hf_client_at_parse_start(tBTA_HF_CLIENT_CB * client_cb)1498 static void bta_hf_client_at_parse_start(tBTA_HF_CLIENT_CB* client_cb) {
1499 char* buf = client_cb->at_cb.buf;
1500
1501 APPL_TRACE_DEBUG("%s", __func__);
1502
1503 #ifdef BTA_HF_CLIENT_AT_DUMP
1504 bta_hf_client_dump_at(client_cb);
1505 #endif
1506
1507 while (*buf != '\0') {
1508 int i;
1509 char* tmp = NULL;
1510
1511 for (i = 0; i < bta_hf_client_parser_cb_count; i++) {
1512 tmp = bta_hf_client_parser_cb[i](client_cb, buf);
1513 if (tmp == NULL) {
1514 APPL_TRACE_ERROR("HFPCient: AT event/reply parsing failed, skipping");
1515 tmp = bta_hf_client_skip_unknown(client_cb, buf);
1516 break;
1517 }
1518
1519 /* matched or unknown skipped, if unknown failed tmp is NULL so
1520 this is also handled */
1521 if (tmp != buf) {
1522 buf = tmp;
1523 break;
1524 }
1525 }
1526
1527 /* could not skip unknown (received garbage?)... disconnect */
1528 if (tmp == NULL) {
1529 APPL_TRACE_ERROR(
1530 "HFPCient: could not skip unknown AT event, disconnecting");
1531 bta_hf_client_at_reset(client_cb);
1532
1533 tBTA_HF_CLIENT_DATA msg;
1534 msg.hdr.layer_specific = client_cb->handle;
1535 bta_hf_client_sm_execute(BTA_HF_CLIENT_API_CLOSE_EVT, &msg);
1536 return;
1537 }
1538
1539 buf = tmp;
1540 }
1541 }
1542
bta_hf_client_check_at_complete(tBTA_HF_CLIENT_CB * client_cb)1543 static bool bta_hf_client_check_at_complete(tBTA_HF_CLIENT_CB* client_cb) {
1544 bool ret = false;
1545 tBTA_HF_CLIENT_AT_CB* at_cb = &client_cb->at_cb;
1546
1547 if (at_cb->offset >= BTA_HF_CLIENT_AT_EVENT_MIN_LEN) {
1548 if (at_cb->buf[at_cb->offset - 2] == '\r' &&
1549 at_cb->buf[at_cb->offset - 1] == '\n') {
1550 ret = true;
1551 }
1552 }
1553
1554 APPL_TRACE_DEBUG("%s: %d", __func__, ret);
1555
1556 return ret;
1557 }
1558
bta_hf_client_at_clear_buf(tBTA_HF_CLIENT_CB * client_cb)1559 static void bta_hf_client_at_clear_buf(tBTA_HF_CLIENT_CB* client_cb) {
1560 memset(client_cb->at_cb.buf, 0, sizeof(client_cb->at_cb.buf));
1561 client_cb->at_cb.offset = 0;
1562 }
1563
1564 /******************************************************************************
1565 *
1566 * MAIN PARSING FUNCTION
1567 *
1568 *
1569 ******************************************************************************/
bta_hf_client_at_parse(tBTA_HF_CLIENT_CB * client_cb,char * buf,unsigned int len)1570 void bta_hf_client_at_parse(tBTA_HF_CLIENT_CB* client_cb, char* buf,
1571 unsigned int len) {
1572 APPL_TRACE_DEBUG("%s: offset: %u len: %u", __func__, client_cb->at_cb.offset,
1573 len);
1574
1575 if (len + client_cb->at_cb.offset > BTA_HF_CLIENT_AT_PARSER_MAX_LEN) {
1576 char tmp_buff[BTA_HF_CLIENT_AT_PARSER_MAX_LEN];
1577 unsigned int tmp = client_cb->at_cb.offset;
1578 unsigned int space_left =
1579 BTA_HF_CLIENT_AT_PARSER_MAX_LEN - client_cb->at_cb.offset;
1580
1581 APPL_TRACE_DEBUG("%s: overrun, trying to recover", __func__);
1582
1583 /* fill up parser buffer */
1584 memcpy(client_cb->at_cb.buf + client_cb->at_cb.offset, buf, space_left);
1585 len -= space_left;
1586 buf += space_left;
1587 client_cb->at_cb.offset += space_left;
1588
1589 /* find end of last complete command before proceeding */
1590 while (bta_hf_client_check_at_complete(client_cb) == false) {
1591 if (client_cb->at_cb.offset == 0) {
1592 APPL_TRACE_ERROR("HFPClient: AT parser buffer overrun, disconnecting");
1593
1594 bta_hf_client_at_reset(client_cb);
1595
1596 tBTA_HF_CLIENT_DATA msg;
1597 msg.hdr.layer_specific = client_cb->handle;
1598 bta_hf_client_sm_execute(BTA_HF_CLIENT_API_CLOSE_EVT, &msg);
1599 return;
1600 }
1601
1602 client_cb->at_cb.offset--;
1603 }
1604
1605 /* cut buffer to complete AT event and keep cut data */
1606 tmp += space_left - client_cb->at_cb.offset;
1607 memcpy(tmp_buff, client_cb->at_cb.buf + client_cb->at_cb.offset, tmp);
1608 client_cb->at_cb.buf[client_cb->at_cb.offset] = '\0';
1609
1610 /* parse */
1611 bta_hf_client_at_parse_start(client_cb);
1612 bta_hf_client_at_clear_buf(client_cb);
1613
1614 /* recover cut data */
1615 memcpy(client_cb->at_cb.buf, tmp_buff, tmp);
1616 client_cb->at_cb.offset += tmp;
1617 }
1618
1619 memcpy(client_cb->at_cb.buf + client_cb->at_cb.offset, buf, len);
1620 client_cb->at_cb.offset += len;
1621
1622 /* If last event is complete, parsing can be started */
1623 if (bta_hf_client_check_at_complete(client_cb) == true) {
1624 bta_hf_client_at_parse_start(client_cb);
1625 bta_hf_client_at_clear_buf(client_cb);
1626 }
1627 }
1628
bta_hf_client_send_at_brsf(tBTA_HF_CLIENT_CB * client_cb,tBTA_HF_CLIENT_FEAT features)1629 void bta_hf_client_send_at_brsf(tBTA_HF_CLIENT_CB* client_cb,
1630 tBTA_HF_CLIENT_FEAT features) {
1631 char buf[BTA_HF_CLIENT_AT_MAX_LEN];
1632 int at_len;
1633
1634 APPL_TRACE_DEBUG("%s", __func__);
1635
1636 at_len = snprintf(buf, sizeof(buf), "AT+BRSF=%u\r", features);
1637 if (at_len < 0) {
1638 APPL_TRACE_ERROR("%s: AT command Framing error", __func__);
1639 return;
1640 }
1641
1642 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_BRSF, buf, at_len);
1643 }
1644
bta_hf_client_send_at_bac(tBTA_HF_CLIENT_CB * client_cb)1645 void bta_hf_client_send_at_bac(tBTA_HF_CLIENT_CB* client_cb) {
1646 const char* buf;
1647
1648 APPL_TRACE_DEBUG("%s", __func__);
1649
1650 buf = "AT+BAC=1,2\r";
1651
1652 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_BAC, buf, strlen(buf));
1653 }
1654
bta_hf_client_send_at_bcs(tBTA_HF_CLIENT_CB * client_cb,uint32_t codec)1655 void bta_hf_client_send_at_bcs(tBTA_HF_CLIENT_CB* client_cb, uint32_t codec) {
1656 char buf[BTA_HF_CLIENT_AT_MAX_LEN];
1657 int at_len;
1658
1659 APPL_TRACE_DEBUG("%s", __func__);
1660
1661 at_len = snprintf(buf, sizeof(buf), "AT+BCS=%u\r", codec);
1662 if (at_len < 0) {
1663 APPL_TRACE_ERROR("%s: AT command Framing error", __func__);
1664 return;
1665 }
1666
1667 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_BCS, buf, at_len);
1668 }
1669
bta_hf_client_send_at_cind(tBTA_HF_CLIENT_CB * client_cb,bool status)1670 void bta_hf_client_send_at_cind(tBTA_HF_CLIENT_CB* client_cb, bool status) {
1671 const char* buf;
1672 tBTA_HF_CLIENT_AT_CMD cmd;
1673
1674 APPL_TRACE_DEBUG("%s", __func__);
1675
1676 if (status) {
1677 buf = "AT+CIND?\r";
1678 cmd = BTA_HF_CLIENT_AT_CIND_STATUS;
1679 } else {
1680 buf = "AT+CIND=?\r";
1681 cmd = BTA_HF_CLIENT_AT_CIND;
1682 }
1683
1684 bta_hf_client_send_at(client_cb, cmd, buf, strlen(buf));
1685 }
1686
bta_hf_client_send_at_cmer(tBTA_HF_CLIENT_CB * client_cb,bool activate)1687 void bta_hf_client_send_at_cmer(tBTA_HF_CLIENT_CB* client_cb, bool activate) {
1688 const char* buf;
1689
1690 APPL_TRACE_DEBUG("%s", __func__);
1691
1692 if (activate)
1693 buf = "AT+CMER=3,0,0,1\r";
1694 else
1695 buf = "AT+CMER=3,0,0,0\r";
1696
1697 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_CMER, buf, strlen(buf));
1698 }
1699
bta_hf_client_send_at_chld(tBTA_HF_CLIENT_CB * client_cb,char cmd,uint32_t idx)1700 void bta_hf_client_send_at_chld(tBTA_HF_CLIENT_CB* client_cb, char cmd,
1701 uint32_t idx) {
1702 char buf[BTA_HF_CLIENT_AT_MAX_LEN];
1703 int at_len;
1704
1705 APPL_TRACE_DEBUG("%s", __func__);
1706
1707 if (idx > 0)
1708 at_len = snprintf(buf, sizeof(buf), "AT+CHLD=%c%u\r", cmd, idx);
1709 else
1710 at_len = snprintf(buf, sizeof(buf), "AT+CHLD=%c\r", cmd);
1711
1712 if (at_len < 0) {
1713 APPL_TRACE_ERROR("%s: AT command Framing error", __func__);
1714 return;
1715 }
1716
1717 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_CHLD, buf, at_len);
1718 }
1719
bta_hf_client_send_at_clip(tBTA_HF_CLIENT_CB * client_cb,bool activate)1720 void bta_hf_client_send_at_clip(tBTA_HF_CLIENT_CB* client_cb, bool activate) {
1721 const char* buf;
1722
1723 APPL_TRACE_DEBUG("%s", __func__);
1724
1725 if (activate)
1726 buf = "AT+CLIP=1\r";
1727 else
1728 buf = "AT+CLIP=0\r";
1729
1730 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_CLIP, buf, strlen(buf));
1731 }
1732
bta_hf_client_send_at_ccwa(tBTA_HF_CLIENT_CB * client_cb,bool activate)1733 void bta_hf_client_send_at_ccwa(tBTA_HF_CLIENT_CB* client_cb, bool activate) {
1734 const char* buf;
1735
1736 APPL_TRACE_DEBUG("%s", __func__);
1737
1738 if (activate)
1739 buf = "AT+CCWA=1\r";
1740 else
1741 buf = "AT+CCWA=0\r";
1742
1743 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_CCWA, buf, strlen(buf));
1744 }
1745
bta_hf_client_send_at_cmee(tBTA_HF_CLIENT_CB * client_cb,bool activate)1746 void bta_hf_client_send_at_cmee(tBTA_HF_CLIENT_CB* client_cb, bool activate) {
1747 const char* buf;
1748
1749 APPL_TRACE_DEBUG("%s", __func__);
1750
1751 if (activate)
1752 buf = "AT+CMEE=1\r";
1753 else
1754 buf = "AT+CMEE=0\r";
1755
1756 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_CMEE, buf, strlen(buf));
1757 }
1758
bta_hf_client_send_at_cops(tBTA_HF_CLIENT_CB * client_cb,bool query)1759 void bta_hf_client_send_at_cops(tBTA_HF_CLIENT_CB* client_cb, bool query) {
1760 const char* buf;
1761
1762 APPL_TRACE_DEBUG("%s", __func__);
1763
1764 if (query)
1765 buf = "AT+COPS?\r";
1766 else
1767 buf = "AT+COPS=3,0\r";
1768
1769 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_COPS, buf, strlen(buf));
1770 }
1771
bta_hf_client_send_at_clcc(tBTA_HF_CLIENT_CB * client_cb)1772 void bta_hf_client_send_at_clcc(tBTA_HF_CLIENT_CB* client_cb) {
1773 const char* buf;
1774
1775 APPL_TRACE_DEBUG("%s", __func__);
1776
1777 buf = "AT+CLCC\r";
1778
1779 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_CLCC, buf, strlen(buf));
1780 }
1781
bta_hf_client_send_at_bvra(tBTA_HF_CLIENT_CB * client_cb,bool enable)1782 void bta_hf_client_send_at_bvra(tBTA_HF_CLIENT_CB* client_cb, bool enable) {
1783 const char* buf;
1784
1785 APPL_TRACE_DEBUG("%s", __func__);
1786
1787 if (enable)
1788 buf = "AT+BVRA=1\r";
1789 else
1790 buf = "AT+BVRA=0\r";
1791
1792 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_BVRA, buf, strlen(buf));
1793 }
1794
bta_hf_client_send_at_vgs(tBTA_HF_CLIENT_CB * client_cb,uint32_t volume)1795 void bta_hf_client_send_at_vgs(tBTA_HF_CLIENT_CB* client_cb, uint32_t volume) {
1796 char buf[BTA_HF_CLIENT_AT_MAX_LEN];
1797 int at_len;
1798
1799 APPL_TRACE_DEBUG("%s", __func__);
1800
1801 at_len = snprintf(buf, sizeof(buf), "AT+VGS=%u\r", volume);
1802 if (at_len < 0) {
1803 APPL_TRACE_ERROR("%s: AT command Framing error", __func__);
1804 return;
1805 }
1806
1807 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_VGS, buf, at_len);
1808 }
1809
bta_hf_client_send_at_vgm(tBTA_HF_CLIENT_CB * client_cb,uint32_t volume)1810 void bta_hf_client_send_at_vgm(tBTA_HF_CLIENT_CB* client_cb, uint32_t volume) {
1811 char buf[BTA_HF_CLIENT_AT_MAX_LEN];
1812 int at_len;
1813
1814 APPL_TRACE_DEBUG("%s", __func__);
1815
1816 at_len = snprintf(buf, sizeof(buf), "AT+VGM=%u\r", volume);
1817 if (at_len < 0) {
1818 APPL_TRACE_ERROR("%s: AT command Framing error", __func__);
1819 return;
1820 }
1821
1822 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_VGM, buf, at_len);
1823 }
1824
bta_hf_client_send_at_atd(tBTA_HF_CLIENT_CB * client_cb,char * number,uint32_t memory)1825 void bta_hf_client_send_at_atd(tBTA_HF_CLIENT_CB* client_cb, char* number,
1826 uint32_t memory) {
1827 char buf[BTA_HF_CLIENT_AT_MAX_LEN];
1828 int at_len;
1829
1830 APPL_TRACE_DEBUG("%s", __func__);
1831
1832 if (number[0] != '\0') {
1833 at_len = snprintf(buf, sizeof(buf), "ATD%s;\r", number);
1834 } else {
1835 at_len = snprintf(buf, sizeof(buf), "ATD>%u;\r", memory);
1836 }
1837
1838 if (at_len < 0) {
1839 APPL_TRACE_ERROR("%s: error preparing ATD command", __func__);
1840 return;
1841 }
1842
1843 at_len = MIN((size_t)at_len, sizeof(buf));
1844
1845 if (at_len < 0) {
1846 APPL_TRACE_ERROR("%s: AT command Framing error", __func__);
1847 return;
1848 }
1849 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_ATD, buf, at_len);
1850 }
1851
bta_hf_client_send_at_bldn(tBTA_HF_CLIENT_CB * client_cb)1852 void bta_hf_client_send_at_bldn(tBTA_HF_CLIENT_CB* client_cb) {
1853 const char* buf;
1854
1855 APPL_TRACE_DEBUG("%s", __func__);
1856
1857 buf = "AT+BLDN\r";
1858
1859 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_BLDN, buf, strlen(buf));
1860 }
1861
bta_hf_client_send_at_ata(tBTA_HF_CLIENT_CB * client_cb)1862 void bta_hf_client_send_at_ata(tBTA_HF_CLIENT_CB* client_cb) {
1863 const char* buf;
1864
1865 APPL_TRACE_DEBUG("%s", __func__);
1866
1867 buf = "ATA\r";
1868
1869 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_ATA, buf, strlen(buf));
1870 }
1871
bta_hf_client_send_at_chup(tBTA_HF_CLIENT_CB * client_cb)1872 void bta_hf_client_send_at_chup(tBTA_HF_CLIENT_CB* client_cb) {
1873 const char* buf;
1874
1875 APPL_TRACE_DEBUG("%s", __func__);
1876
1877 buf = "AT+CHUP\r";
1878
1879 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_CHUP, buf, strlen(buf));
1880 }
1881
bta_hf_client_send_at_btrh(tBTA_HF_CLIENT_CB * client_cb,bool query,uint32_t val)1882 void bta_hf_client_send_at_btrh(tBTA_HF_CLIENT_CB* client_cb, bool query,
1883 uint32_t val) {
1884 char buf[BTA_HF_CLIENT_AT_MAX_LEN];
1885 int at_len;
1886
1887 APPL_TRACE_DEBUG("%s", __func__);
1888
1889 if (query == true) {
1890 at_len = snprintf(buf, sizeof(buf), "AT+BTRH?\r");
1891 } else {
1892 at_len = snprintf(buf, sizeof(buf), "AT+BTRH=%u\r", val);
1893 }
1894
1895 if (at_len < 0) {
1896 APPL_TRACE_ERROR("%s: AT command Framing error", __func__);
1897 return;
1898 }
1899
1900 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_BTRH, buf, at_len);
1901 }
1902
bta_hf_client_send_at_vts(tBTA_HF_CLIENT_CB * client_cb,char code)1903 void bta_hf_client_send_at_vts(tBTA_HF_CLIENT_CB* client_cb, char code) {
1904 char buf[BTA_HF_CLIENT_AT_MAX_LEN];
1905 int at_len;
1906
1907 APPL_TRACE_DEBUG("%s", __func__);
1908
1909 at_len = snprintf(buf, sizeof(buf), "AT+VTS=%c\r", code);
1910
1911 if (at_len < 0) {
1912 APPL_TRACE_ERROR("%s: AT command Framing error", __func__);
1913 return;
1914 }
1915
1916 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_VTS, buf, at_len);
1917 }
1918
bta_hf_client_send_at_bcc(tBTA_HF_CLIENT_CB * client_cb)1919 void bta_hf_client_send_at_bcc(tBTA_HF_CLIENT_CB* client_cb) {
1920 const char* buf;
1921
1922 APPL_TRACE_DEBUG("%s", __func__);
1923
1924 buf = "AT+BCC\r";
1925
1926 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_BCC, buf, strlen(buf));
1927 }
1928
bta_hf_client_send_at_cnum(tBTA_HF_CLIENT_CB * client_cb)1929 void bta_hf_client_send_at_cnum(tBTA_HF_CLIENT_CB* client_cb) {
1930 const char* buf;
1931
1932 APPL_TRACE_DEBUG("%s", __func__);
1933
1934 buf = "AT+CNUM\r";
1935
1936 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_CNUM, buf, strlen(buf));
1937 }
1938
bta_hf_client_send_at_nrec(tBTA_HF_CLIENT_CB * client_cb)1939 void bta_hf_client_send_at_nrec(tBTA_HF_CLIENT_CB* client_cb) {
1940 const char* buf;
1941
1942 APPL_TRACE_DEBUG("%s", __func__);
1943
1944 if (!(client_cb->peer_features & BTA_HF_CLIENT_PEER_FEAT_ECNR)) {
1945 APPL_TRACE_ERROR("%s: Remote does not support NREC.", __func__);
1946 return;
1947 }
1948
1949 buf = "AT+NREC=0\r";
1950
1951 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_NREC, buf, strlen(buf));
1952 }
1953
bta_hf_client_send_at_binp(tBTA_HF_CLIENT_CB * client_cb,uint32_t action)1954 void bta_hf_client_send_at_binp(tBTA_HF_CLIENT_CB* client_cb, uint32_t action) {
1955 char buf[BTA_HF_CLIENT_AT_MAX_LEN];
1956 int at_len;
1957
1958 APPL_TRACE_DEBUG("%s", __func__);
1959
1960 at_len = snprintf(buf, sizeof(buf), "AT+BINP=%u\r", action);
1961
1962 if (at_len < 0) {
1963 APPL_TRACE_ERROR("%s: AT command Framing error", __func__);
1964 return;
1965 }
1966
1967 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_BINP, buf, at_len);
1968 }
1969
bta_hf_client_send_at_bia(tBTA_HF_CLIENT_CB * client_cb)1970 void bta_hf_client_send_at_bia(tBTA_HF_CLIENT_CB* client_cb) {
1971 char buf[BTA_HF_CLIENT_AT_MAX_LEN];
1972 int at_len;
1973 int i;
1974
1975 APPL_TRACE_DEBUG("%s", __func__);
1976 if (client_cb->peer_version < HFP_VERSION_1_6) {
1977 APPL_TRACE_DEBUG("Remote does not Support AT+BIA");
1978 return;
1979 }
1980
1981 at_len = snprintf(buf, sizeof(buf), "AT+BIA=");
1982
1983 for (i = 0; i < BTA_HF_CLIENT_AT_INDICATOR_COUNT; i++) {
1984 int sup = client_cb->at_cb.indicator_lookup[i] == -1 ? 0 : 1;
1985
1986 at_len += snprintf(buf + at_len, sizeof(buf) - at_len, "%u,", sup);
1987 }
1988
1989 buf[at_len - 1] = '\r';
1990
1991 if (at_len < 0) {
1992 APPL_TRACE_ERROR("%s: AT command Framing error", __func__);
1993 return;
1994 }
1995
1996 bta_hf_client_send_at(client_cb, BTA_HF_CLIENT_AT_BIA, buf, at_len);
1997 }
1998
bta_hf_client_at_init(tBTA_HF_CLIENT_CB * client_cb)1999 void bta_hf_client_at_init(tBTA_HF_CLIENT_CB* client_cb) {
2000 alarm_free(client_cb->at_cb.resp_timer);
2001 alarm_free(client_cb->at_cb.hold_timer);
2002 memset(&(client_cb->at_cb), 0, sizeof(tBTA_HF_CLIENT_AT_CB));
2003 client_cb->at_cb.resp_timer = alarm_new("bta_hf_client.scb_at_resp_timer");
2004 client_cb->at_cb.hold_timer = alarm_new("bta_hf_client.scb_at_hold_timer");
2005 bta_hf_client_at_reset(client_cb);
2006 }
2007
bta_hf_client_at_reset(tBTA_HF_CLIENT_CB * client_cb)2008 void bta_hf_client_at_reset(tBTA_HF_CLIENT_CB* client_cb) {
2009 int i;
2010
2011 bta_hf_client_stop_at_resp_timer(client_cb);
2012 bta_hf_client_stop_at_hold_timer(client_cb);
2013
2014 bta_hf_client_clear_queued_at(client_cb);
2015
2016 bta_hf_client_at_clear_buf(client_cb);
2017
2018 for (i = 0; i < BTA_HF_CLIENT_AT_INDICATOR_COUNT; i++) {
2019 client_cb->at_cb.indicator_lookup[i] = -1;
2020 }
2021
2022 client_cb->at_cb.current_cmd = BTA_HF_CLIENT_AT_NONE;
2023 }
2024
bta_hf_client_send_at_cmd(tBTA_HF_CLIENT_DATA * p_data)2025 void bta_hf_client_send_at_cmd(tBTA_HF_CLIENT_DATA* p_data) {
2026 tBTA_HF_CLIENT_CB* client_cb =
2027 bta_hf_client_find_cb_by_handle(p_data->hdr.layer_specific);
2028 if (!client_cb) {
2029 APPL_TRACE_ERROR("%s: cb not found for handle %d", __func__,
2030 p_data->hdr.layer_specific);
2031 return;
2032 }
2033
2034 tBTA_HF_CLIENT_DATA_VAL* p_val = (tBTA_HF_CLIENT_DATA_VAL*)p_data;
2035 char buf[BTA_HF_CLIENT_AT_MAX_LEN];
2036
2037 APPL_TRACE_DEBUG("%s: at cmd: %d", __func__, p_val->uint8_val);
2038 switch (p_val->uint8_val) {
2039 case BTA_HF_CLIENT_AT_CMD_VTS:
2040 bta_hf_client_send_at_vts(client_cb, (char)p_val->uint32_val1);
2041 break;
2042 case BTA_HF_CLIENT_AT_CMD_BTRH:
2043 bta_hf_client_send_at_btrh(client_cb, false, p_val->uint32_val1);
2044 break;
2045 case BTA_HF_CLIENT_AT_CMD_CHUP:
2046 bta_hf_client_send_at_chup(client_cb);
2047 break;
2048 case BTA_HF_CLIENT_AT_CMD_CHLD:
2049 /* expects ascii code for command */
2050 bta_hf_client_send_at_chld(client_cb, '0' + p_val->uint32_val1,
2051 p_val->uint32_val2);
2052 break;
2053 case BTA_HF_CLIENT_AT_CMD_BCC:
2054 bta_hf_client_send_at_bcc(client_cb);
2055 break;
2056 case BTA_HF_CLIENT_AT_CMD_CNUM:
2057 bta_hf_client_send_at_cnum(client_cb);
2058 break;
2059 case BTA_HF_CLIENT_AT_CMD_ATA:
2060 bta_hf_client_send_at_ata(client_cb);
2061 break;
2062 case BTA_HF_CLIENT_AT_CMD_COPS:
2063 bta_hf_client_send_at_cops(client_cb, true);
2064 break;
2065 case BTA_HF_CLIENT_AT_CMD_ATD:
2066 bta_hf_client_send_at_atd(client_cb, p_val->str, p_val->uint32_val1);
2067 break;
2068 case BTA_HF_CLIENT_AT_CMD_VGM:
2069 bta_hf_client_send_at_vgm(client_cb, p_val->uint32_val1);
2070 break;
2071 case BTA_HF_CLIENT_AT_CMD_VGS:
2072 bta_hf_client_send_at_vgs(client_cb, p_val->uint32_val1);
2073 break;
2074 case BTA_HF_CLIENT_AT_CMD_BVRA:
2075 bta_hf_client_send_at_bvra(client_cb,
2076 p_val->uint32_val1 == 0 ? false : true);
2077 break;
2078 case BTA_HF_CLIENT_AT_CMD_CLCC:
2079 bta_hf_client_send_at_clcc(client_cb);
2080 break;
2081 case BTA_HF_CLIENT_AT_CMD_BINP:
2082 bta_hf_client_send_at_binp(client_cb, p_val->uint32_val1);
2083 break;
2084 case BTA_HF_CLIENT_AT_CMD_BLDN:
2085 bta_hf_client_send_at_bldn(client_cb);
2086 break;
2087 case BTA_HF_CLIENT_AT_CMD_NREC:
2088 bta_hf_client_send_at_nrec(client_cb);
2089 break;
2090 default:
2091 APPL_TRACE_ERROR("Default case");
2092 snprintf(buf, BTA_HF_CLIENT_AT_MAX_LEN,
2093 "Cmd %d 1st arg %u 2nd arg %u string arg %s", p_val->uint8_val,
2094 p_val->uint32_val1, p_val->uint32_val2, p_val->str);
2095 APPL_TRACE_ERROR("%s: AT buffer: %s ", __func__, buf);
2096 break;
2097 }
2098 }
2099