bt_app_sink.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666
  1. /*
  2. This example code is in the Public Domain (or CC0 licensed, at your option.)
  3. Unless required by applicable law or agreed to in writing, this
  4. software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
  5. CONDITIONS OF ANY KIND, either express or implied.
  6. */
  7. #include <stdint.h>
  8. #include <stdbool.h>
  9. #include <stdlib.h>
  10. #include <string.h>
  11. #include "esp_log.h"
  12. #include "bt_app_core.h"
  13. #include "bt_app_sink.h"
  14. #include "esp_bt.h"
  15. #include "esp_bt_main.h"
  16. #include "esp_bt_device.h"
  17. #include "esp_gap_bt_api.h"
  18. #include "esp_a2dp_api.h"
  19. #include "esp_avrc_api.h"
  20. #include "nvs.h"
  21. #include "config.h"
  22. #include "freertos/FreeRTOS.h"
  23. #include "freertos/task.h"
  24. #include "trace.h"
  25. #include "audio_controls.h"
  26. #include "sys/lock.h"
  27. #include "display.h"
  28. // AVRCP used transaction label
  29. #define APP_RC_CT_TL_GET_CAPS (0)
  30. #define APP_RC_CT_TL_GET_META_DATA (1)
  31. #define APP_RC_CT_TL_RN_TRACK_CHANGE (2)
  32. #define APP_RC_CT_TL_RN_PLAYBACK_CHANGE (3)
  33. #define APP_RC_CT_TL_RN_PLAY_POS_CHANGE (4)
  34. #define BT_AV_TAG "BT_AV"
  35. #define BT_RC_TG_TAG "RCTG"
  36. #define BT_RC_CT_TAG "RCCT"
  37. #ifndef CONFIG_BT_NAME
  38. #define CONFIG_BT_NAME "ESP32-BT"
  39. #endif
  40. /* event for handler "bt_av_hdl_stack_up */
  41. enum {
  42. BT_APP_EVT_STACK_UP = 0,
  43. };
  44. char * bt_name = NULL;
  45. static bool (*bt_app_a2d_cmd_cb)(bt_sink_cmd_t cmd, ...);
  46. static void (*bt_app_a2d_data_cb)(const uint8_t *data, uint32_t len);
  47. /* handler for bluetooth stack enabled events */
  48. static void bt_av_hdl_stack_evt(uint16_t event, void *p_param);
  49. /* a2dp event handler */
  50. static void bt_av_hdl_a2d_evt(uint16_t event, void *p_param);
  51. /* avrc CT event handler */
  52. static void bt_av_hdl_avrc_ct_evt(uint16_t event, void *p_param);
  53. /* avrc TG event handler */
  54. static void bt_av_hdl_avrc_tg_evt(uint16_t event, void *p_param);
  55. static void volume_set_by_local_host(uint8_t volume);
  56. static esp_a2d_audio_state_t s_audio_state = ESP_A2D_AUDIO_STATE_STOPPED;
  57. static const char *s_a2d_conn_state_str[] = {"Disconnected", "Connecting", "Connected", "Disconnecting"};
  58. static const char *s_a2d_audio_state_str[] = {"Suspended", "Stopped", "Started"};
  59. static esp_avrc_rn_evt_cap_mask_t s_avrc_peer_rn_cap;
  60. static _lock_t s_volume_lock;
  61. static uint8_t s_volume = 0;
  62. static bool s_volume_notify;
  63. static esp_avrc_playback_stat_t s_play_status = ESP_AVRC_PLAYBACK_STOPPED;
  64. static uint8_t s_remote_bda[6];
  65. static int tl;
  66. static bt_cmd_vcb_t cmd_handler_chain;
  67. static char *s_artist, *s_album, *s_title;
  68. static void bt_volume_up(void) {
  69. // volume UP/DOWN buttons are not supported by iPhone/Android
  70. volume_set_by_local_host(s_volume < 127-3 ? s_volume + 3 : 127);
  71. (*bt_app_a2d_cmd_cb)(BT_SINK_VOLUME, s_volume);
  72. ESP_LOGI(BT_AV_TAG, "BT volume up %u", s_volume);
  73. }
  74. static void bt_volume_down(void) {
  75. // volume UP/DOWN buttons are not supported by iPhone/Android
  76. volume_set_by_local_host(s_volume > 3 ? s_volume - 3 : 0);
  77. (*bt_app_a2d_cmd_cb)(BT_SINK_VOLUME, s_volume);
  78. }
  79. static void bt_toggle(void) {
  80. if (s_play_status != ESP_AVRC_PLAYBACK_PLAYING) esp_avrc_ct_send_passthrough_cmd(tl++, ESP_AVRC_PT_CMD_PLAY, ESP_AVRC_PT_CMD_STATE_PRESSED);
  81. else esp_avrc_ct_send_passthrough_cmd(tl++ & 0x0f, ESP_AVRC_PT_CMD_STOP, ESP_AVRC_PT_CMD_STATE_PRESSED);
  82. //s_audio_state = ESP_A2D_AUDIO_STATE_STOPPED;
  83. }
  84. static void bt_play(void) {
  85. esp_avrc_ct_send_passthrough_cmd(tl++ & 0x0f, ESP_AVRC_PT_CMD_PLAY, ESP_AVRC_PT_CMD_STATE_PRESSED);
  86. }
  87. static void bt_pause(void) {
  88. esp_avrc_ct_send_passthrough_cmd(tl++ & 0x0f, ESP_AVRC_PT_CMD_PAUSE, ESP_AVRC_PT_CMD_STATE_PRESSED);
  89. }
  90. static void bt_stop(void) {
  91. esp_avrc_ct_send_passthrough_cmd(tl++ & 0x0f, ESP_AVRC_PT_CMD_STOP, ESP_AVRC_PT_CMD_STATE_PRESSED);
  92. }
  93. static void bt_prev(void) {
  94. esp_avrc_ct_send_passthrough_cmd(tl++ & 0x0f, ESP_AVRC_PT_CMD_BACKWARD, ESP_AVRC_PT_CMD_STATE_PRESSED);
  95. }
  96. static void bt_next(void) {
  97. esp_avrc_ct_send_passthrough_cmd(tl++ & 0x0f, ESP_AVRC_PT_CMD_FORWARD, ESP_AVRC_PT_CMD_STATE_PRESSED);
  98. }
  99. const static actrls_t controls = {
  100. bt_volume_up, bt_volume_down, // volume up, volume down
  101. bt_toggle, bt_play, // toggle, play
  102. bt_pause, bt_stop, // pause, stop
  103. NULL, NULL, // rew, fwd
  104. bt_prev, bt_next, // prev, next
  105. };
  106. /* taking/giving audio system's control */
  107. void bt_master(bool on) {
  108. if (on) actrls_set(controls, NULL);
  109. else actrls_unset();
  110. }
  111. /* disconnection */
  112. void bt_disconnect(void) {
  113. displayer_control(DISPLAYER_SHUTDOWN);
  114. esp_avrc_ct_send_passthrough_cmd(tl++ & 0x0f, ESP_AVRC_PT_CMD_STOP, ESP_AVRC_PT_CMD_STATE_PRESSED);
  115. esp_a2d_sink_disconnect(s_remote_bda);
  116. actrls_unset();
  117. ESP_LOGI(BT_AV_TAG, "forced disconnection");
  118. }
  119. /* command handler */
  120. static bool cmd_handler(bt_sink_cmd_t cmd, ...) {
  121. bool chain = true, res = true;
  122. va_list args;
  123. va_start(args, cmd);
  124. switch(cmd) {
  125. case BT_SINK_CONNECTED:
  126. displayer_control(DISPLAYER_ACTIVATE, "BLUETOOTH");
  127. break;
  128. case BT_SINK_PLAY:
  129. displayer_control(DISPLAYER_TIMER_RESUME);
  130. break;
  131. case BT_SINK_PAUSE:
  132. displayer_control(DISPLAYER_TIMER_PAUSE);
  133. break;
  134. case BT_SINK_STOP:
  135. displayer_control(DISPLAYER_DISABLE);
  136. break;
  137. case BT_SINK_DISCONNECTED:
  138. displayer_control(DISPLAYER_DISABLE);
  139. break;
  140. case BT_SINK_METADATA: {
  141. char *artist = va_arg(args, char*), *album = va_arg(args, char*), *title = va_arg(args, char*);
  142. displayer_metadata(artist, album, title);
  143. chain = false;
  144. break;
  145. }
  146. case BT_SINK_PROGRESS: {
  147. uint32_t elapsed = va_arg(args, uint32_t), duration = va_arg(args, uint32_t);
  148. displayer_timer(DISPLAYER_ELAPSED, elapsed, duration);
  149. chain = false;
  150. break;
  151. }
  152. default:
  153. break;
  154. }
  155. if (chain) res = cmd_handler_chain(cmd, args);
  156. va_end(args);
  157. return res;
  158. }
  159. /* callback for A2DP sink */
  160. void bt_app_a2d_cb(esp_a2d_cb_event_t event, esp_a2d_cb_param_t *param)
  161. {
  162. switch (event) {
  163. case ESP_A2D_CONNECTION_STATE_EVT:
  164. case ESP_A2D_AUDIO_STATE_EVT:
  165. case ESP_A2D_AUDIO_CFG_EVT: {
  166. bt_app_work_dispatch(bt_av_hdl_a2d_evt, event, param, sizeof(esp_a2d_cb_param_t), NULL);
  167. break;
  168. }
  169. default:
  170. ESP_LOGE(BT_AV_TAG, "Invalid A2DP event: %d", event);
  171. break;
  172. }
  173. }
  174. void bt_app_alloc_meta_buffer(esp_avrc_ct_cb_param_t *param)
  175. {
  176. esp_avrc_ct_cb_param_t *rc = (esp_avrc_ct_cb_param_t *)(param);
  177. uint8_t *attr_text = (uint8_t *) malloc (rc->meta_rsp.attr_length + 1);
  178. memcpy(attr_text, rc->meta_rsp.attr_text, rc->meta_rsp.attr_length);
  179. attr_text[rc->meta_rsp.attr_length] = 0;
  180. rc->meta_rsp.attr_text = attr_text;
  181. }
  182. void bt_app_rc_ct_cb(esp_avrc_ct_cb_event_t event, esp_avrc_ct_cb_param_t *param)
  183. {
  184. switch (event) {
  185. case ESP_AVRC_CT_METADATA_RSP_EVT:
  186. bt_app_alloc_meta_buffer(param);
  187. /* fall through */
  188. case ESP_AVRC_CT_CONNECTION_STATE_EVT:
  189. case ESP_AVRC_CT_PASSTHROUGH_RSP_EVT:
  190. case ESP_AVRC_CT_CHANGE_NOTIFY_EVT:
  191. case ESP_AVRC_CT_REMOTE_FEATURES_EVT:
  192. case ESP_AVRC_CT_GET_RN_CAPABILITIES_RSP_EVT: {
  193. bt_app_work_dispatch(bt_av_hdl_avrc_ct_evt, event, param, sizeof(esp_avrc_ct_cb_param_t), NULL);
  194. break;
  195. }
  196. default:
  197. ESP_LOGE(BT_RC_CT_TAG, "Invalid AVRC event: %d", event);
  198. break;
  199. }
  200. }
  201. void bt_app_rc_tg_cb(esp_avrc_tg_cb_event_t event, esp_avrc_tg_cb_param_t *param)
  202. {
  203. switch (event) {
  204. case ESP_AVRC_TG_CONNECTION_STATE_EVT:
  205. case ESP_AVRC_TG_REMOTE_FEATURES_EVT:
  206. case ESP_AVRC_TG_PASSTHROUGH_CMD_EVT:
  207. case ESP_AVRC_TG_SET_ABSOLUTE_VOLUME_CMD_EVT:
  208. case ESP_AVRC_TG_REGISTER_NOTIFICATION_EVT:
  209. bt_app_work_dispatch(bt_av_hdl_avrc_tg_evt, event, param, sizeof(esp_avrc_tg_cb_param_t), NULL);
  210. break;
  211. default:
  212. ESP_LOGE(BT_RC_TG_TAG, "Invalid AVRC event: %d", event);
  213. break;
  214. }
  215. }
  216. static void bt_av_hdl_a2d_evt(uint16_t event, void *p_param)
  217. {
  218. ESP_LOGD(BT_AV_TAG, "%s evt %d", __func__, event);
  219. esp_a2d_cb_param_t *a2d = NULL;
  220. switch (event) {
  221. case ESP_A2D_CONNECTION_STATE_EVT: {
  222. a2d = (esp_a2d_cb_param_t *)(p_param);
  223. uint8_t *bda = a2d->conn_stat.remote_bda;
  224. ESP_LOGI(BT_AV_TAG, "A2DP connection state: %s, [%02x:%02x:%02x:%02x:%02x:%02x]",
  225. s_a2d_conn_state_str[a2d->conn_stat.state], bda[0], bda[1], bda[2], bda[3], bda[4], bda[5]);
  226. if (a2d->conn_stat.state == ESP_A2D_CONNECTION_STATE_DISCONNECTED) {
  227. esp_bt_gap_set_scan_mode(ESP_BT_CONNECTABLE, ESP_BT_GENERAL_DISCOVERABLE);
  228. (*bt_app_a2d_cmd_cb)(BT_SINK_DISCONNECTED);
  229. actrls_unset();
  230. } else if (a2d->conn_stat.state == ESP_A2D_CONNECTION_STATE_CONNECTED){
  231. memcpy(s_remote_bda, bda, 6);
  232. esp_bt_gap_set_scan_mode(ESP_BT_NON_CONNECTABLE, ESP_BT_NON_DISCOVERABLE);
  233. if (!(*bt_app_a2d_cmd_cb)(BT_SINK_CONNECTED)){
  234. esp_avrc_ct_send_passthrough_cmd(tl++ & 0x0f, ESP_AVRC_PT_CMD_STOP, ESP_AVRC_PT_CMD_STATE_PRESSED);
  235. esp_a2d_sink_disconnect(s_remote_bda);
  236. }
  237. }
  238. break;
  239. }
  240. case ESP_A2D_AUDIO_STATE_EVT: {
  241. a2d = (esp_a2d_cb_param_t *)(p_param);
  242. ESP_LOGI(BT_AV_TAG, "A2DP audio state: %s", s_a2d_audio_state_str[a2d->audio_stat.state]);
  243. s_audio_state = a2d->audio_stat.state;
  244. if (ESP_A2D_AUDIO_STATE_STARTED == a2d->audio_stat.state) {
  245. (*bt_app_a2d_cmd_cb)(BT_SINK_PLAY);
  246. } else if (ESP_A2D_AUDIO_STATE_STOPPED == a2d->audio_stat.state ||
  247. ESP_A2D_AUDIO_STATE_REMOTE_SUSPEND == a2d->audio_stat.state) {
  248. (*bt_app_a2d_cmd_cb)(BT_SINK_STOP);
  249. }
  250. break;
  251. }
  252. case ESP_A2D_AUDIO_CFG_EVT: {
  253. a2d = (esp_a2d_cb_param_t *)(p_param);
  254. ESP_LOGI(BT_AV_TAG, "A2DP audio stream configuration, codec type %d", a2d->audio_cfg.mcc.type);
  255. // for now only SBC stream is supported
  256. if (a2d->audio_cfg.mcc.type == ESP_A2D_MCT_SBC) {
  257. int sample_rate = 16000;
  258. char oct0 = a2d->audio_cfg.mcc.cie.sbc[0];
  259. if (oct0 & (0x01 << 6)) {
  260. sample_rate = 32000;
  261. } else if (oct0 & (0x01 << 5)) {
  262. sample_rate = 44100;
  263. } else if (oct0 & (0x01 << 4)) {
  264. sample_rate = 48000;
  265. }
  266. (*bt_app_a2d_cmd_cb)(BT_SINK_RATE, sample_rate);
  267. ESP_LOGI(BT_AV_TAG, "Configure audio player %x-%x-%x-%x",
  268. a2d->audio_cfg.mcc.cie.sbc[0],
  269. a2d->audio_cfg.mcc.cie.sbc[1],
  270. a2d->audio_cfg.mcc.cie.sbc[2],
  271. a2d->audio_cfg.mcc.cie.sbc[3]);
  272. ESP_LOGI(BT_AV_TAG, "Audio player configured, sample rate=%d", sample_rate);
  273. }
  274. break;
  275. }
  276. default:
  277. ESP_LOGE(BT_AV_TAG, "%s unhandled evt %d", __func__, event);
  278. break;
  279. }
  280. }
  281. static void bt_av_new_track(void)
  282. {
  283. // request metadata
  284. uint8_t attr_mask = ESP_AVRC_MD_ATTR_TITLE | ESP_AVRC_MD_ATTR_ARTIST | ESP_AVRC_MD_ATTR_ALBUM;
  285. esp_avrc_ct_send_metadata_cmd(APP_RC_CT_TL_GET_META_DATA, attr_mask);
  286. // register notification if peer support the event_id
  287. if (esp_avrc_rn_evt_bit_mask_operation(ESP_AVRC_BIT_MASK_OP_TEST, &s_avrc_peer_rn_cap,
  288. ESP_AVRC_RN_TRACK_CHANGE)) {
  289. esp_avrc_ct_send_register_notification_cmd(APP_RC_CT_TL_RN_TRACK_CHANGE, ESP_AVRC_RN_TRACK_CHANGE, 0);
  290. }
  291. }
  292. static void bt_av_playback_changed(void)
  293. {
  294. if (esp_avrc_rn_evt_bit_mask_operation(ESP_AVRC_BIT_MASK_OP_TEST, &s_avrc_peer_rn_cap,
  295. ESP_AVRC_RN_PLAY_STATUS_CHANGE)) {
  296. esp_avrc_ct_send_register_notification_cmd(APP_RC_CT_TL_RN_PLAYBACK_CHANGE, ESP_AVRC_RN_PLAY_STATUS_CHANGE, 0);
  297. }
  298. }
  299. static void bt_av_play_pos_changed(void)
  300. {
  301. if (esp_avrc_rn_evt_bit_mask_operation(ESP_AVRC_BIT_MASK_OP_TEST, &s_avrc_peer_rn_cap,
  302. ESP_AVRC_RN_PLAY_POS_CHANGED)) {
  303. esp_avrc_ct_send_register_notification_cmd(APP_RC_CT_TL_RN_PLAY_POS_CHANGE, ESP_AVRC_RN_PLAY_POS_CHANGED, 10);
  304. }
  305. }
  306. void bt_av_notify_evt_handler(uint8_t event_id, esp_avrc_rn_param_t *event_parameter)
  307. {
  308. switch (event_id) {
  309. case ESP_AVRC_RN_TRACK_CHANGE:
  310. bt_av_new_track();
  311. (*bt_app_a2d_cmd_cb)(BT_SINK_PROGRESS, 0, 0);
  312. break;
  313. case ESP_AVRC_RN_PLAY_STATUS_CHANGE:
  314. ESP_LOGI(BT_AV_TAG, "Playback status changed: 0x%x", event_parameter->playback);
  315. s_play_status = event_parameter->playback;
  316. bt_av_playback_changed();
  317. break;
  318. case ESP_AVRC_RN_PLAY_POS_CHANGED:
  319. ESP_LOGI(BT_AV_TAG, "Play position changed: %d-ms", event_parameter->play_pos);
  320. bt_av_play_pos_changed();
  321. break;
  322. }
  323. }
  324. static void bt_av_hdl_avrc_ct_evt(uint16_t event, void *p_param)
  325. {
  326. ESP_LOGD(BT_RC_CT_TAG, "%s evt %d", __func__, event);
  327. esp_avrc_ct_cb_param_t *rc = (esp_avrc_ct_cb_param_t *)(p_param);
  328. switch (event) {
  329. case ESP_AVRC_CT_CONNECTION_STATE_EVT: {
  330. uint8_t *bda = rc->conn_stat.remote_bda;
  331. ESP_LOGI(BT_RC_CT_TAG, "AVRC conn_state evt: state %d, [%02x:%02x:%02x:%02x:%02x:%02x]",
  332. rc->conn_stat.connected, bda[0], bda[1], bda[2], bda[3], bda[4], bda[5]);
  333. if (rc->conn_stat.connected) {
  334. // get remote supported event_ids of peer AVRCP Target
  335. esp_avrc_ct_send_get_rn_capabilities_cmd(APP_RC_CT_TL_GET_CAPS);
  336. } else {
  337. // clear peer notification capability record
  338. s_avrc_peer_rn_cap.bits = 0;
  339. }
  340. break;
  341. }
  342. case ESP_AVRC_CT_PASSTHROUGH_RSP_EVT: {
  343. ESP_LOGI(BT_RC_CT_TAG, "AVRC passthrough rsp: key_code 0x%x, key_state %d", rc->psth_rsp.key_code, rc->psth_rsp.key_state);
  344. break;
  345. }
  346. case ESP_AVRC_CT_METADATA_RSP_EVT: {
  347. char **p = NULL;
  348. ESP_LOGI(BT_RC_CT_TAG, "AVRC metadata rsp: attribute id 0x%x, %s", rc->meta_rsp.attr_id, rc->meta_rsp.attr_text);
  349. if (rc->meta_rsp.attr_id == 0x01) p = &s_title;
  350. else if (rc->meta_rsp.attr_id == 0x02) p = &s_artist;
  351. else if (rc->meta_rsp.attr_id == 0x04) p = &s_album;
  352. // not very pretty, but this is bluetooth anyway
  353. if (p) {
  354. if (*p) free(*p);
  355. *p = strdup((char*) rc->meta_rsp.attr_text);
  356. if (s_artist && s_album && s_title) {
  357. (*bt_app_a2d_cmd_cb)(BT_SINK_METADATA, s_artist, s_album, s_title);
  358. free(s_artist); free(s_album); free(s_title);
  359. s_artist = s_album = s_title = NULL;
  360. }
  361. }
  362. free(rc->meta_rsp.attr_text);
  363. break;
  364. }
  365. case ESP_AVRC_CT_CHANGE_NOTIFY_EVT: {
  366. ESP_LOGI(BT_RC_CT_TAG, "AVRC event notification: %d", rc->change_ntf.event_id);
  367. bt_av_notify_evt_handler(rc->change_ntf.event_id, &rc->change_ntf.event_parameter);
  368. break;
  369. }
  370. case ESP_AVRC_CT_REMOTE_FEATURES_EVT: {
  371. ESP_LOGI(BT_RC_CT_TAG, "AVRC remote features %x, TG features %x", rc->rmt_feats.feat_mask, rc->rmt_feats.tg_feat_flag);
  372. break;
  373. }
  374. case ESP_AVRC_CT_GET_RN_CAPABILITIES_RSP_EVT: {
  375. ESP_LOGI(BT_RC_CT_TAG, "remote rn_cap: count %d, bitmask 0x%x", rc->get_rn_caps_rsp.cap_count,
  376. rc->get_rn_caps_rsp.evt_set.bits);
  377. s_avrc_peer_rn_cap.bits = rc->get_rn_caps_rsp.evt_set.bits;
  378. bt_av_new_track();
  379. bt_av_playback_changed();
  380. bt_av_play_pos_changed();
  381. break;
  382. }
  383. default:
  384. ESP_LOGE(BT_RC_CT_TAG, "%s unhandled evt %d", __func__, event);
  385. break;
  386. }
  387. }
  388. static void volume_set_by_controller(uint8_t volume)
  389. {
  390. ESP_LOGI(BT_RC_TG_TAG, "Volume is set by remote controller %d%%\n", (uint32_t)volume * 100 / 0x7f);
  391. _lock_acquire(&s_volume_lock);
  392. s_volume = volume;
  393. _lock_release(&s_volume_lock);
  394. (*bt_app_a2d_cmd_cb)(BT_SINK_VOLUME, volume);
  395. }
  396. static void volume_set_by_local_host(uint8_t volume)
  397. {
  398. ESP_LOGI(BT_RC_TG_TAG, "Volume is set locally to: %d%%", (uint32_t)volume * 100 / 0x7f);
  399. _lock_acquire(&s_volume_lock);
  400. s_volume = volume;
  401. _lock_release(&s_volume_lock);
  402. if (s_volume_notify) {
  403. esp_avrc_rn_param_t rn_param;
  404. rn_param.volume = s_volume;
  405. esp_avrc_tg_send_rn_rsp(ESP_AVRC_RN_VOLUME_CHANGE, ESP_AVRC_RN_RSP_CHANGED, &rn_param);
  406. s_volume_notify = false;
  407. }
  408. }
  409. static void bt_av_hdl_avrc_tg_evt(uint16_t event, void *p_param)
  410. {
  411. ESP_LOGD(BT_RC_TG_TAG, "%s evt %d", __func__, event);
  412. esp_avrc_tg_cb_param_t *rc = (esp_avrc_tg_cb_param_t *)(p_param);
  413. switch (event) {
  414. case ESP_AVRC_TG_CONNECTION_STATE_EVT: {
  415. uint8_t *bda = rc->conn_stat.remote_bda;
  416. ESP_LOGI(BT_RC_TG_TAG, "AVRC conn_state evt: state %d, [%02x:%02x:%02x:%02x:%02x:%02x]",
  417. rc->conn_stat.connected, bda[0], bda[1], bda[2], bda[3], bda[4], bda[5]);
  418. break;
  419. }
  420. case ESP_AVRC_TG_PASSTHROUGH_CMD_EVT: {
  421. ESP_LOGI(BT_RC_TG_TAG, "AVRC passthrough cmd: key_code 0x%x, key_state %d", rc->psth_cmd.key_code, rc->psth_cmd.key_state);
  422. break;
  423. }
  424. case ESP_AVRC_TG_SET_ABSOLUTE_VOLUME_CMD_EVT: {
  425. ESP_LOGI(BT_RC_TG_TAG, "AVRC set absolute volume: %d%%", (int)rc->set_abs_vol.volume * 100/ 0x7f);
  426. volume_set_by_controller(rc->set_abs_vol.volume);
  427. break;
  428. }
  429. case ESP_AVRC_TG_REGISTER_NOTIFICATION_EVT: {
  430. ESP_LOGI(BT_RC_TG_TAG, "AVRC register event notification: %d, param: 0x%x", rc->reg_ntf.event_id, rc->reg_ntf.event_parameter);
  431. if (rc->reg_ntf.event_id == ESP_AVRC_RN_VOLUME_CHANGE) {
  432. s_volume_notify = true;
  433. esp_avrc_rn_param_t rn_param;
  434. rn_param.volume = s_volume;
  435. esp_avrc_tg_send_rn_rsp(ESP_AVRC_RN_VOLUME_CHANGE, ESP_AVRC_RN_RSP_INTERIM, &rn_param);
  436. }
  437. break;
  438. }
  439. case ESP_AVRC_TG_REMOTE_FEATURES_EVT: {
  440. ESP_LOGI(BT_RC_TG_TAG, "AVRC remote features %x, CT features %x", rc->rmt_feats.feat_mask, rc->rmt_feats.ct_feat_flag);
  441. break;
  442. }
  443. default:
  444. ESP_LOGE(BT_RC_TG_TAG, "%s unhandled evt %d", __func__, event);
  445. break;
  446. }
  447. }
  448. void bt_sink_init(bt_cmd_vcb_t cmd_cb, bt_data_cb_t data_cb)
  449. {
  450. esp_err_t err;
  451. bt_app_a2d_cmd_cb = cmd_handler;
  452. cmd_handler_chain = cmd_cb;
  453. bt_app_a2d_data_cb = data_cb;
  454. ESP_ERROR_CHECK(esp_bt_controller_mem_release(ESP_BT_MODE_BLE));
  455. esp_bt_controller_config_t bt_cfg = BT_CONTROLLER_INIT_CONFIG_DEFAULT();
  456. if ((err = esp_bt_controller_init(&bt_cfg)) != ESP_OK) {
  457. ESP_LOGE(BT_AV_TAG, "%s initialize controller failed: %s\n", __func__, esp_err_to_name(err));
  458. return;
  459. }
  460. if ((err = esp_bt_controller_enable(ESP_BT_MODE_CLASSIC_BT)) != ESP_OK) {
  461. ESP_LOGE(BT_AV_TAG, "%s enable controller failed: %s\n", __func__, esp_err_to_name(err));
  462. return;
  463. }
  464. if ((err = esp_bluedroid_init()) != ESP_OK) {
  465. ESP_LOGE(BT_AV_TAG, "%s initialize bluedroid failed: %s\n", __func__, esp_err_to_name(err));
  466. return;
  467. }
  468. if ((err = esp_bluedroid_enable()) != ESP_OK) {
  469. ESP_LOGE(BT_AV_TAG, "%s enable bluedroid failed: %s\n", __func__, esp_err_to_name(err));
  470. return;
  471. }
  472. /* create application task */
  473. bt_app_task_start_up();
  474. /* Bluetooth device name, connection mode and profile set up */
  475. bt_app_work_dispatch(bt_av_hdl_stack_evt, BT_APP_EVT_STACK_UP, NULL, 0, NULL);
  476. #if (CONFIG_BT_SSP_ENABLED == true)
  477. /* Set default parameters for Secure Simple Pairing */
  478. esp_bt_sp_param_t param_type = ESP_BT_SP_IOCAP_MODE;
  479. esp_bt_io_cap_t iocap = ESP_BT_IO_CAP_IO;
  480. esp_bt_gap_set_security_param(param_type, &iocap, sizeof(uint8_t));
  481. #endif
  482. /*
  483. * Set default parameters for Legacy Pairing
  484. */
  485. esp_bt_pin_type_t pin_type = ESP_BT_PIN_TYPE_FIXED;
  486. char * pin_code = config_alloc_get_default(NVS_TYPE_STR, "bt_sink_pin", STR(CONFIG_BT_SINK_PIN), 0);
  487. if(strlen(pin_code)>ESP_BT_PIN_CODE_LEN){
  488. ESP_LOGW(BT_AV_TAG, "BT Sink pin code [%s] too long. ", pin_code);
  489. pin_code[ESP_BT_PIN_CODE_LEN] = '\0';
  490. ESP_LOGW(BT_AV_TAG, "BT Sink pin truncated code [%s]. ", pin_code);
  491. }
  492. esp_bt_pin_code_t esp_pin_code;
  493. bool bError=false;
  494. memset(esp_pin_code, 0x00, sizeof(esp_pin_code) );
  495. ESP_LOGW(BT_AV_TAG, "BT Sink pin code is: [%s] ", pin_code);
  496. for(int i=0;i<strlen(pin_code);i++){
  497. if(pin_code[i] < '0' || pin_code[i] > '9' ) {
  498. ESP_LOGE(BT_AV_TAG,"Invalid number found in sequence");
  499. bError=true;
  500. }
  501. esp_pin_code[i]= pin_code[i];
  502. }
  503. if(bError){
  504. esp_pin_code[0]='1';
  505. esp_pin_code[1]='2';
  506. esp_pin_code[2]='3';
  507. esp_pin_code[3]='4';
  508. }
  509. esp_bt_gap_set_pin(pin_type, strlen(pin_code), esp_pin_code);
  510. }
  511. void bt_sink_deinit(void)
  512. {
  513. /* this still does not work, can't figure out how to stop properly this BT stack */
  514. bt_app_task_shut_down();
  515. ESP_LOGI(BT_AV_TAG, "bt_app_task shutdown successfully");
  516. if (esp_bluedroid_disable() != ESP_OK) return;
  517. ESP_LOGI(BT_AV_TAG, "esp_bluedroid_disable called successfully");
  518. if (esp_bluedroid_deinit() != ESP_OK) return;
  519. ESP_LOGI(BT_AV_TAG, "esp_bluedroid_deinit called successfully");
  520. if (esp_bt_controller_disable() != ESP_OK) return;
  521. ESP_LOGI(BT_AV_TAG, "esp_bt_controller_disable called successfully");
  522. if (esp_bt_controller_deinit() != ESP_OK) return;
  523. ESP_LOGI(BT_AV_TAG, "bt stopped successfully");
  524. }
  525. static void bt_app_gap_cb(esp_bt_gap_cb_event_t event, esp_bt_gap_cb_param_t *param)
  526. {
  527. switch (event) {
  528. case ESP_BT_GAP_AUTH_CMPL_EVT: {
  529. if (param->auth_cmpl.stat == ESP_BT_STATUS_SUCCESS) {
  530. ESP_LOGI(BT_AV_TAG, "authentication success: %s", param->auth_cmpl.device_name);
  531. esp_log_buffer_hex(BT_AV_TAG, param->auth_cmpl.bda, ESP_BD_ADDR_LEN);
  532. } else {
  533. ESP_LOGE(BT_AV_TAG, "authentication failed, status:%d", param->auth_cmpl.stat);
  534. }
  535. break;
  536. }
  537. #if (CONFIG_BT_SSP_ENABLED == true)
  538. case ESP_BT_GAP_CFM_REQ_EVT:
  539. ESP_LOGI(BT_AV_TAG, "ESP_BT_GAP_CFM_REQ_EVT Please compare the numeric value: %d", param->cfm_req.num_val);
  540. esp_bt_gap_ssp_confirm_reply(param->cfm_req.bda, true);
  541. break;
  542. case ESP_BT_GAP_KEY_NOTIF_EVT:
  543. ESP_LOGI(BT_AV_TAG, "ESP_BT_GAP_KEY_NOTIF_EVT passkey:%d", param->key_notif.passkey);
  544. break;
  545. case ESP_BT_GAP_KEY_REQ_EVT:
  546. ESP_LOGI(BT_AV_TAG, "ESP_BT_GAP_KEY_REQ_EVT Please enter passkey!");
  547. break;
  548. #endif
  549. default: {
  550. ESP_LOGI(BT_AV_TAG, "event: %d", event);
  551. break;
  552. }
  553. }
  554. return;
  555. }
  556. static void bt_av_hdl_stack_evt(uint16_t event, void *p_param)
  557. {
  558. ESP_LOGD(BT_AV_TAG, "%s evt %d", __func__, event);
  559. switch (event) {
  560. case BT_APP_EVT_STACK_UP: {
  561. /* set up device name */
  562. bt_name = (char * )config_alloc_get_default(NVS_TYPE_STR, "bt_name", CONFIG_BT_NAME, 0);
  563. esp_bt_dev_set_device_name(bt_name);
  564. free(bt_name);
  565. esp_bt_gap_register_callback(bt_app_gap_cb);
  566. /* initialize AVRCP controller */
  567. esp_avrc_ct_init();
  568. esp_avrc_ct_register_callback(bt_app_rc_ct_cb);
  569. /* initialize AVRCP target */
  570. assert (esp_avrc_tg_init() == ESP_OK);
  571. esp_avrc_tg_register_callback(bt_app_rc_tg_cb);
  572. esp_avrc_rn_evt_cap_mask_t evt_set = {0};
  573. esp_avrc_rn_evt_bit_mask_operation(ESP_AVRC_BIT_MASK_OP_SET, &evt_set, ESP_AVRC_RN_VOLUME_CHANGE);
  574. assert(esp_avrc_tg_set_rn_evt_cap(&evt_set) == ESP_OK);
  575. /* initialize A2DP sink */
  576. esp_a2d_register_callback(&bt_app_a2d_cb);
  577. esp_a2d_sink_register_data_callback(bt_app_a2d_data_cb);
  578. esp_a2d_sink_init();
  579. /* set discoverable and connectable mode, wait to be connected */
  580. esp_bt_gap_set_scan_mode(ESP_BT_CONNECTABLE, ESP_BT_GENERAL_DISCOVERABLE);
  581. break;
  582. }
  583. default:
  584. ESP_LOGE(BT_AV_TAG, "%s unhandled evt %d", __func__, event);
  585. break;
  586. }
  587. }