hal-handsfree.c 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882
  1. // SPDX-License-Identifier: Apache-2.0
  2. /*
  3. * Copyright (C) 2014 Intel Corporation
  4. *
  5. */
  6. #define _GNU_SOURCE
  7. #include <stdbool.h>
  8. #include <stddef.h>
  9. #include <string.h>
  10. #include <stdlib.h>
  11. #include <cutils/properties.h>
  12. #include "hal-log.h"
  13. #include "hal.h"
  14. #include "hal-msg.h"
  15. #include "ipc-common.h"
  16. #include "hal-ipc.h"
  17. #include "hal-utils.h"
  18. static const bthf_callbacks_t *cbs = NULL;
  19. static bool interface_ready(void)
  20. {
  21. return cbs != NULL;
  22. }
  23. static void handle_conn_state(void *buf, uint16_t len, int fd)
  24. {
  25. struct hal_ev_handsfree_conn_state *ev = buf;
  26. if (cbs->connection_state_cb)
  27. cbs->connection_state_cb(ev->state,
  28. (bt_bdaddr_t *) (ev->bdaddr));
  29. }
  30. static void handle_audio_state(void *buf, uint16_t len, int fd)
  31. {
  32. struct hal_ev_handsfree_audio_state *ev = buf;
  33. if (cbs->audio_state_cb)
  34. cbs->audio_state_cb(ev->state, (bt_bdaddr_t *) (ev->bdaddr));
  35. }
  36. static void handle_vr_state(void *buf, uint16_t len, int fd)
  37. {
  38. struct hal_ev_handsfree_vr_state *ev = buf;
  39. if (cbs->vr_cmd_cb)
  40. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  41. cbs->vr_cmd_cb(ev->state, (bt_bdaddr_t *) (ev->bdaddr));
  42. #else
  43. cbs->vr_cmd_cb(ev->state);
  44. #endif
  45. }
  46. static void handle_answer(void *buf, uint16_t len, int fd)
  47. {
  48. if (cbs->answer_call_cmd_cb) {
  49. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  50. struct hal_ev_handsfree_answer *ev = buf;
  51. cbs->answer_call_cmd_cb((bt_bdaddr_t *) (ev->bdaddr));
  52. #else
  53. cbs->answer_call_cmd_cb();
  54. #endif
  55. }
  56. }
  57. static void handle_hangup(void *buf, uint16_t len, int fd)
  58. {
  59. if (cbs->hangup_call_cmd_cb) {
  60. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  61. struct hal_ev_handsfree_hangup *ev = buf;
  62. cbs->hangup_call_cmd_cb((bt_bdaddr_t *) (ev->bdaddr));
  63. #else
  64. cbs->hangup_call_cmd_cb();
  65. #endif
  66. }
  67. }
  68. static void handle_volume(void *buf, uint16_t len, int fd)
  69. {
  70. struct hal_ev_handsfree_volume *ev = buf;
  71. if (cbs->volume_cmd_cb)
  72. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  73. cbs->volume_cmd_cb(ev->type, ev->volume,
  74. (bt_bdaddr_t *) (ev->bdaddr));
  75. #else
  76. cbs->volume_cmd_cb(ev->type, ev->volume);
  77. #endif
  78. }
  79. static void handle_dial(void *buf, uint16_t len, int fd)
  80. {
  81. struct hal_ev_handsfree_dial *ev = buf;
  82. uint16_t num_len = ev->number_len;
  83. char *number = NULL;
  84. if (len != sizeof(*ev) + num_len ||
  85. (num_len != 0 && ev->number[num_len - 1] != '\0')) {
  86. error("invalid dial event, aborting");
  87. exit(EXIT_FAILURE);
  88. }
  89. if (!cbs->dial_call_cmd_cb)
  90. return;
  91. if (ev->number_len)
  92. number = (char *) ev->number;
  93. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  94. cbs->dial_call_cmd_cb(number, (bt_bdaddr_t *) (ev->bdaddr));
  95. #else
  96. cbs->dial_call_cmd_cb(number);
  97. #endif
  98. }
  99. static void handle_dtmf(void *buf, uint16_t len, int fd)
  100. {
  101. struct hal_ev_handsfree_dtmf *ev = buf;
  102. if (cbs->dtmf_cmd_cb)
  103. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  104. cbs->dtmf_cmd_cb(ev->tone, (bt_bdaddr_t *) (ev->bdaddr));
  105. #else
  106. cbs->dtmf_cmd_cb(ev->tone);
  107. #endif
  108. }
  109. static void handle_nrec(void *buf, uint16_t len, int fd)
  110. {
  111. struct hal_ev_handsfree_nrec *ev = buf;
  112. if (cbs->nrec_cmd_cb)
  113. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  114. cbs->nrec_cmd_cb(ev->nrec, (bt_bdaddr_t *) (ev->bdaddr));
  115. #else
  116. cbs->nrec_cmd_cb(ev->nrec);
  117. #endif
  118. }
  119. static void handle_wbs(void *buf, uint16_t len, int fd)
  120. {
  121. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  122. struct hal_ev_handsfree_wbs *ev = buf;
  123. if (cbs->wbs_cb)
  124. cbs->wbs_cb(ev->wbs, (bt_bdaddr_t *) (ev->bdaddr));
  125. #endif
  126. }
  127. static void handle_chld(void *buf, uint16_t len, int fd)
  128. {
  129. struct hal_ev_handsfree_chld *ev = buf;
  130. if (cbs->chld_cmd_cb)
  131. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  132. cbs->chld_cmd_cb(ev->chld, (bt_bdaddr_t *) (ev->bdaddr));
  133. #else
  134. cbs->chld_cmd_cb(ev->chld);
  135. #endif
  136. }
  137. static void handle_cnum(void *buf, uint16_t len, int fd)
  138. {
  139. if (cbs->cnum_cmd_cb) {
  140. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  141. struct hal_ev_handsfree_cnum *ev = buf;
  142. cbs->cnum_cmd_cb((bt_bdaddr_t *) (ev->bdaddr));
  143. #else
  144. cbs->cnum_cmd_cb(NULL);
  145. #endif
  146. }
  147. }
  148. static void handle_cind(void *buf, uint16_t len, int fd)
  149. {
  150. if (cbs->cind_cmd_cb) {
  151. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  152. struct hal_ev_handsfree_cind *ev = buf;
  153. cbs->cind_cmd_cb((bt_bdaddr_t *) (ev->bdaddr));
  154. #else
  155. cbs->cind_cmd_cb();
  156. #endif
  157. }
  158. }
  159. static void handle_cops(void *buf, uint16_t len, int fd)
  160. {
  161. if (cbs->cops_cmd_cb) {
  162. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  163. struct hal_ev_handsfree_cops *ev = buf;
  164. cbs->cops_cmd_cb((bt_bdaddr_t *) (ev->bdaddr));
  165. #else
  166. cbs->cops_cmd_cb();
  167. #endif
  168. }
  169. }
  170. static void handle_clcc(void *buf, uint16_t len, int fd)
  171. {
  172. if (cbs->clcc_cmd_cb) {
  173. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  174. struct hal_ev_handsfree_clcc *ev = buf;
  175. cbs->clcc_cmd_cb((bt_bdaddr_t *) (ev->bdaddr));
  176. #else
  177. cbs->clcc_cmd_cb();
  178. #endif
  179. }
  180. }
  181. static void handle_unknown_at(void *buf, uint16_t len, int fd)
  182. {
  183. struct hal_ev_handsfree_unknown_at *ev = buf;
  184. if (len != sizeof(*ev) + ev->len ||
  185. (ev->len != 0 && ev->buf[ev->len - 1] != '\0')) {
  186. error("invalid unknown command event, aborting");
  187. exit(EXIT_FAILURE);
  188. }
  189. if (cbs->unknown_at_cmd_cb)
  190. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  191. cbs->unknown_at_cmd_cb((char *) ev->buf,
  192. (bt_bdaddr_t *) (ev->bdaddr));
  193. #else
  194. cbs->unknown_at_cmd_cb((char *) ev->buf);
  195. #endif
  196. }
  197. static void handle_hsp_key_press(void *buf, uint16_t len, int fd)
  198. {
  199. if (cbs->key_pressed_cmd_cb) {
  200. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  201. struct hal_ev_handsfree_hsp_key_press *ev = buf;
  202. cbs->key_pressed_cmd_cb((bt_bdaddr_t *) (ev->bdaddr));
  203. #else
  204. cbs->key_pressed_cmd_cb();
  205. #endif
  206. }
  207. }
  208. /*
  209. * handlers will be called from notification thread context,
  210. * index in table equals to 'opcode - HAL_MINIMUM_EVENT'
  211. */
  212. static const struct hal_ipc_handler ev_handlers[] = {
  213. /* HAL_EV_HANDSFREE_CONN_STATE */
  214. { handle_conn_state, false,
  215. sizeof(struct hal_ev_handsfree_conn_state) },
  216. /* HAL_EV_HANDSFREE_AUDIO_STATE */
  217. { handle_audio_state, false,
  218. sizeof(struct hal_ev_handsfree_audio_state) },
  219. /* HAL_EV_HANDSFREE_VR */
  220. { handle_vr_state, false, sizeof(struct hal_ev_handsfree_vr_state) },
  221. /* HAL_EV_HANDSFREE_ANSWER */
  222. { handle_answer, false, sizeof(struct hal_ev_handsfree_answer) },
  223. /* HAL_EV_HANDSFREE_HANGUP */
  224. { handle_hangup, false, sizeof(struct hal_ev_handsfree_hangup) },
  225. /* HAL_EV_HANDSFREE_VOLUME */
  226. { handle_volume, false, sizeof(struct hal_ev_handsfree_volume) },
  227. /* HAL_EV_HANDSFREE_DIAL */
  228. { handle_dial, true, sizeof(struct hal_ev_handsfree_dial) },
  229. /* HAL_EV_HANDSFREE_DTMF */
  230. { handle_dtmf, false, sizeof(struct hal_ev_handsfree_dtmf) },
  231. /* HAL_EV_HANDSFREE_NREC */
  232. { handle_nrec, false, sizeof(struct hal_ev_handsfree_nrec) },
  233. /* HAL_EV_HANDSFREE_CHLD */
  234. { handle_chld, false, sizeof(struct hal_ev_handsfree_chld) },
  235. /* HAL_EV_HANDSFREE_CNUM */
  236. { handle_cnum, false, sizeof(struct hal_ev_handsfree_cnum) },
  237. /* HAL_EV_HANDSFREE_CIND */
  238. { handle_cind, false, sizeof(struct hal_ev_handsfree_cind) },
  239. /* HAL_EV_HANDSFREE_COPS */
  240. { handle_cops, false, sizeof(struct hal_ev_handsfree_cops) },
  241. /* HAL_EV_HANDSFREE_CLCC */
  242. { handle_clcc, false, sizeof(struct hal_ev_handsfree_clcc) },
  243. /* HAL_EV_HANDSFREE_UNKNOWN_AT */
  244. { handle_unknown_at, true, sizeof(struct hal_ev_handsfree_unknown_at) },
  245. /* HAL_EV_HANDSFREE_HSP_KEY_PRESS */
  246. { handle_hsp_key_press, false,
  247. sizeof(struct hal_ev_handsfree_hsp_key_press) },
  248. /* HAL_EV_HANDSFREE_WBS */
  249. { handle_wbs, false, sizeof(struct hal_ev_handsfree_wbs) },
  250. };
  251. static uint8_t get_mode(void)
  252. {
  253. char value[PROPERTY_VALUE_MAX];
  254. if (get_config("handsfree", value, NULL) > 0) {
  255. if (!strcasecmp(value, "hfp"))
  256. return HAL_MODE_HANDSFREE_HFP;
  257. if (!strcasecmp(value, "hfp_wbs"))
  258. return HAL_MODE_HANDSFREE_HFP_WBS;
  259. }
  260. return HAL_MODE_HANDSFREE_HSP_ONLY;
  261. }
  262. static bt_status_t init_real(bthf_callbacks_t *callbacks, int max_hf_clients)
  263. {
  264. struct hal_cmd_register_module cmd;
  265. int ret;
  266. DBG("");
  267. if (interface_ready())
  268. return BT_STATUS_DONE;
  269. cbs = callbacks;
  270. hal_ipc_register(HAL_SERVICE_ID_HANDSFREE, ev_handlers,
  271. sizeof(ev_handlers)/sizeof(ev_handlers[0]));
  272. cmd.service_id = HAL_SERVICE_ID_HANDSFREE;
  273. cmd.mode = get_mode();
  274. cmd.max_clients = max_hf_clients;
  275. ret = hal_ipc_cmd(HAL_SERVICE_ID_CORE, HAL_OP_REGISTER_MODULE,
  276. sizeof(cmd), &cmd, NULL, NULL, NULL);
  277. if (ret != BT_STATUS_SUCCESS) {
  278. cbs = NULL;
  279. hal_ipc_unregister(HAL_SERVICE_ID_HANDSFREE);
  280. }
  281. return ret;
  282. }
  283. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  284. static bt_status_t init(bthf_callbacks_t *callbacks, int max_hf_clients)
  285. {
  286. return init_real(callbacks, max_hf_clients);
  287. }
  288. #else
  289. static bt_status_t init(bthf_callbacks_t *callbacks)
  290. {
  291. return init_real(callbacks, 1);
  292. }
  293. #endif
  294. static bt_status_t handsfree_connect(bt_bdaddr_t *bd_addr)
  295. {
  296. struct hal_cmd_handsfree_connect cmd;
  297. DBG("");
  298. if (!interface_ready())
  299. return BT_STATUS_NOT_READY;
  300. if (!bd_addr)
  301. return BT_STATUS_PARM_INVALID;
  302. memcpy(cmd.bdaddr, bd_addr, sizeof(cmd.bdaddr));
  303. return hal_ipc_cmd(HAL_SERVICE_ID_HANDSFREE, HAL_OP_HANDSFREE_CONNECT,
  304. sizeof(cmd), &cmd, NULL, NULL, NULL);
  305. }
  306. static bt_status_t disconnect(bt_bdaddr_t *bd_addr)
  307. {
  308. struct hal_cmd_handsfree_disconnect cmd;
  309. DBG("");
  310. if (!interface_ready())
  311. return BT_STATUS_NOT_READY;
  312. if (!bd_addr)
  313. return BT_STATUS_PARM_INVALID;
  314. memcpy(cmd.bdaddr, bd_addr, sizeof(cmd.bdaddr));
  315. return hal_ipc_cmd(HAL_SERVICE_ID_HANDSFREE,
  316. HAL_OP_HANDSFREE_DISCONNECT, sizeof(cmd), &cmd,
  317. NULL, NULL, NULL);
  318. }
  319. static bt_status_t connect_audio(bt_bdaddr_t *bd_addr)
  320. {
  321. struct hal_cmd_handsfree_connect_audio cmd;
  322. DBG("");
  323. if (!interface_ready())
  324. return BT_STATUS_NOT_READY;
  325. if (!bd_addr)
  326. return BT_STATUS_PARM_INVALID;
  327. memcpy(cmd.bdaddr, bd_addr, sizeof(cmd.bdaddr));
  328. return hal_ipc_cmd(HAL_SERVICE_ID_HANDSFREE,
  329. HAL_OP_HANDSFREE_CONNECT_AUDIO, sizeof(cmd),
  330. &cmd, NULL, NULL, NULL);
  331. }
  332. static bt_status_t disconnect_audio(bt_bdaddr_t *bd_addr)
  333. {
  334. struct hal_cmd_handsfree_disconnect_audio cmd;
  335. DBG("");
  336. if (!interface_ready())
  337. return BT_STATUS_NOT_READY;
  338. if (!bd_addr)
  339. return BT_STATUS_PARM_INVALID;
  340. memcpy(cmd.bdaddr, bd_addr, sizeof(cmd.bdaddr));
  341. return hal_ipc_cmd(HAL_SERVICE_ID_HANDSFREE,
  342. HAL_OP_HANDSFREE_DISCONNECT_AUDIO, sizeof(cmd),
  343. &cmd, NULL, NULL, NULL);
  344. }
  345. static bt_status_t start_voice_recognition_real(bt_bdaddr_t *bd_addr)
  346. {
  347. struct hal_cmd_handsfree_start_vr cmd;
  348. DBG("");
  349. if (!interface_ready())
  350. return BT_STATUS_NOT_READY;
  351. memset(&cmd, 0, sizeof(cmd));
  352. if (bd_addr)
  353. memcpy(cmd.bdaddr, bd_addr, sizeof(cmd.bdaddr));
  354. return hal_ipc_cmd(HAL_SERVICE_ID_HANDSFREE, HAL_OP_HANDSFREE_START_VR,
  355. sizeof(cmd), &cmd, NULL, NULL, NULL);
  356. }
  357. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  358. static bt_status_t start_voice_recognition(bt_bdaddr_t *bd_addr)
  359. {
  360. return start_voice_recognition_real(bd_addr);
  361. }
  362. #else
  363. static bt_status_t start_voice_recognition(void)
  364. {
  365. return start_voice_recognition_real(NULL);
  366. }
  367. #endif
  368. static bt_status_t stop_voice_recognition_real(bt_bdaddr_t *bd_addr)
  369. {
  370. struct hal_cmd_handsfree_stop_vr cmd;
  371. DBG("");
  372. if (!interface_ready())
  373. return BT_STATUS_NOT_READY;
  374. memset(&cmd, 0, sizeof(cmd));
  375. if (bd_addr)
  376. memcpy(cmd.bdaddr, bd_addr, sizeof(cmd.bdaddr));
  377. return hal_ipc_cmd(HAL_SERVICE_ID_HANDSFREE, HAL_OP_HANDSFREE_STOP_VR,
  378. sizeof(cmd), &cmd, NULL, NULL, NULL);
  379. }
  380. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  381. static bt_status_t stop_voice_recognition(bt_bdaddr_t *bd_addr)
  382. {
  383. return stop_voice_recognition_real(bd_addr);
  384. }
  385. #else
  386. static bt_status_t stop_voice_recognition(void)
  387. {
  388. return stop_voice_recognition_real(NULL);
  389. }
  390. #endif
  391. static bt_status_t volume_control_real(bthf_volume_type_t type, int volume,
  392. bt_bdaddr_t *bd_addr)
  393. {
  394. struct hal_cmd_handsfree_volume_control cmd;
  395. DBG("");
  396. if (!interface_ready())
  397. return BT_STATUS_NOT_READY;
  398. memset(&cmd, 0, sizeof(cmd));
  399. cmd.type = type;
  400. cmd.volume = volume;
  401. if (bd_addr)
  402. memcpy(cmd.bdaddr, bd_addr, sizeof(cmd.bdaddr));
  403. return hal_ipc_cmd(HAL_SERVICE_ID_HANDSFREE,
  404. HAL_OP_HANDSFREE_VOLUME_CONTROL, sizeof(cmd),
  405. &cmd, NULL, NULL, NULL);
  406. }
  407. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  408. static bt_status_t volume_control(bthf_volume_type_t type, int volume,
  409. bt_bdaddr_t *bd_addr)
  410. {
  411. return volume_control_real(type, volume, bd_addr);
  412. }
  413. #else
  414. static bt_status_t volume_control(bthf_volume_type_t type, int volume)
  415. {
  416. return volume_control_real(type, volume, NULL);
  417. }
  418. #endif
  419. static bt_status_t device_status_notification(bthf_network_state_t state,
  420. bthf_service_type_t type,
  421. int signal, int battery)
  422. {
  423. struct hal_cmd_handsfree_device_status_notif cmd;
  424. DBG("");
  425. if (!interface_ready())
  426. return BT_STATUS_NOT_READY;
  427. cmd.state = state;
  428. cmd.type = type;
  429. cmd.signal = signal;
  430. cmd.battery = battery;
  431. return hal_ipc_cmd(HAL_SERVICE_ID_HANDSFREE,
  432. HAL_OP_HANDSFREE_DEVICE_STATUS_NOTIF,
  433. sizeof(cmd), &cmd, NULL, NULL, NULL);
  434. }
  435. static bt_status_t cops_response_real(const char *cops, bt_bdaddr_t *bd_addr)
  436. {
  437. char buf[IPC_MTU];
  438. struct hal_cmd_handsfree_cops_response *cmd = (void *) buf;
  439. size_t len;
  440. DBG("");
  441. if (!interface_ready())
  442. return BT_STATUS_NOT_READY;
  443. if (!cops)
  444. return BT_STATUS_PARM_INVALID;
  445. memset(cmd, 0, sizeof(*cmd));
  446. if (bd_addr)
  447. memcpy(cmd->bdaddr, bd_addr, sizeof(cmd->bdaddr));
  448. /* Size of cmd.buf */
  449. cmd->len = strlen(cops) + 1;
  450. memcpy(cmd->buf, cops, cmd->len);
  451. len = sizeof(*cmd) + cmd->len;
  452. return hal_ipc_cmd(HAL_SERVICE_ID_HANDSFREE,
  453. HAL_OP_HANDSFREE_COPS_RESPONSE,
  454. len, cmd, NULL, NULL, NULL);
  455. }
  456. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  457. static bt_status_t cops_response(const char *cops, bt_bdaddr_t *bd_addr)
  458. {
  459. return cops_response_real(cops, bd_addr);
  460. }
  461. #else
  462. static bt_status_t cops_response(const char *cops)
  463. {
  464. return cops_response_real(cops, NULL);
  465. }
  466. #endif
  467. static bt_status_t cind_response_real(int svc, int num_active, int num_held,
  468. bthf_call_state_t state, int signal,
  469. int roam, int batt_chg,
  470. bt_bdaddr_t *bd_addr)
  471. {
  472. struct hal_cmd_handsfree_cind_response cmd;
  473. DBG("");
  474. if (!interface_ready())
  475. return BT_STATUS_NOT_READY;
  476. memset(&cmd, 0, sizeof(cmd));
  477. if (bd_addr)
  478. memcpy(cmd.bdaddr, bd_addr, sizeof(cmd.bdaddr));
  479. cmd.svc = svc;
  480. cmd.num_active = num_active;
  481. cmd.num_held = num_held;
  482. cmd.state = state;
  483. cmd.signal = signal;
  484. cmd.roam = roam;
  485. cmd.batt_chg = batt_chg;
  486. return hal_ipc_cmd(HAL_SERVICE_ID_HANDSFREE,
  487. HAL_OP_HANDSFREE_CIND_RESPONSE,
  488. sizeof(cmd), &cmd, NULL, NULL, NULL);
  489. }
  490. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  491. static bt_status_t cind_response(int svc, int num_active, int num_held,
  492. bthf_call_state_t state, int signal,
  493. int roam, int batt_chg,
  494. bt_bdaddr_t *bd_addr)
  495. {
  496. return cind_response_real(svc, num_active, num_held, state, signal,
  497. roam, batt_chg, bd_addr);
  498. }
  499. #else
  500. static bt_status_t cind_response(int svc, int num_active, int num_held,
  501. bthf_call_state_t state, int signal,
  502. int roam, int batt_chg)
  503. {
  504. return cind_response_real(svc, num_active, num_held, state, signal,
  505. roam, batt_chg, NULL);
  506. }
  507. #endif
  508. static bt_status_t formatted_at_response_real(const char *rsp,
  509. bt_bdaddr_t *bd_addr)
  510. {
  511. char buf[IPC_MTU];
  512. struct hal_cmd_handsfree_formatted_at_response *cmd = (void *) buf;
  513. size_t len;
  514. DBG("");
  515. if (!interface_ready())
  516. return BT_STATUS_NOT_READY;
  517. if (!rsp)
  518. return BT_STATUS_PARM_INVALID;
  519. memset(cmd, 0, sizeof(*cmd));
  520. if (bd_addr)
  521. memcpy(cmd->bdaddr, bd_addr, sizeof(cmd->bdaddr));
  522. cmd->len = strlen(rsp) + 1;
  523. memcpy(cmd->buf, rsp, cmd->len);
  524. len = sizeof(*cmd) + cmd->len;
  525. return hal_ipc_cmd(HAL_SERVICE_ID_HANDSFREE,
  526. HAL_OP_HANDSFREE_FORMATTED_AT_RESPONSE,
  527. len, cmd, NULL, NULL, NULL);
  528. }
  529. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  530. static bt_status_t formatted_at_response(const char *rsp, bt_bdaddr_t *bd_addr)
  531. {
  532. return formatted_at_response_real(rsp, bd_addr);
  533. }
  534. #else
  535. static bt_status_t formatted_at_response(const char *rsp)
  536. {
  537. return formatted_at_response_real(rsp, NULL);
  538. }
  539. #endif
  540. static bt_status_t at_response_real(bthf_at_response_t response, int error,
  541. bt_bdaddr_t *bd_addr)
  542. {
  543. struct hal_cmd_handsfree_at_response cmd;
  544. DBG("");
  545. if (!interface_ready())
  546. return BT_STATUS_NOT_READY;
  547. if (bd_addr)
  548. memcpy(cmd.bdaddr, bd_addr, sizeof(cmd.bdaddr));
  549. memset(&cmd, 0, sizeof(cmd));
  550. cmd.response = response;
  551. cmd.error = error;
  552. return hal_ipc_cmd(HAL_SERVICE_ID_HANDSFREE,
  553. HAL_OP_HANDSFREE_AT_RESPONSE,
  554. sizeof(cmd), &cmd, NULL, NULL, NULL);
  555. }
  556. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  557. static bt_status_t at_response(bthf_at_response_t response, int error,
  558. bt_bdaddr_t *bd_addr)
  559. {
  560. return at_response_real(response, error, bd_addr);
  561. }
  562. #else
  563. static bt_status_t at_response(bthf_at_response_t response, int error)
  564. {
  565. return at_response_real(response, error, NULL);
  566. }
  567. #endif
  568. static bt_status_t clcc_response_real(int index, bthf_call_direction_t dir,
  569. bthf_call_state_t state,
  570. bthf_call_mode_t mode,
  571. bthf_call_mpty_type_t mpty,
  572. const char *number,
  573. bthf_call_addrtype_t type,
  574. bt_bdaddr_t *bd_addr)
  575. {
  576. char buf[IPC_MTU];
  577. struct hal_cmd_handsfree_clcc_response *cmd = (void *) buf;
  578. size_t len;
  579. DBG("");
  580. if (!interface_ready())
  581. return BT_STATUS_NOT_READY;
  582. memset(cmd, 0, sizeof(*cmd));
  583. if (bd_addr)
  584. memcpy(cmd->bdaddr, bd_addr, sizeof(cmd->bdaddr));
  585. cmd->index = index;
  586. cmd->dir = dir;
  587. cmd->state = state;
  588. cmd->mode = mode;
  589. cmd->mpty = mpty;
  590. cmd->type = type;
  591. if (number) {
  592. cmd->number_len = strlen(number) + 1;
  593. memcpy(cmd->number, number, cmd->number_len);
  594. } else {
  595. cmd->number_len = 0;
  596. }
  597. len = sizeof(*cmd) + cmd->number_len;
  598. return hal_ipc_cmd(HAL_SERVICE_ID_HANDSFREE,
  599. HAL_OP_HANDSFREE_CLCC_RESPONSE,
  600. len, cmd, NULL, NULL, NULL);
  601. }
  602. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  603. static bt_status_t clcc_response(int index, bthf_call_direction_t dir,
  604. bthf_call_state_t state,
  605. bthf_call_mode_t mode,
  606. bthf_call_mpty_type_t mpty,
  607. const char *number,
  608. bthf_call_addrtype_t type,
  609. bt_bdaddr_t *bd_addr)
  610. {
  611. return clcc_response_real(index, dir, state, mode, mpty, number, type,
  612. bd_addr);
  613. }
  614. #else
  615. static bt_status_t clcc_response(int index, bthf_call_direction_t dir,
  616. bthf_call_state_t state,
  617. bthf_call_mode_t mode,
  618. bthf_call_mpty_type_t mpty,
  619. const char *number,
  620. bthf_call_addrtype_t type)
  621. {
  622. return clcc_response_real(index, dir, state, mode, mpty, number, type,
  623. NULL);
  624. }
  625. #endif
  626. static bt_status_t phone_state_change(int num_active, int num_held,
  627. bthf_call_state_t state,
  628. const char *number,
  629. bthf_call_addrtype_t type)
  630. {
  631. char buf[IPC_MTU];
  632. struct hal_cmd_handsfree_phone_state_change *cmd = (void *) buf;
  633. size_t len;
  634. DBG("");
  635. if (!interface_ready())
  636. return BT_STATUS_NOT_READY;
  637. cmd->num_active = num_active;
  638. cmd->num_held = num_held;
  639. cmd->state = state;
  640. cmd->type = type;
  641. if (number) {
  642. cmd->number_len = strlen(number) + 1;
  643. memcpy(cmd->number, number, cmd->number_len);
  644. } else {
  645. cmd->number_len = 0;
  646. }
  647. len = sizeof(*cmd) + cmd->number_len;
  648. return hal_ipc_cmd(HAL_SERVICE_ID_HANDSFREE,
  649. HAL_OP_HANDSFREE_PHONE_STATE_CHANGE,
  650. len, cmd, NULL, NULL, NULL);
  651. }
  652. static void cleanup(void)
  653. {
  654. struct hal_cmd_unregister_module cmd;
  655. DBG("");
  656. if (!interface_ready())
  657. return;
  658. cmd.service_id = HAL_SERVICE_ID_HANDSFREE;
  659. hal_ipc_cmd(HAL_SERVICE_ID_CORE, HAL_OP_UNREGISTER_MODULE,
  660. sizeof(cmd), &cmd, NULL, NULL, NULL);
  661. hal_ipc_unregister(HAL_SERVICE_ID_HANDSFREE);
  662. cbs = NULL;
  663. }
  664. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  665. static bt_status_t configure_wbs(bt_bdaddr_t *bd_addr, bthf_wbs_config_t config)
  666. {
  667. struct hal_cmd_handsfree_configure_wbs cmd;
  668. DBG("%u", config);
  669. if (!interface_ready())
  670. return BT_STATUS_NOT_READY;
  671. if (!bd_addr)
  672. return BT_STATUS_PARM_INVALID;
  673. memcpy(cmd.bdaddr, bd_addr, sizeof(cmd.bdaddr));
  674. cmd.config = config;
  675. return hal_ipc_cmd(HAL_SERVICE_ID_HANDSFREE,
  676. HAL_OP_HANDSFREE_CONFIGURE_WBS,
  677. sizeof(cmd), &cmd, NULL, NULL, NULL);
  678. }
  679. #endif
  680. static bthf_interface_t iface = {
  681. .size = sizeof(iface),
  682. .init = init,
  683. .connect = handsfree_connect,
  684. .disconnect = disconnect,
  685. .connect_audio = connect_audio,
  686. .disconnect_audio = disconnect_audio,
  687. .start_voice_recognition = start_voice_recognition,
  688. .stop_voice_recognition = stop_voice_recognition,
  689. .volume_control = volume_control,
  690. .device_status_notification = device_status_notification,
  691. .cops_response = cops_response,
  692. .cind_response = cind_response,
  693. .formatted_at_response = formatted_at_response,
  694. .at_response = at_response,
  695. .clcc_response = clcc_response,
  696. .phone_state_change = phone_state_change,
  697. .cleanup = cleanup,
  698. #if ANDROID_VERSION >= PLATFORM_VER(5, 0, 0)
  699. .configure_wbs = configure_wbs,
  700. #endif
  701. };
  702. bthf_interface_t *bt_get_handsfree_interface(void)
  703. {
  704. return &iface;
  705. }