bt_manager_hfp.c 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978
  1. /*
  2. * Copyright (c) 2019 Actions Semi Co., Inc.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief bt manager a2dp profile.
  9. */
  10. #define SYS_LOG_NO_NEWLINE
  11. #define SYS_LOG_DOMAIN "bt manager"
  12. #include <os_common_api.h>
  13. #include <zephyr.h>
  14. #include <stdlib.h>
  15. #include <stdio.h>
  16. #include <string.h>
  17. #include <stream.h>
  18. #include <bt_manager.h>
  19. #include <power_manager.h>
  20. #include <app_manager.h>
  21. #include <sys_event.h>
  22. #include <mem_manager.h>
  23. #include "bt_manager_inner.h"
  24. #include <assert.h>
  25. #include "btservice_api.h"
  26. #include <alarm_manager.h>
  27. #ifdef CONFIG_MEDIA
  28. #include <audio_policy.h>
  29. #include <audio_system.h>
  30. #include <volume_manager.h>
  31. #endif
  32. #define USE_SWITCH_APP 0
  33. #if USE_SWITCH_APP
  34. #include "app_switch.h"
  35. #endif
  36. #define HFP_SIMULATE_CALL_THRESHOLD (300) /* 300ms */
  37. #define HFP_ONGOING_ALLOW_SCO_TIME (2000) /* 2s */
  38. enum {
  39. SIRI_INIT_STATE,
  40. SIRI_START_STATE,
  41. SIRI_RUNNING_STATE,
  42. };
  43. enum {
  44. HFP_APP_ALLOW_SCO = 0x1,
  45. HFP_MGR_ALLOW_SCO = 0x2,
  46. HFP_ALLOW_SCO = (HFP_APP_ALLOW_SCO | HFP_MGR_ALLOW_SCO),
  47. };
  48. struct bt_manager_hfp_info_t {
  49. uint8_t connected:1;
  50. uint8_t siri_mode:3;
  51. uint8_t only_sco:1;
  52. uint8_t allow_sco:2;
  53. uint8_t phone_num:1;
  54. uint8_t accept_call:1;
  55. uint8_t simulate_outgoing:1; /* Just like WeCha call out */
  56. uint8_t ongoing_before_connected:1;
  57. uint8_t siri_sco_connected:1;
  58. uint8_t pending_incoming:1;
  59. uint8_t hprec_battery:1;
  60. uint32_t hfp_status;
  61. uint32_t sco_connect_time;
  62. uint32_t send_atcmd_time;
  63. uint32_t outgoing_time;
  64. uint32_t ongoing_time;
  65. os_delayed_work hfp_active_source_switch_work;
  66. };
  67. static void btmgr_hfp_call_ongoing_proc(void);
  68. static struct bt_manager_hfp_info_t hfp_manager;
  69. static struct btmgr_map_time cclk_tm;
  70. static void btmgr_hfp_allow_sco(uint8_t type, bool allow)
  71. {
  72. uint8_t old_allow = hfp_manager.allow_sco;
  73. #ifdef CONFIG_BT_AUDIO_TEST
  74. hfp_manager.allow_sco = HFP_ALLOW_SCO;
  75. #else
  76. if (bt_manager_config_pts_test()) {
  77. hfp_manager.allow_sco = HFP_ALLOW_SCO;
  78. } else {
  79. if (allow) {
  80. hfp_manager.allow_sco |= type;
  81. } else {
  82. hfp_manager.allow_sco &= (~type);
  83. }
  84. }
  85. #endif
  86. SYS_LOG_INF("allow sco %d->%d (%d, %d)", old_allow, hfp_manager.allow_sco, type, allow);
  87. if ((old_allow == HFP_ALLOW_SCO) && (hfp_manager.allow_sco != HFP_ALLOW_SCO)) {
  88. btif_br_allow_sco_connect(false);
  89. } else if ((old_allow != HFP_ALLOW_SCO) && (hfp_manager.allow_sco == HFP_ALLOW_SCO)) {
  90. btif_br_allow_sco_connect(true);
  91. }
  92. }
  93. static bool btmgr_hfp_is_sco_allow(void)
  94. {
  95. return (hfp_manager.allow_sco == HFP_ALLOW_SCO) ? true : false;
  96. }
  97. static uint32_t _bt_manager_call_to_hfp_volume(uint32_t call_vol)
  98. {
  99. uint32_t hfp_vol = 0;
  100. #ifdef CONFIG_AUDIO
  101. if (call_vol == 0) {
  102. hfp_vol = 0;
  103. } else if (call_vol >= audio_policy_get_volume_level()) {
  104. hfp_vol = 0x0F;
  105. } else {
  106. hfp_vol = (call_vol * 0x10 / audio_policy_get_volume_level());
  107. if (hfp_vol == 0) {
  108. hfp_vol = 1;
  109. }
  110. }
  111. #endif
  112. return hfp_vol;
  113. }
  114. #ifdef CONFIG_VOLUME_MANAGER
  115. static uint32_t _bt_manager_hfp_to_call_volume(uint32_t hfp_vol)
  116. {
  117. uint32_t call_vol = 0;
  118. #ifdef CONFIG_AUDIO
  119. if (hfp_vol == 0) {
  120. call_vol = 0;
  121. } else if (hfp_vol >= 0x0F) {
  122. call_vol = audio_policy_get_volume_level();
  123. } else {
  124. call_vol = (hfp_vol + 1) * audio_policy_get_volume_level() / 0x10;
  125. if (call_vol == 0) {
  126. call_vol = 1;
  127. }
  128. }
  129. #endif
  130. return call_vol;
  131. }
  132. #endif
  133. int bt_manager_hfp_set_status(uint32_t state)
  134. {
  135. SYS_LOG_INF("Set hfp state 0x%x -> 0x%x", hfp_manager.hfp_status, state);
  136. hfp_manager.hfp_status = state;
  137. return 0;
  138. }
  139. int bt_manager_hfp_get_status(void)
  140. {
  141. return hfp_manager.hfp_status;
  142. }
  143. static uint8_t string_to_num(uint8_t c)
  144. {
  145. if (c >= '0' && c <= '9') {
  146. return c - '0';
  147. } else {
  148. return 0;
  149. }
  150. }
  151. //format like "20/11/30, 18:58:09"
  152. //https://m2msupport.net/m2msupport/atcclk-clock/
  153. static int _bt_hfp_set_time(uint8_t *time)
  154. {
  155. cclk_tm.tm_year = 2000 + string_to_num(time[0])*10 + string_to_num(time[1]);
  156. cclk_tm.tm_mon = string_to_num(time[3])*10 + string_to_num(time[4]);
  157. cclk_tm.tm_mday = string_to_num(time[6])*10 + string_to_num(time[7]);
  158. if (time[9] == ' ') {
  159. cclk_tm.tm_hour = string_to_num(time[10])*10 + string_to_num(time[11]);
  160. cclk_tm.tm_min = string_to_num(time[13])*10 + string_to_num(time[14]);
  161. cclk_tm.tm_sec = string_to_num(time[16])*10 + string_to_num(time[17]);
  162. } else if(time[9] >= '0' && time[9] <= '9') {
  163. cclk_tm.tm_hour = string_to_num(time[9])*10 + string_to_num(time[10]);
  164. cclk_tm.tm_min = string_to_num(time[12])*10 + string_to_num(time[13]);
  165. cclk_tm.tm_sec = string_to_num(time[15])*10 + string_to_num(time[16]);
  166. }
  167. struct app_msg msg = {0};
  168. msg.type = MSG_BT_MGR_EVENT;
  169. msg.cmd = BT_MAP_SET_TIME_EVENT;
  170. msg.ptr = &cclk_tm;
  171. send_async_msg(app_manager_get_current_app(), &msg);
  172. return 0;
  173. }
  174. static void btmgr_hfp_clcc_info_proc(uint8_t *param, int param_size)
  175. {
  176. uint32_t state;
  177. int ret;
  178. uint8_t call_state, held_single;
  179. struct btsrv_hfp_clcc_info* clcc = (struct btsrv_hfp_clcc_info*)param;
  180. size_t number_size = strlen(clcc->number);
  181. if (hfp_manager.simulate_outgoing) {
  182. SYS_LOG_INF("hfp clcc simu return");
  183. return;
  184. }
  185. ret = bt_manager_hfp_get_call_state(1, &call_state);
  186. held_single = ((call_state & 0x4) && (!(call_state & 0x10)));
  187. SYS_LOG_INF("ret %d call_state %d held_single %d.", ret, call_state, held_single);
  188. SYS_LOG_INF("hfp clcc phone num %s size %d\n", clcc->number, number_size);
  189. if (number_size) {
  190. state = bt_manager_hfp_get_status();
  191. if ((state == BT_STATUS_OUTGOING) ||
  192. (state == BT_STATUS_ONGOING && clcc->status == 0) ||
  193. (state == BT_STATUS_MULTIPARTY && (clcc->status == 0 || (held_single && clcc->status == 1))) ||
  194. (state == BT_STATUS_3WAYIN && clcc->status == 0) ||
  195. (state == BT_STATUS_INCOMING && clcc->status == 4)) {
  196. hfp_manager.phone_num = 1;
  197. bt_manager_event_notify(BT_HFP_CALL_CLCC_INFO, param, param_size);
  198. }
  199. if (clcc->status == 0) {
  200. if ((state == BT_STATUS_MULTIPARTY || state == BT_STATUS_3WAYIN) && (!hfp_manager.sco_connect_time)) {
  201. /* Sco not in device, phone operate 3way, not allow sco connect untill ongoing */
  202. } else {
  203. /* Call in active state, allow sco connect,
  204. * in outgoing/alerted state, some phone will try connect sco several times,
  205. * so in outgoing/alerted state, can't allow sco connect.
  206. */
  207. if ((state == BT_STATUS_ONGOING) &&
  208. ((os_uptime_get_32() - hfp_manager.ongoing_time) < HFP_ONGOING_ALLOW_SCO_TIME)) {
  209. /* Ongoing not too long, allow sco connect later by hfp_active_source_switch_work */
  210. } else {
  211. btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, true);
  212. }
  213. }
  214. }
  215. }
  216. }
  217. static void btmgr_hfp_connected_proc(void)
  218. {
  219. hfp_manager.connected = 1;
  220. #ifdef CONFIG_POWER
  221. if (false == hfp_manager.hprec_battery) {
  222. bt_manager_hfp_battery_report(BT_BATTERY_REPORT_INIT, 0);
  223. bt_manager_hfp_battery_report(BT_BATTERY_REPORT_VAL, power_manager_get_battery_capacity());
  224. } else {
  225. bt_manager_hfp_battery_hprec_report(power_manager_get_battery_capacity());
  226. }
  227. #endif
  228. #ifdef CONFIG_VOLUME_MANAGER
  229. bt_manager_hfp_sync_vol_to_remote(audio_system_get_stream_volume(AUDIO_STREAM_VOICE));
  230. #endif
  231. bt_manager_hfp_get_time();
  232. if (hfp_manager.ongoing_before_connected) {
  233. hfp_manager.ongoing_before_connected = 0;
  234. SYS_LOG_INF("Proc pending ongoing");
  235. btmgr_hfp_call_ongoing_proc();
  236. }
  237. }
  238. static void btmgr_hfp_disconnected_proc(void)
  239. {
  240. bt_manager_hfp_set_status(BT_STATUS_HFP_NONE);
  241. btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, false);
  242. hfp_manager.connected = 0;
  243. hfp_manager.siri_mode = SIRI_INIT_STATE;
  244. //hfp_manager.only_sco = 0; /* Clear by BTSRV_SCO_DISCONNECTED event */
  245. hfp_manager.phone_num = 0;
  246. hfp_manager.accept_call = 0;
  247. hfp_manager.simulate_outgoing = 0;
  248. hfp_manager.ongoing_before_connected = 0;
  249. hfp_manager.siri_sco_connected = 0;
  250. hfp_manager.pending_incoming = 0;
  251. hfp_manager.sco_connect_time = 0;
  252. hfp_manager.hprec_battery = 0;
  253. }
  254. static void btmgr_hft_query_curr_call(bool force)
  255. {
  256. static uint32_t pre_time;
  257. uint32_t curr_time = os_uptime_get_32();
  258. if (force) {
  259. pre_time = curr_time;
  260. bt_manager_hfp_send_at_command("AT+CLCC", 1);
  261. } else if ((curr_time - pre_time) > 600) {
  262. pre_time = curr_time;
  263. bt_manager_hfp_send_at_command("AT+CLCC", 1);
  264. }
  265. }
  266. static void btmgr_hfp_call_incoming_proc(void)
  267. {
  268. if ((bt_manager_hfp_get_status() == BT_STATUS_SIRI) ||
  269. (hfp_manager.siri_sco_connected == 1)) {
  270. /* Incoming break siri, process imcoming after siri exit and sco disconnected. */
  271. hfp_manager.pending_incoming = 1;
  272. SYS_LOG_INF("hfp incoming break siri");
  273. return;
  274. } else {
  275. hfp_manager.pending_incoming = 0;
  276. }
  277. if (bt_manager_hfp_get_status() != BT_STATUS_3WAYIN) {
  278. /* Incoming in 3wayin state, still keep in 3wayin state untill goto ongoing state */
  279. bt_manager_hfp_set_status(BT_STATUS_INCOMING);
  280. }
  281. #if USE_SWITCH_APP
  282. if (btmgr_hfp_is_sco_allow() && strcmp("tws", app_manager_get_current_app())) {
  283. app_switch("btcall", APP_SWITCH_CURR, true);
  284. app_switch_lock(1);
  285. }
  286. #else
  287. bt_manager_event_notify(BT_HFP_CALL_STATE_START, NULL, 0);
  288. #endif
  289. hfp_manager.only_sco = 0;
  290. hfp_manager.simulate_outgoing = 0;
  291. if ((bt_manager_hfp_get_status() == BT_STATUS_3WAYIN) && (!hfp_manager.sco_connect_time)) {
  292. /* 3Waycall state, soc not on device,
  293. * Not display incoming view,
  294. * not allow sco connect untill ongoing swith allow sco.
  295. */
  296. } else {
  297. /* only support local ringtone.*/
  298. // btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, true);
  299. bt_manager_event_notify(BT_HFP_CALL_INCOMING, NULL, 0);
  300. }
  301. btmgr_hft_query_curr_call(false);
  302. }
  303. static void btmgr_hfp_check_start_pending_incoming(void)
  304. {
  305. /* Some phone siri exit before sco disconnected,
  306. * some phone sco disconnected before siri exit,
  307. * Have pending incoming, siri exit, sco diconected.
  308. */
  309. if (hfp_manager.pending_incoming &&
  310. (bt_manager_hfp_get_status() == BT_STATUS_HFP_NONE) &&
  311. (!hfp_manager.siri_sco_connected)) {
  312. /* Resume incoming after siri exit, and siri sco discnnected */
  313. SYS_LOG_INF("hfp resume incoming");
  314. btmgr_hfp_call_incoming_proc();
  315. }
  316. }
  317. static void btmgr_hfp_call_outgoing_alerted_proc(uint8_t event)
  318. {
  319. if (!hfp_manager.phone_num) {
  320. /* Just for get outgoing phone number */
  321. btmgr_hft_query_curr_call(false);
  322. }
  323. bt_manager_hfp_set_status(BT_STATUS_OUTGOING);
  324. #if USE_SWITCH_APP
  325. if (btmgr_hfp_is_sco_allow() && strcmp("tws", app_manager_get_current_app())) {
  326. app_switch("btcall", APP_SWITCH_CURR, true);
  327. app_switch_lock(1);
  328. }
  329. #else
  330. bt_manager_event_notify(BT_HFP_CALL_STATE_START, NULL, 0);
  331. #endif
  332. hfp_manager.only_sco = 0;
  333. hfp_manager.simulate_outgoing = 0;
  334. if (event == BTSRV_HFP_CALL_OUTGOING) {
  335. /* Can update time in alerted, alerted too close to ongoing. */
  336. hfp_manager.outgoing_time = os_uptime_get_32();
  337. }
  338. bt_manager_event_notify(BT_HFP_CALL_OUTGOING, NULL, 0);
  339. }
  340. static void btmgr_hfp_call_ongoing_proc(void)
  341. {
  342. uint32_t pre_state = bt_manager_hfp_get_status();
  343. if (!hfp_manager.connected) {
  344. SYS_LOG_INF("Pending ongoing");
  345. hfp_manager.ongoing_before_connected = 1;
  346. return;
  347. }
  348. if (pre_state == BT_STATUS_3WAYIN || pre_state == BT_STATUS_MULTIPARTY) {
  349. /* Operate 3way in phoone
  350. * from 3wayin state to ongoing state, phone number may change, query again
  351. */
  352. btmgr_hft_query_curr_call(true);
  353. }
  354. bt_manager_hfp_set_status(BT_STATUS_ONGOING);
  355. #if USE_SWITCH_APP
  356. if (btmgr_hfp_is_sco_allow() && strcmp("tws", app_manager_get_current_app())) {
  357. app_switch("btcall", APP_SWITCH_CURR, true);
  358. app_switch_lock(1);
  359. }
  360. #else
  361. bt_manager_event_notify(BT_HFP_CALL_STATE_START, NULL, 0);
  362. #endif
  363. if ((pre_state == BT_STATUS_INCOMING) && (hfp_manager.accept_call == 0)) {
  364. /* Incoming, accepte on phone, allow sco connect 2s later. */
  365. btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, false);
  366. /* if in CONFIG_BT_AUDIO_TEST mode, not exit call view, not disconnect sco. */
  367. if (!btmgr_hfp_is_sco_allow()) {
  368. /* Sco connected or not, need notify app exit call */
  369. bt_manager_event_notify(BT_HFP_ESCO_RELEASED_EVENT, NULL, 0);
  370. if (hfp_manager.sco_connect_time) {
  371. btif_hfp_hf_switch_sound_source();
  372. }
  373. }
  374. }
  375. if ((os_uptime_get_32() - hfp_manager.outgoing_time) < HFP_SIMULATE_CALL_THRESHOLD) {
  376. hfp_manager.simulate_outgoing = 1;
  377. SYS_LOG_INF("hfp simu outgoing");
  378. }
  379. hfp_manager.only_sco = 0;
  380. hfp_manager.accept_call = 0;
  381. hfp_manager.ongoing_time = os_uptime_get_32();
  382. bt_manager_event_notify(BT_HFP_CALL_ONGOING, NULL, 0);
  383. os_delayed_work_submit(&hfp_manager.hfp_active_source_switch_work, HFP_ONGOING_ALLOW_SCO_TIME);
  384. }
  385. static void btmgr_hfp_call_3wayin_proc(void)
  386. {
  387. bt_manager_hfp_set_status(BT_STATUS_3WAYIN);
  388. hfp_manager.only_sco = 0;
  389. if (!hfp_manager.sco_connect_time) {
  390. /* Sco not in device, phone operate 3way, not allow sco connect untill ongoing */
  391. btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, false);
  392. }
  393. }
  394. static void btmgr_hfp_call_multiparty_proc(void)
  395. {
  396. uint8_t call_state;
  397. int ret;
  398. bt_manager_hfp_set_status(BT_STATUS_MULTIPARTY);
  399. hfp_manager.only_sco = 0;
  400. if (!hfp_manager.sco_connect_time) {
  401. /* Sco not in device, phone operate 3way, not allow sco connect untill ongoing or 2s later */
  402. btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, false);
  403. os_delayed_work_submit(&hfp_manager.hfp_active_source_switch_work, HFP_ONGOING_ALLOW_SCO_TIME);
  404. btmgr_hft_query_curr_call(true);
  405. } else {
  406. btmgr_hft_query_curr_call(true); /* Phone swith active and hold call, query and update phone number */
  407. }
  408. ret = bt_manager_hfp_get_call_state(1, &call_state);
  409. SYS_LOG_INF("call_state %d %d", ret, call_state);
  410. if ((ret != BTSRV_HFP_STATE_INIT) && (call_state & 0xF0)) {
  411. bt_manager_event_notify(BT_HFP_CALL_ONGOING, NULL, 0);
  412. }
  413. }
  414. static void btmgr_hfp_call_exit_proc(void)
  415. {
  416. bt_manager_hfp_set_status(BT_STATUS_HFP_NONE);
  417. btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, false);
  418. #if USE_SWITCH_APP
  419. if (!strcmp("btcall", app_manager_get_current_app())) {
  420. bt_manager_event_notify(BT_HFP_CALL_HUNGUP, NULL, 0);
  421. app_switch_unlock(1);
  422. app_switch(NULL, APP_SWITCH_LAST, false);
  423. }
  424. #else
  425. bt_manager_event_notify(BT_HFP_CALL_HUNGUP, NULL, 0);
  426. bt_manager_event_notify(BT_HFP_CALL_STATE_EXIT, NULL, 0);
  427. #endif
  428. hfp_manager.phone_num = 0;
  429. hfp_manager.accept_call = 0;
  430. hfp_manager.simulate_outgoing = 0;
  431. hfp_manager.siri_sco_connected = 0;
  432. hfp_manager.pending_incoming = 0;
  433. }
  434. /* Callback when sco connected withou incoming/outgoing,
  435. * just like some device create siri sco without hfp notify,
  436. * BTSRV_SCO_CONNECTED callback will follow BTSRV_HFP_SCO callback.
  437. */
  438. static void btmgr_hfp_sco_proc(void)
  439. {
  440. #if USE_SWITCH_APP
  441. if (btmgr_hfp_is_sco_allow() && strcmp("tws", app_manager_get_current_app())) {
  442. hfp_manager.only_sco = 1;
  443. app_switch("btcall", APP_SWITCH_CURR, true);
  444. app_switch_lock(1);
  445. }
  446. #else
  447. hfp_manager.only_sco = 1;
  448. bt_manager_event_notify(BT_HFP_CALL_STATE_START, NULL, 0);
  449. #endif
  450. if (hfp_manager.siri_mode == SIRI_START_STATE) {
  451. hfp_manager.siri_mode = SIRI_RUNNING_STATE;
  452. }
  453. if (hfp_manager.siri_mode == SIRI_RUNNING_STATE) {
  454. bt_manager_hfp_set_status(BT_STATUS_SIRI);
  455. bt_manager_event_notify(BT_HFP_CALL_SIRI_MODE, NULL, 0);
  456. }
  457. }
  458. static void btmgr_hfp_volume_change_proc(uint8_t *param)
  459. {
  460. uint32_t volume = *(uint8_t *)(param);
  461. SYS_LOG_INF("volume %d", volume);
  462. uint32_t pass_time = os_uptime_get_32() - hfp_manager.sco_connect_time;
  463. /* some phone may send volume sync event after sco connect. */
  464. /* we must drop these event to avoid volume conflict */
  465. /* To Do: do we have better solution?? */
  466. #if USE_SWITCH_APP
  467. if ((hfp_manager.sco_connect_time && (pass_time < 500))
  468. || strcmp("btcall", app_manager_get_current_app())) {
  469. SYS_LOG_INF("drop volume sync event from phone\n");
  470. return;
  471. }
  472. #else
  473. if(hfp_manager.sco_connect_time && (pass_time < 500)){
  474. SYS_LOG_INF("drop volume sync event from phone\n");
  475. return;
  476. }
  477. #endif
  478. #ifdef CONFIG_VOLUME_MANAGER
  479. uint32_t call_vol = _bt_manager_hfp_to_call_volume(volume);
  480. system_volume_sync_remote_to_device(AUDIO_STREAM_VOICE, call_vol);
  481. #endif
  482. }
  483. static void btmgr_hfp_siri_state_change_proc(uint8_t *param)
  484. {
  485. uint32_t state = *(uint8_t *)(param);
  486. SYS_LOG_INF("hfp siri change state %d", state);
  487. if (state == BTSRV_HFP_SIRI_CHANGE_ACTIVE) {
  488. /* Receive AT notify siri enable */
  489. hfp_manager.siri_mode = SIRI_RUNNING_STATE;
  490. bt_manager_hfp_set_status(BT_STATUS_SIRI);
  491. sys_event_notify(SYS_EVENT_SIRI_START);
  492. } else if (state == BTSRV_HFP_SIRI_CHANGE_START) {
  493. /* start siri cmd send OK,just play tts */
  494. btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, true);
  495. hfp_manager.siri_mode = SIRI_START_STATE;
  496. sys_event_notify(SYS_EVENT_SIRI_START);
  497. } else if (state == BTSRV_HFP_SIRI_CHANGE_DEACTIVE) {
  498. /* Receive AT notify siri disable */
  499. hfp_manager.siri_mode = SIRI_INIT_STATE;
  500. if (bt_manager_hfp_get_status() == BT_STATUS_SIRI) {
  501. bt_manager_hfp_set_status(BT_STATUS_HFP_NONE);
  502. }
  503. sys_event_notify(SYS_EVENT_SIRI_STOP);
  504. if (!hfp_manager.only_sco) {
  505. btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, false);
  506. }
  507. btmgr_hfp_check_start_pending_incoming();
  508. } else {
  509. SYS_LOG_ERR("Siri unknow change %d", state);
  510. }
  511. }
  512. static void btmgr_hfp_sco_connected_proc(void)
  513. {
  514. uint32_t state = bt_manager_hfp_get_status();
  515. SYS_LOG_INF("allow_sco %d status 0x%x", btmgr_hfp_is_sco_allow(), state);
  516. bt_manager_event_notify(BT_HFP_CALL_STATE_START, NULL, 0);
  517. hfp_manager.sco_connect_time = os_uptime_get_32();
  518. bt_manager_event_notify(BT_HFP_ESCO_ESTABLISHED_EVENT, NULL, 0);
  519. if (state == BT_STATUS_INCOMING) {
  520. btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, false);
  521. }
  522. if (state == BT_STATUS_SIRI) {
  523. hfp_manager.siri_sco_connected = 1;
  524. }
  525. }
  526. static void btmgr_hfp_sco_disconnected_proc(void)
  527. {
  528. uint32_t pre_state = bt_manager_hfp_get_status();
  529. SYS_LOG_INF("hfp only_sco %d state 0x%x", hfp_manager.only_sco, pre_state);
  530. hfp_manager.sco_connect_time = 0;
  531. bt_manager_event_notify(BT_HFP_ESCO_RELEASED_EVENT, NULL, 0);
  532. #if USE_SWITCH_APP
  533. /**hfp_manager.only_sco == 0 means sco disconnect by switch sound source, not exit btcall app */
  534. if (hfp_manager.only_sco && !strcmp("btcall", app_manager_get_current_app())) {
  535. hfp_manager.only_sco = 0;
  536. app_switch_unlock(1);
  537. app_switch(NULL, APP_SWITCH_LAST, false);
  538. }
  539. #else
  540. if (hfp_manager.only_sco) {
  541. hfp_manager.only_sco = 0;
  542. btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, false);
  543. bt_manager_hfp_set_status(BT_STATUS_HFP_NONE);
  544. bt_manager_event_notify(BT_HFP_CALL_STATE_EXIT, NULL, 0);
  545. }
  546. #endif
  547. hfp_manager.siri_mode = SIRI_INIT_STATE;
  548. hfp_manager.siri_sco_connected = 0;
  549. if (pre_state == BT_STATUS_HFP_NONE) {
  550. /* Already exit siri and disconnected sco. */
  551. btmgr_hfp_check_start_pending_incoming();
  552. }
  553. }
  554. static void _bt_manager_hfp_callback(btsrv_hfp_event_e event, uint8_t *param, int param_size)
  555. {
  556. // temp debug for testing
  557. bt_manager_hfp_dump_info();
  558. switch (event) {
  559. case BTSRV_HFP_CONNECTED:
  560. SYS_LOG_INF("hfp connected");
  561. btmgr_hfp_connected_proc();
  562. break;
  563. case BTSRV_HFP_DISCONNECTED:
  564. SYS_LOG_INF("hfp disconnected");
  565. btmgr_hfp_disconnected_proc();
  566. break;
  567. case BTSRV_HFP_PHONE_NUM:
  568. SYS_LOG_INF("hfp phone num %s size %d", param, param_size);
  569. hfp_manager.phone_num = 1;
  570. bt_manager_event_notify(BT_HFP_CALL_RING_STATR_EVENT, param, param_size);
  571. break;
  572. case BTSRV_HFP_PHONE_NUM_STOP:
  573. SYS_LOG_INF("hfp phone stop");
  574. bt_manager_event_notify(BT_HFP_CALL_RING_STOP_EVENT, NULL, 0);
  575. break;
  576. case BTSRV_HFP_PHONE_NAME:
  577. SYS_LOG_INF("hfp phone name %s size %d", param, param_size);
  578. break;
  579. case BTSRV_HFP_CCWA_PHONE_NUM:
  580. SYS_LOG_INF("hfp ccwa phone num %s size %d", param, param_size);
  581. if (hfp_manager.sco_connect_time) {
  582. /* Only notify CCWA phone number when sco connected */
  583. bt_manager_event_notify(BT_HFP_CALL_CCWA_EVENT, param, param_size);
  584. }
  585. break;
  586. case BTSRV_HFP_CLCC_INFO:
  587. SYS_LOG_INF("hfp clcc info");
  588. btmgr_hfp_clcc_info_proc(param, param_size);
  589. break;
  590. case BTSRV_HFP_CODEC_INFO:
  591. SYS_LOG_INF("codec_id %d sample_rate %d", param[0], param[1]);
  592. bt_manager_sco_set_codec_info(param[0], param[1]);
  593. break;
  594. case BTSRV_HFP_CALL_INCOMING:
  595. SYS_LOG_INF("hfp call incoming");
  596. btmgr_hfp_call_incoming_proc();
  597. break;
  598. case BTSRV_HFP_CALL_OUTGOING:
  599. case BTSRV_HFP_CALL_ALERTED:
  600. SYS_LOG_INF("hfp call %s", (event == BTSRV_HFP_CALL_OUTGOING) ? "outgoing" : "alerted");
  601. btmgr_hfp_call_outgoing_alerted_proc(event);
  602. break;
  603. case BTSRV_HFP_CALL_ONGOING:
  604. SYS_LOG_INF("hfp call ongoing\n");
  605. btmgr_hfp_call_ongoing_proc();
  606. break;
  607. case BTSRV_HFP_CALL_3WAYIN:
  608. SYS_LOG_INF("hfp 3wayin");
  609. btmgr_hfp_call_3wayin_proc();
  610. break;
  611. case BTSRV_HFP_CALL_MULTIPARTY:
  612. SYS_LOG_INF("hfp call multiparty");
  613. btmgr_hfp_call_multiparty_proc();
  614. break;
  615. case BTSRV_HFP_CALL_EXIT:
  616. SYS_LOG_INF("hfp call exit");
  617. btmgr_hfp_call_exit_proc();
  618. break;
  619. case BTSRV_HFP_SCO:
  620. SYS_LOG_INF("hfp sco");
  621. btmgr_hfp_sco_proc();
  622. break;
  623. case BTSRV_HFP_VOLUME_CHANGE:
  624. SYS_LOG_INF("hfp volume change");
  625. btmgr_hfp_volume_change_proc(param);
  626. break;
  627. case BTSRV_HFP_SIRI_STATE_CHANGE:
  628. btmgr_hfp_siri_state_change_proc(param);
  629. break;
  630. case BTSRV_SCO_CONNECTED:
  631. SYS_LOG_INF("hfp sco cnnected");
  632. btmgr_hfp_sco_connected_proc();
  633. break;
  634. case BTSRV_SCO_DISCONNECTED:
  635. SYS_LOG_INF("hfp sco discnnected");
  636. btmgr_hfp_sco_disconnected_proc();
  637. break;
  638. case BTSRV_HFP_ACTIVE_DEV_CHANGE:
  639. SYS_LOG_INF("hfp dev changed");
  640. bt_manager_event_notify(BT_HFP_ACTIVEDEV_CHANGE_EVENT, NULL, 0);
  641. break;
  642. case BTSRV_HFP_TIME_UPDATE:
  643. SYS_LOG_INF("time update %s", param);
  644. _bt_hfp_set_time(param);
  645. break;
  646. case BTSRV_HFP_SCO_REJECTED:
  647. SYS_LOG_INF("hfp sco rejected");
  648. if (!btmgr_hfp_is_sco_allow()) {
  649. btmgr_hft_query_curr_call(false);
  650. }
  651. break;
  652. case BTSRV_HFP_BATTERY_HPREC:
  653. SYS_LOG_INF("hfp battery Indicator enable.");
  654. hfp_manager.hprec_battery = true;
  655. //bt_manager_hfp_battery_hprec_report(power_manager_get_battery_capacity());
  656. break;
  657. default:
  658. break;
  659. }
  660. }
  661. int bt_manager_hfp_profile_start(void)
  662. {
  663. return btif_hfp_start(&_bt_manager_hfp_callback);
  664. }
  665. int bt_manager_hfp_profile_stop(void)
  666. {
  667. return btif_hfp_stop();
  668. }
  669. int bt_manager_hfp_get_call_state(uint8_t active_call,uint8_t *call_state)
  670. {
  671. return btif_hfp_hf_get_call_state(active_call, call_state);
  672. }
  673. int bt_manager_hfp_check_atcmd_time(void)
  674. {
  675. uint32_t pass_time = os_uptime_get_32() - hfp_manager.send_atcmd_time;
  676. if (hfp_manager.send_atcmd_time && (pass_time < 600)) {
  677. return 0;
  678. }
  679. hfp_manager.send_atcmd_time = os_uptime_get_32();
  680. return 1;
  681. }
  682. int bt_manager_hfp_dial_number(uint8_t *number)
  683. {
  684. if (!bt_manager_hfp_check_atcmd_time() ||
  685. !btif_hfp_active_phone_service_ready()) {
  686. return -1;
  687. }
  688. btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, true);
  689. return btif_hfp_hf_dial_number(number);
  690. }
  691. int bt_manager_hfp_dial_last_number(void)
  692. {
  693. if (!bt_manager_hfp_check_atcmd_time() ||
  694. !btif_hfp_active_phone_service_ready()) {
  695. return -1;
  696. }
  697. btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, true);
  698. return btif_hfp_hf_dial_last_number();
  699. }
  700. int bt_manager_hfp_dial_memory(int location)
  701. {
  702. if (!bt_manager_hfp_check_atcmd_time() ||
  703. !btif_hfp_active_phone_service_ready()) {
  704. return -1;
  705. }
  706. btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, true);
  707. return btif_hfp_hf_dial_memory(location);
  708. }
  709. int bt_manager_hfp_volume_control(uint8_t type, uint8_t volume)
  710. {
  711. #ifdef CONFIG_BT_HFP_HF_VOL_SYNC
  712. return btif_hfp_hf_volume_control(type, volume);
  713. #else
  714. return 0;
  715. #endif
  716. }
  717. int bt_manager_hfp_battery_report(uint8_t mode, uint8_t bat_val)
  718. {
  719. if (mode == BT_BATTERY_REPORT_VAL) {
  720. bat_val = bat_val / 10;
  721. if (bat_val > 9) {
  722. bat_val = 9;
  723. }
  724. }
  725. SYS_LOG_INF("mode %d , bat_val %d\n", mode, bat_val);
  726. return btif_hfp_hf_battery_report(mode, bat_val);
  727. }
  728. int bt_manager_hfp_battery_hprec_report(uint8_t bat_val)
  729. {
  730. SYS_LOG_INF("bat_val %d\n", bat_val);
  731. return btif_hfp_hf_battery_hprec_report(bat_val);
  732. }
  733. int bt_manager_hfp_accept_call(void)
  734. {
  735. SYS_LOG_INF("hfp accept call");
  736. if (bt_manager_hfp_get_status() == BT_STATUS_INCOMING) {
  737. hfp_manager.accept_call = 1;
  738. }
  739. btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, true);
  740. return btif_hfp_hf_accept_call();
  741. }
  742. int bt_manager_hfp_reject_call(void)
  743. {
  744. SYS_LOG_INF("hfp reject call");
  745. return btif_hfp_hf_reject_call();
  746. }
  747. int bt_manager_hfp_hangup_call(void)
  748. {
  749. SYS_LOG_INF("hfp hangup call");
  750. return btif_hfp_hf_hangup_call();
  751. }
  752. int bt_manager_hfp_hangup_another_call(void)
  753. {
  754. if (!bt_manager_hfp_check_atcmd_time()) {
  755. return -1;
  756. }
  757. SYS_LOG_INF("hfp hangup another call");
  758. return btif_hfp_hf_hangup_another_call();
  759. }
  760. int bt_manager_hfp_holdcur_answer_call(void)
  761. {
  762. if (!bt_manager_hfp_check_atcmd_time()) {
  763. return -1;
  764. }
  765. SYS_LOG_INF("hfp holdcur answer call");
  766. return btif_hfp_hf_holdcur_answer_call();
  767. }
  768. int bt_manager_hfp_hangupcur_answer_call(void)
  769. {
  770. if (!bt_manager_hfp_check_atcmd_time()) {
  771. return -1;
  772. }
  773. SYS_LOG_INF("hfp hangupcur answer call");
  774. return btif_hfp_hf_hangupcur_answer_call();
  775. }
  776. int bt_manager_hfp_start_siri(void)
  777. {
  778. int ret = 0;
  779. if (!bt_manager_hfp_check_atcmd_time()) {
  780. return -1;
  781. }
  782. if (hfp_manager.siri_mode != SIRI_RUNNING_STATE) {
  783. ret = btif_hfp_hf_voice_recognition_start();
  784. }
  785. return ret;
  786. }
  787. int bt_manager_hfp_stop_siri(void)
  788. {
  789. int ret = 0;
  790. if (!bt_manager_hfp_check_atcmd_time()) {
  791. return -1;
  792. }
  793. if (hfp_manager.siri_mode != SIRI_INIT_STATE) {
  794. ret = btif_hfp_hf_voice_recognition_stop();
  795. sys_event_notify(SYS_EVENT_SIRI_STOP);
  796. if (hfp_manager.only_sco == 0) {
  797. btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, false);
  798. hfp_manager.siri_mode = SIRI_INIT_STATE;
  799. }
  800. /* When sco connected, BTSRV_SCO_DISCONNECTED will callback */
  801. }
  802. return ret;
  803. }
  804. int bt_manager_hfp_switch_sound_source(void)
  805. {
  806. if (!bt_manager_hfp_check_atcmd_time()) {
  807. return -1;
  808. }
  809. return btif_hfp_hf_switch_sound_source();
  810. }
  811. int bt_manager_hfp_get_time(void)
  812. {
  813. return btif_hfp_hf_get_time();
  814. }
  815. /* Just call by app */
  816. int bt_manager_allow_sco_connect(bool allowed)
  817. {
  818. btmgr_hfp_allow_sco(HFP_APP_ALLOW_SCO, allowed);
  819. return 0;
  820. }
  821. int bt_manager_hfp_send_at_command(uint8_t *command,uint8_t active_call)
  822. {
  823. return btif_hfp_hf_send_at_command(command,active_call);
  824. }
  825. int bt_manager_hfp_sync_vol_to_remote(uint32_t call_vol)
  826. {
  827. uint32_t hfp_vol = _bt_manager_call_to_hfp_volume(call_vol);
  828. return bt_manager_hfp_volume_control(1, (uint8_t)hfp_vol);
  829. }
  830. static void bt_manager_hfp_active_source_switch(struct k_work *work)
  831. {
  832. SYS_LOG_INF("status %d num %d\n", bt_manager_hfp_get_status(), hfp_manager.phone_num);
  833. if ((bt_manager_hfp_get_status() == BT_STATUS_ONGOING) && (!btmgr_hfp_is_sco_allow())) {
  834. if (hfp_manager.phone_num) {
  835. btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, true);
  836. } else {
  837. /* Some phone, in calling state, disconnect and connect bt,
  838. * will not create sco link, so need query current call again.
  839. */
  840. btmgr_hft_query_curr_call(false);
  841. }
  842. } else if ((bt_manager_hfp_get_status() == BT_STATUS_MULTIPARTY) && (!hfp_manager.sco_connect_time)) {
  843. btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, true);
  844. }
  845. }
  846. int bt_manager_hfp_init(void)
  847. {
  848. memset(&hfp_manager, 0, sizeof(struct bt_manager_hfp_info_t));
  849. hfp_manager.allow_sco = HFP_ALLOW_SCO;
  850. btmgr_hfp_allow_sco(HFP_MGR_ALLOW_SCO, false);
  851. os_delayed_work_init(&hfp_manager.hfp_active_source_switch_work, bt_manager_hfp_active_source_switch);
  852. return 0;
  853. }
  854. void bt_manager_hfp_dump_info(void)
  855. {
  856. printk("hfp info\n");
  857. printk("\t connected %d state 0x%x siri %d sco %d %d PASO %d %d %d %d %d %d T %d\n",
  858. hfp_manager.connected, hfp_manager.hfp_status, hfp_manager.siri_mode, hfp_manager.only_sco,
  859. hfp_manager.allow_sco, hfp_manager.phone_num, hfp_manager.accept_call, hfp_manager.simulate_outgoing,
  860. hfp_manager.ongoing_before_connected, hfp_manager.pending_incoming, hfp_manager.siri_sco_connected, hfp_manager.sco_connect_time);
  861. }