bt_app_sink.c 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686
  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 "platform_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. static const char BT_AV_TAG[] = "BT_AV";
  35. static const char BT_RC_TG_TAG[] = "RCTG";
  36. static const char BT_RC_CT_TAG[] = "RCCT";
  37. #ifndef CONFIG_BT_NAME
  38. #define CONFIG_BT_NAME "ESP32-BT"
  39. #endif
  40. static char * bt_name = NULL;
  41. static bool (*bt_app_a2d_cmd_cb)(bt_sink_cmd_t cmd, ...);
  42. static void (*bt_app_a2d_data_cb)(const uint8_t *data, uint32_t len);
  43. /* handler for bluetooth stack enabled events */
  44. static void bt_av_hdl_stack_evt(uint16_t event, void *p_param);
  45. /* a2dp event handler */
  46. static void bt_av_hdl_a2d_evt(uint16_t event, void *p_param);
  47. /* avrc CT event handler */
  48. static void bt_av_hdl_avrc_ct_evt(uint16_t event, void *p_param);
  49. /* avrc TG event handler */
  50. static void bt_av_hdl_avrc_tg_evt(uint16_t event, void *p_param);
  51. static void volume_set_by_local_host(int value, bool is_step);
  52. static void bt_av_notify_evt_handler(uint8_t event_id, esp_avrc_rn_param_t *event_parameter);
  53. static const char *s_a2d_conn_state_str[] = {"Disconnected", "Connecting", "Connected", "Disconnecting"};
  54. static const char *s_a2d_audio_state_str[] = {"Suspended", "Stopped", "Started"};
  55. static esp_avrc_rn_evt_cap_mask_t s_avrc_peer_rn_cap;
  56. static _lock_t s_volume_lock;
  57. static int s_volume, abs_volume, sink_volume;
  58. static bool s_volume_notify;
  59. static enum { AUDIO_IDLE, AUDIO_CONNECTED, AUDIO_PLAYING } s_audio = AUDIO_IDLE;
  60. static int s_sample_rate;
  61. static int tl;
  62. static bt_cmd_vcb_t cmd_handler_chain;
  63. #define METADATA_LEN 128
  64. static EXT_RAM_ATTR struct {
  65. char artist[METADATA_LEN + 1];
  66. char album[METADATA_LEN + 1];
  67. char title[METADATA_LEN + 1];
  68. int duration;
  69. bool updated;
  70. } s_metadata;
  71. static void bt_volume_up(bool pressed) {
  72. if (!pressed) return;
  73. volume_set_by_local_host(+3, true);
  74. (*bt_app_a2d_cmd_cb)(BT_SINK_VOLUME, s_volume);
  75. ESP_LOGD(BT_AV_TAG, "BT volume up %u", s_volume);
  76. }
  77. static void bt_volume_down(bool pressed) {
  78. if (!pressed) return;
  79. volume_set_by_local_host(-3, true);
  80. (*bt_app_a2d_cmd_cb)(BT_SINK_VOLUME, s_volume);
  81. }
  82. static void bt_toggle(bool pressed) {
  83. if (!pressed) return;
  84. if (s_audio == AUDIO_PLAYING) esp_avrc_ct_send_passthrough_cmd(tl++ & 0x0f, ESP_AVRC_PT_CMD_STOP, ESP_AVRC_PT_CMD_STATE_PRESSED);
  85. else esp_avrc_ct_send_passthrough_cmd(tl++, ESP_AVRC_PT_CMD_PLAY, ESP_AVRC_PT_CMD_STATE_PRESSED);
  86. }
  87. static void bt_play(bool pressed) {
  88. if (!pressed) return;
  89. esp_avrc_ct_send_passthrough_cmd(tl++ & 0x0f, ESP_AVRC_PT_CMD_PLAY, ESP_AVRC_PT_CMD_STATE_PRESSED);
  90. }
  91. static void bt_pause(bool pressed) {
  92. if (!pressed) return;
  93. esp_avrc_ct_send_passthrough_cmd(tl++ & 0x0f, ESP_AVRC_PT_CMD_PAUSE, ESP_AVRC_PT_CMD_STATE_PRESSED);
  94. }
  95. static void bt_stop(bool pressed) {
  96. if (!pressed) return;
  97. esp_avrc_ct_send_passthrough_cmd(tl++ & 0x0f, ESP_AVRC_PT_CMD_STOP, ESP_AVRC_PT_CMD_STATE_PRESSED);
  98. }
  99. static void bt_prev(bool pressed) {
  100. if (!pressed) return;
  101. esp_avrc_ct_send_passthrough_cmd(tl++ & 0x0f, ESP_AVRC_PT_CMD_BACKWARD, ESP_AVRC_PT_CMD_STATE_PRESSED);
  102. }
  103. static void bt_next(bool pressed) {
  104. if (!pressed) return;
  105. esp_avrc_ct_send_passthrough_cmd(tl++ & 0x0f, ESP_AVRC_PT_CMD_FORWARD, ESP_AVRC_PT_CMD_STATE_PRESSED);
  106. }
  107. const static actrls_t controls = {
  108. NULL, // power
  109. bt_volume_up, bt_volume_down, // volume up, volume down
  110. bt_toggle, bt_play, // toggle, play
  111. bt_pause, bt_stop, // pause, stop
  112. NULL, NULL, // rew, fwd
  113. bt_prev, bt_next, // prev, next
  114. NULL, NULL, NULL, NULL, // left, right, up, down
  115. NULL, NULL, NULL, NULL, NULL, NULL, // pre1-6
  116. bt_volume_down, bt_volume_up, bt_toggle// knob left, knob_right, knob push
  117. };
  118. /* disconnection */
  119. void bt_disconnect(void) {
  120. displayer_control(DISPLAYER_SHUTDOWN);
  121. if (s_audio == AUDIO_PLAYING) esp_avrc_ct_send_passthrough_cmd(tl++ & 0x0f, ESP_AVRC_PT_CMD_STOP, ESP_AVRC_PT_CMD_STATE_PRESSED);
  122. actrls_unset();
  123. ESP_LOGD(BT_AV_TAG, "forced disconnection %d", s_audio);
  124. }
  125. /* update metadata if any */
  126. void update_metadata(bool force) {
  127. if ((s_metadata.updated || force) && s_audio == AUDIO_PLAYING) {
  128. (*bt_app_a2d_cmd_cb)(BT_SINK_PROGRESS, -1, s_metadata.duration);
  129. (*bt_app_a2d_cmd_cb)(BT_SINK_METADATA, s_metadata.artist, s_metadata.album, s_metadata.title);
  130. s_metadata.updated = false;
  131. } else s_metadata.updated = force;
  132. }
  133. /* command handler */
  134. static bool cmd_handler(bt_sink_cmd_t cmd, ...) {
  135. va_list args;
  136. va_start(args, cmd);
  137. // handle audio event and stop if forbidden
  138. if (!cmd_handler_chain(cmd, args)) {
  139. va_end(args);
  140. return false;
  141. }
  142. // now handle events for display
  143. switch(cmd) {
  144. case BT_SINK_AUDIO_STARTED:
  145. displayer_control(DISPLAYER_ACTIVATE, "BLUETOOTH", false);
  146. break;
  147. case BT_SINK_AUDIO_STOPPED:
  148. displayer_control(DISPLAYER_SUSPEND);
  149. break;
  150. case BT_SINK_PLAY:
  151. displayer_control(DISPLAYER_TIMER_RUN);
  152. break;
  153. case BT_SINK_STOP:
  154. // not sure of difference between pause and stop for displayer
  155. case BT_SINK_PAUSE:
  156. displayer_control(DISPLAYER_TIMER_PAUSE);
  157. break;
  158. case BT_SINK_METADATA: {
  159. char *artist = va_arg(args, char*), *album = va_arg(args, char*), *title = va_arg(args, char*);
  160. displayer_metadata(artist, album, title);
  161. break;
  162. }
  163. case BT_SINK_PROGRESS: {
  164. int elapsed = va_arg(args, int), duration = va_arg(args, int);
  165. displayer_timer(DISPLAYER_ELAPSED, elapsed, duration);
  166. break;
  167. }
  168. default:
  169. break;
  170. }
  171. va_end(args);
  172. return true;
  173. }
  174. /* callback for A2DP sink */
  175. void bt_app_a2d_cb(esp_a2d_cb_event_t event, esp_a2d_cb_param_t *param)
  176. {
  177. switch (event) {
  178. case ESP_A2D_CONNECTION_STATE_EVT:
  179. case ESP_A2D_AUDIO_STATE_EVT:
  180. case ESP_A2D_AUDIO_CFG_EVT: {
  181. bt_app_work_dispatch(bt_av_hdl_a2d_evt, event, param, sizeof(esp_a2d_cb_param_t), NULL);
  182. break;
  183. }
  184. default:
  185. ESP_LOGE(BT_AV_TAG, "Invalid A2DP event: %d", event);
  186. break;
  187. }
  188. }
  189. void bt_app_alloc_meta_buffer(esp_avrc_ct_cb_param_t *param)
  190. {
  191. esp_avrc_ct_cb_param_t *rc = (esp_avrc_ct_cb_param_t *)(param);
  192. uint8_t *attr_text = (uint8_t *) malloc (rc->meta_rsp.attr_length + 1);
  193. memcpy(attr_text, rc->meta_rsp.attr_text, rc->meta_rsp.attr_length);
  194. attr_text[rc->meta_rsp.attr_length] = 0;
  195. rc->meta_rsp.attr_text = attr_text;
  196. }
  197. void bt_app_rc_ct_cb(esp_avrc_ct_cb_event_t event, esp_avrc_ct_cb_param_t *param)
  198. {
  199. switch (event) {
  200. case ESP_AVRC_CT_METADATA_RSP_EVT:
  201. bt_app_alloc_meta_buffer(param);
  202. /* fall through */
  203. case ESP_AVRC_CT_CONNECTION_STATE_EVT:
  204. case ESP_AVRC_CT_PASSTHROUGH_RSP_EVT:
  205. case ESP_AVRC_CT_CHANGE_NOTIFY_EVT:
  206. case ESP_AVRC_CT_REMOTE_FEATURES_EVT:
  207. case ESP_AVRC_CT_GET_RN_CAPABILITIES_RSP_EVT: {
  208. bt_app_work_dispatch(bt_av_hdl_avrc_ct_evt, event, param, sizeof(esp_avrc_ct_cb_param_t), NULL);
  209. break;
  210. }
  211. default:
  212. ESP_LOGE(BT_RC_CT_TAG, "Invalid AVRC event: %d", event);
  213. break;
  214. }
  215. }
  216. void bt_app_rc_tg_cb(esp_avrc_tg_cb_event_t event, esp_avrc_tg_cb_param_t *param)
  217. {
  218. switch (event) {
  219. case ESP_AVRC_TG_CONNECTION_STATE_EVT:
  220. case ESP_AVRC_TG_REMOTE_FEATURES_EVT:
  221. case ESP_AVRC_TG_PASSTHROUGH_CMD_EVT:
  222. case ESP_AVRC_TG_SET_ABSOLUTE_VOLUME_CMD_EVT:
  223. case ESP_AVRC_TG_REGISTER_NOTIFICATION_EVT:
  224. bt_app_work_dispatch(bt_av_hdl_avrc_tg_evt, event, param, sizeof(esp_avrc_tg_cb_param_t), NULL);
  225. break;
  226. default:
  227. ESP_LOGE(BT_RC_TG_TAG, "Invalid AVRC event: %d", event);
  228. break;
  229. }
  230. }
  231. static void bt_av_hdl_a2d_evt(uint16_t event, void *p_param)
  232. {
  233. ESP_LOGD(BT_AV_TAG, "%s evt %d", __func__, event);
  234. esp_a2d_cb_param_t *a2d = NULL;
  235. switch (event) {
  236. case ESP_A2D_CONNECTION_STATE_EVT: {
  237. a2d = (esp_a2d_cb_param_t *)(p_param);
  238. uint8_t *bda = a2d->conn_stat.remote_bda;
  239. ESP_LOGD(BT_AV_TAG, "A2DP connection state: %s, [%02x:%02x:%02x:%02x:%02x:%02x]",
  240. s_a2d_conn_state_str[a2d->conn_stat.state], bda[0], bda[1], bda[2], bda[3], bda[4], bda[5]);
  241. if (a2d->conn_stat.state == ESP_A2D_CONNECTION_STATE_DISCONNECTED) {
  242. esp_bt_gap_set_scan_mode(ESP_BT_CONNECTABLE, ESP_BT_GENERAL_DISCOVERABLE);
  243. (*bt_app_a2d_cmd_cb)(BT_SINK_DISCONNECTED);
  244. } else if (a2d->conn_stat.state == ESP_A2D_CONNECTION_STATE_CONNECTED){
  245. abs_volume = -1;
  246. s_volume = sink_volume;
  247. esp_bt_gap_set_scan_mode(ESP_BT_NON_CONNECTABLE, ESP_BT_NON_DISCOVERABLE);
  248. (*bt_app_a2d_cmd_cb)(BT_SINK_CONNECTED);
  249. }
  250. break;
  251. }
  252. case ESP_A2D_AUDIO_STATE_EVT: {
  253. a2d = (esp_a2d_cb_param_t *)(p_param);
  254. ESP_LOGD(BT_AV_TAG, "A2DP audio state: %s", s_a2d_audio_state_str[a2d->audio_stat.state]);
  255. if (ESP_A2D_AUDIO_STATE_STARTED == a2d->audio_stat.state) {
  256. s_audio = AUDIO_CONNECTED;
  257. // send memorized volume for devices that can't do absolute volume
  258. (*bt_app_a2d_cmd_cb)(BT_SINK_VOLUME, s_volume);
  259. // verify that we can take control
  260. if ((*bt_app_a2d_cmd_cb)(BT_SINK_AUDIO_STARTED, s_sample_rate)) {
  261. // if PLAY is sent before AUDIO_STARTED, generate the event here
  262. s_audio = AUDIO_PLAYING;
  263. (*bt_app_a2d_cmd_cb)(BT_SINK_PLAY);
  264. // force metadata update
  265. update_metadata(true);
  266. actrls_set(controls, false, NULL, actrls_ir_action);
  267. } else {
  268. // if decoder is busy, stop it (would be better to not ACK this command, but don't know how)
  269. esp_avrc_ct_send_passthrough_cmd(tl++ & 0x0f, ESP_AVRC_PT_CMD_STOP, ESP_AVRC_PT_CMD_STATE_PRESSED);
  270. }
  271. } else if (ESP_A2D_AUDIO_STATE_STOPPED == a2d->audio_stat.state ||
  272. ESP_A2D_AUDIO_STATE_REMOTE_SUSPEND == a2d->audio_stat.state) {
  273. (*bt_app_a2d_cmd_cb)(BT_SINK_AUDIO_STOPPED);
  274. s_audio = AUDIO_IDLE;
  275. actrls_unset();
  276. }
  277. break;
  278. }
  279. case ESP_A2D_AUDIO_CFG_EVT: {
  280. a2d = (esp_a2d_cb_param_t *)(p_param);
  281. ESP_LOGD(BT_AV_TAG, "A2DP audio stream configuration, codec type %d", a2d->audio_cfg.mcc.type);
  282. // for now only SBC stream is supported
  283. if (a2d->audio_cfg.mcc.type == ESP_A2D_MCT_SBC) {
  284. s_sample_rate = 16000;
  285. char oct0 = a2d->audio_cfg.mcc.cie.sbc[0];
  286. if (oct0 & (0x01 << 6)) {
  287. s_sample_rate = 32000;
  288. } else if (oct0 & (0x01 << 5)) {
  289. s_sample_rate = 44100;
  290. } else if (oct0 & (0x01 << 4)) {
  291. s_sample_rate = 48000;
  292. }
  293. (*bt_app_a2d_cmd_cb)(BT_SINK_RATE, s_sample_rate);
  294. ESP_LOGI(BT_AV_TAG, "Configure audio player %x-%x-%x-%x",
  295. a2d->audio_cfg.mcc.cie.sbc[0],
  296. a2d->audio_cfg.mcc.cie.sbc[1],
  297. a2d->audio_cfg.mcc.cie.sbc[2],
  298. a2d->audio_cfg.mcc.cie.sbc[3]);
  299. ESP_LOGI(BT_AV_TAG, "Audio player configured, sample rate=%d", s_sample_rate);
  300. }
  301. break;
  302. }
  303. default:
  304. ESP_LOGE(BT_AV_TAG, "%s unhandled evt %d", __func__, event);
  305. break;
  306. }
  307. }
  308. static void bt_av_new_track(void)
  309. {
  310. // request metadata
  311. uint8_t attr_mask = ESP_AVRC_MD_ATTR_TITLE | ESP_AVRC_MD_ATTR_ARTIST | ESP_AVRC_MD_ATTR_ALBUM | ESP_AVRC_MD_ATTR_PLAYING_TIME;
  312. esp_avrc_ct_send_metadata_cmd(APP_RC_CT_TL_GET_META_DATA, attr_mask);
  313. // register notification if peer support the event_id
  314. if (esp_avrc_rn_evt_bit_mask_operation(ESP_AVRC_BIT_MASK_OP_TEST, &s_avrc_peer_rn_cap,
  315. ESP_AVRC_RN_TRACK_CHANGE)) {
  316. esp_avrc_ct_send_register_notification_cmd(APP_RC_CT_TL_RN_TRACK_CHANGE, ESP_AVRC_RN_TRACK_CHANGE, 0);
  317. }
  318. }
  319. static void bt_av_playback_changed(void)
  320. {
  321. if (esp_avrc_rn_evt_bit_mask_operation(ESP_AVRC_BIT_MASK_OP_TEST, &s_avrc_peer_rn_cap,
  322. ESP_AVRC_RN_PLAY_STATUS_CHANGE)) {
  323. esp_avrc_ct_send_register_notification_cmd(APP_RC_CT_TL_RN_PLAYBACK_CHANGE, ESP_AVRC_RN_PLAY_STATUS_CHANGE, 0);
  324. }
  325. }
  326. static void bt_av_play_pos_changed(void)
  327. {
  328. if (esp_avrc_rn_evt_bit_mask_operation(ESP_AVRC_BIT_MASK_OP_TEST, &s_avrc_peer_rn_cap,
  329. ESP_AVRC_RN_PLAY_POS_CHANGED)) {
  330. esp_avrc_ct_send_register_notification_cmd(APP_RC_CT_TL_RN_PLAY_POS_CHANGE, ESP_AVRC_RN_PLAY_POS_CHANGED, 10);
  331. }
  332. }
  333. static void bt_av_notify_evt_handler(uint8_t event_id, esp_avrc_rn_param_t *event_parameter)
  334. {
  335. switch (event_id) {
  336. case ESP_AVRC_RN_TRACK_CHANGE:
  337. ESP_LOGD(BT_AV_TAG, "Track changed");
  338. bt_av_new_track();
  339. (*bt_app_a2d_cmd_cb)(BT_SINK_PROGRESS, 0, 0);
  340. break;
  341. case ESP_AVRC_RN_PLAY_STATUS_CHANGE:
  342. ESP_LOGD(BT_AV_TAG, "Playback status changed: 0x%x", event_parameter->playback);
  343. if (s_audio != AUDIO_IDLE) {
  344. switch (event_parameter->playback) {
  345. case ESP_AVRC_PLAYBACK_PLAYING:
  346. // if decoder is busy then stop (would be better to not ACK this command, but don't know how)
  347. if (s_audio != AUDIO_PLAYING && !(*bt_app_a2d_cmd_cb)(BT_SINK_PLAY)) {
  348. ESP_LOGW(BT_AV_TAG, "Player busy with another controller");
  349. esp_avrc_ct_send_passthrough_cmd(tl++ & 0x0f, ESP_AVRC_PT_CMD_STOP, ESP_AVRC_PT_CMD_STATE_PRESSED);
  350. } else {
  351. s_audio = AUDIO_PLAYING;
  352. update_metadata(false);
  353. }
  354. break;
  355. case ESP_AVRC_PLAYBACK_PAUSED:
  356. s_audio = AUDIO_CONNECTED;
  357. (*bt_app_a2d_cmd_cb)(BT_SINK_PAUSE);
  358. break;
  359. case ESP_AVRC_PLAYBACK_STOPPED:
  360. s_audio = AUDIO_CONNECTED;
  361. (*bt_app_a2d_cmd_cb)(BT_SINK_PROGRESS, 0, -1);
  362. (*bt_app_a2d_cmd_cb)(BT_SINK_STOP);
  363. break;
  364. default:
  365. ESP_LOGW(BT_AV_TAG, "Un-handled event");
  366. break;
  367. }
  368. } else {
  369. ESP_LOGW(BT_AV_TAG, "Not yet in BT connected mode: 0x%x", event_parameter->playback);
  370. }
  371. bt_av_playback_changed();
  372. break;
  373. case ESP_AVRC_RN_PLAY_POS_CHANGED:
  374. ESP_LOGD(BT_AV_TAG, "Play position changed: %d (ms)", event_parameter->play_pos);
  375. (*bt_app_a2d_cmd_cb)(BT_SINK_PROGRESS, event_parameter->play_pos, -1);
  376. bt_av_play_pos_changed();
  377. break;
  378. }
  379. }
  380. static void bt_av_hdl_avrc_ct_evt(uint16_t event, void *p_param)
  381. {
  382. ESP_LOGD(BT_RC_CT_TAG, "%s evt %d", __func__, event);
  383. esp_avrc_ct_cb_param_t *rc = (esp_avrc_ct_cb_param_t *)(p_param);
  384. switch (event) {
  385. case ESP_AVRC_CT_CONNECTION_STATE_EVT: {
  386. uint8_t *bda = rc->conn_stat.remote_bda;
  387. ESP_LOGD(BT_RC_CT_TAG, "AVRC conn_state evt: state %d, [%02x:%02x:%02x:%02x:%02x:%02x]",
  388. rc->conn_stat.connected, bda[0], bda[1], bda[2], bda[3], bda[4], bda[5]);
  389. if (rc->conn_stat.connected) {
  390. // get remote supported event_ids of peer AVRCP Target
  391. esp_avrc_ct_send_get_rn_capabilities_cmd(APP_RC_CT_TL_GET_CAPS);
  392. } else {
  393. // clear peer notification capability record
  394. s_avrc_peer_rn_cap.bits = 0;
  395. }
  396. break;
  397. }
  398. case ESP_AVRC_CT_PASSTHROUGH_RSP_EVT: {
  399. ESP_LOGD(BT_RC_CT_TAG, "AVRC passthrough rsp: key_code 0x%x, key_state %d", rc->psth_rsp.key_code, rc->psth_rsp.key_state);
  400. break;
  401. }
  402. case ESP_AVRC_CT_METADATA_RSP_EVT: {
  403. ESP_LOGD(BT_RC_CT_TAG, "AVRC metadata rsp: attribute id 0x%x, %s", rc->meta_rsp.attr_id, rc->meta_rsp.attr_text);
  404. if (rc->meta_rsp.attr_id == ESP_AVRC_MD_ATTR_PLAYING_TIME) s_metadata.duration = atoi((char*) rc->meta_rsp.attr_text);
  405. else if (rc->meta_rsp.attr_id == ESP_AVRC_MD_ATTR_TITLE) strncpy(s_metadata.title, (char*) rc->meta_rsp.attr_text, METADATA_LEN);
  406. else if (rc->meta_rsp.attr_id == ESP_AVRC_MD_ATTR_ARTIST) strncpy(s_metadata.artist, (char*) rc->meta_rsp.attr_text, METADATA_LEN);
  407. else if (rc->meta_rsp.attr_id == ESP_AVRC_MD_ATTR_ALBUM) strncpy(s_metadata.album, (char*) rc->meta_rsp.attr_text, METADATA_LEN);
  408. update_metadata(true);
  409. free(rc->meta_rsp.attr_text);
  410. break;
  411. }
  412. case ESP_AVRC_CT_CHANGE_NOTIFY_EVT: {
  413. ESP_LOGD(BT_RC_CT_TAG, "AVRC event notification: %d", rc->change_ntf.event_id);
  414. bt_av_notify_evt_handler(rc->change_ntf.event_id, &rc->change_ntf.event_parameter);
  415. break;
  416. }
  417. case ESP_AVRC_CT_REMOTE_FEATURES_EVT: {
  418. ESP_LOGD(BT_RC_CT_TAG, "AVRC remote features %x, TG features %x", rc->rmt_feats.feat_mask, rc->rmt_feats.tg_feat_flag);
  419. break;
  420. }
  421. case ESP_AVRC_CT_GET_RN_CAPABILITIES_RSP_EVT: {
  422. ESP_LOGD(BT_RC_CT_TAG, "remote rn_cap: count %d, bitmask 0x%x", rc->get_rn_caps_rsp.cap_count,
  423. rc->get_rn_caps_rsp.evt_set.bits);
  424. s_avrc_peer_rn_cap.bits = rc->get_rn_caps_rsp.evt_set.bits;
  425. bt_av_new_track();
  426. bt_av_playback_changed();
  427. bt_av_play_pos_changed();
  428. break;
  429. }
  430. default:
  431. ESP_LOGE(BT_RC_CT_TAG, "%s unhandled evt %d", __func__, event);
  432. break;
  433. }
  434. }
  435. static void volume_set_by_controller(uint8_t volume)
  436. {
  437. // do not modified NVS volume
  438. _lock_acquire(&s_volume_lock);
  439. s_volume = abs_volume = (volume * 100) / 127;
  440. _lock_release(&s_volume_lock);
  441. (*bt_app_a2d_cmd_cb)(BT_SINK_VOLUME, s_volume);
  442. }
  443. static void volume_set_by_local_host(int value, bool is_step)
  444. {
  445. _lock_acquire(&s_volume_lock);
  446. s_volume = is_step ? s_volume + value : value;
  447. if (s_volume > 127) s_volume = 127;
  448. else if (s_volume < 0) s_volume = 0;
  449. if (abs_volume >= 0) abs_volume = s_volume;
  450. else sink_volume = s_volume;
  451. _lock_release(&s_volume_lock);
  452. // volume has been set by controller, do not store it in NVS
  453. if (abs_volume < 0) {
  454. char p[4];
  455. config_set_value(NVS_TYPE_STR, "bt_sink_volume", itoa(s_volume, p, 10));
  456. }
  457. if (s_volume_notify) {
  458. esp_avrc_rn_param_t rn_param;
  459. rn_param.volume = s_volume;
  460. esp_avrc_tg_send_rn_rsp(ESP_AVRC_RN_VOLUME_CHANGE, ESP_AVRC_RN_RSP_CHANGED, &rn_param);
  461. s_volume_notify = false;
  462. }
  463. }
  464. static void bt_av_hdl_avrc_tg_evt(uint16_t event, void *p_param)
  465. {
  466. ESP_LOGD(BT_RC_TG_TAG, "%s evt %d", __func__, event);
  467. esp_avrc_tg_cb_param_t *rc = (esp_avrc_tg_cb_param_t *)(p_param);
  468. switch (event) {
  469. case ESP_AVRC_TG_CONNECTION_STATE_EVT: {
  470. uint8_t *bda = rc->conn_stat.remote_bda;
  471. ESP_LOGD(BT_RC_TG_TAG, "AVRC conn_state evt: state %d, [%02x:%02x:%02x:%02x:%02x:%02x]",
  472. rc->conn_stat.connected, bda[0], bda[1], bda[2], bda[3], bda[4], bda[5]);
  473. break;
  474. }
  475. case ESP_AVRC_TG_PASSTHROUGH_CMD_EVT: {
  476. ESP_LOGD(BT_RC_TG_TAG, "AVRC passthrough cmd: key_code 0x%x, key_state %d", rc->psth_cmd.key_code, rc->psth_cmd.key_state);
  477. break;
  478. }
  479. case ESP_AVRC_TG_SET_ABSOLUTE_VOLUME_CMD_EVT: {
  480. ESP_LOGD(BT_RC_TG_TAG, "AVRC set absolute volume: %d%%", (rc->set_abs_vol.volume * 100) / 127);
  481. volume_set_by_controller(rc->set_abs_vol.volume);
  482. break;
  483. }
  484. case ESP_AVRC_TG_REGISTER_NOTIFICATION_EVT: {
  485. ESP_LOGD(BT_RC_TG_TAG, "AVRC register event notification: %d, param: 0x%x", rc->reg_ntf.event_id, rc->reg_ntf.event_parameter);
  486. if (rc->reg_ntf.event_id == ESP_AVRC_RN_VOLUME_CHANGE) {
  487. s_volume_notify = true;
  488. esp_avrc_rn_param_t rn_param;
  489. rn_param.volume = s_volume;
  490. esp_avrc_tg_send_rn_rsp(ESP_AVRC_RN_VOLUME_CHANGE, ESP_AVRC_RN_RSP_INTERIM, &rn_param);
  491. }
  492. break;
  493. }
  494. case ESP_AVRC_TG_REMOTE_FEATURES_EVT: {
  495. ESP_LOGD(BT_RC_TG_TAG, "AVRC remote features %x, CT features %x", rc->rmt_feats.feat_mask, rc->rmt_feats.ct_feat_flag);
  496. break;
  497. }
  498. default:
  499. ESP_LOGE(BT_RC_TG_TAG, "%s unhandled evt %d", __func__, event);
  500. break;
  501. }
  502. }
  503. void bt_sink_init(bt_cmd_vcb_t cmd_cb, bt_data_cb_t data_cb)
  504. {
  505. bt_app_a2d_cmd_cb = cmd_handler;
  506. cmd_handler_chain = cmd_cb;
  507. bt_app_a2d_data_cb = data_cb;
  508. // create task and run event loop
  509. bt_app_task_start_up(bt_av_hdl_stack_evt);
  510. char *item = config_alloc_get_default(NVS_TYPE_STR, "bt_sink_volume", "127", 0);
  511. sink_volume = atol(item);
  512. free(item);
  513. /*
  514. * Set default parameters for Legacy Pairing
  515. */
  516. esp_bt_pin_type_t pin_type = ESP_BT_PIN_TYPE_FIXED;
  517. char * pin_code = config_alloc_get_default(NVS_TYPE_STR, "bt_sink_pin", STR(CONFIG_BT_SINK_PIN), 0);
  518. if(strlen(pin_code)>ESP_BT_PIN_CODE_LEN){
  519. ESP_LOGW(BT_AV_TAG, "BT Sink pin code [%s] too long. ", pin_code);
  520. pin_code[ESP_BT_PIN_CODE_LEN] = '\0';
  521. ESP_LOGW(BT_AV_TAG, "BT Sink pin truncated code [%s]. ", pin_code);
  522. }
  523. esp_bt_pin_code_t esp_pin_code;
  524. bool bError=false;
  525. memset(esp_pin_code, 0x00, sizeof(esp_pin_code) );
  526. ESP_LOGW(BT_AV_TAG, "BT Sink pin code is: [%s] ", pin_code);
  527. for(int i=0;i<strlen(pin_code);i++){
  528. if(pin_code[i] < '0' || pin_code[i] > '9' ) {
  529. ESP_LOGE(BT_AV_TAG,"Invalid number found in sequence");
  530. bError=true;
  531. }
  532. esp_pin_code[i]= pin_code[i];
  533. }
  534. if (bError) memcpy(esp_pin_code, "1234", 4);
  535. esp_bt_gap_set_pin(pin_type, strlen(pin_code), esp_pin_code);
  536. free(pin_code);
  537. }
  538. void bt_sink_deinit(void)
  539. {
  540. bt_app_task_shut_down();
  541. }
  542. static void bt_app_gap_cb(esp_bt_gap_cb_event_t event, esp_bt_gap_cb_param_t *param)
  543. {
  544. switch (event) {
  545. case ESP_BT_GAP_AUTH_CMPL_EVT: {
  546. if (param->auth_cmpl.stat == ESP_BT_STATUS_SUCCESS) {
  547. ESP_LOGD(BT_AV_TAG, "authentication success: %s", param->auth_cmpl.device_name);
  548. esp_log_buffer_hex(BT_AV_TAG, param->auth_cmpl.bda, ESP_BD_ADDR_LEN);
  549. } else {
  550. ESP_LOGE(BT_AV_TAG, "authentication failed, status:%d", param->auth_cmpl.stat);
  551. }
  552. break;
  553. }
  554. #if (CONFIG_BT_SSP_ENABLED == true)
  555. case ESP_BT_GAP_CFM_REQ_EVT:
  556. ESP_LOGD(BT_AV_TAG, "ESP_BT_GAP_CFM_REQ_EVT Please compare the numeric value: %d", param->cfm_req.num_val);
  557. esp_bt_gap_ssp_confirm_reply(param->cfm_req.bda, true);
  558. break;
  559. case ESP_BT_GAP_KEY_NOTIF_EVT:
  560. ESP_LOGD(BT_AV_TAG, "ESP_BT_GAP_KEY_NOTIF_EVT passkey:%d", param->key_notif.passkey);
  561. break;
  562. case ESP_BT_GAP_KEY_REQ_EVT:
  563. ESP_LOGD(BT_AV_TAG, "ESP_BT_GAP_KEY_REQ_EVT Please enter passkey!");
  564. break;
  565. #endif
  566. default: {
  567. ESP_LOGD(BT_AV_TAG, "event: %d", event);
  568. break;
  569. }
  570. }
  571. return;
  572. }
  573. static void bt_av_hdl_stack_evt(uint16_t event, void *p_param)
  574. {
  575. ESP_LOGD(BT_AV_TAG, "%s evt %d", __func__, event);
  576. switch (event) {
  577. case BT_APP_EVT_STACK_UP: {
  578. /* set up device name */
  579. bt_name = (char * )config_alloc_get_default(NVS_TYPE_STR, "bt_name", CONFIG_BT_NAME, 0);
  580. esp_bt_dev_set_device_name(bt_name);
  581. free(bt_name);
  582. esp_bt_gap_register_callback(bt_app_gap_cb);
  583. /* initialize AVRCP controller */
  584. esp_avrc_ct_init();
  585. esp_avrc_ct_register_callback(bt_app_rc_ct_cb);
  586. /* initialize AVRCP target */
  587. assert (esp_avrc_tg_init() == ESP_OK);
  588. esp_avrc_tg_register_callback(bt_app_rc_tg_cb);
  589. esp_avrc_rn_evt_cap_mask_t evt_set = {0};
  590. esp_avrc_rn_evt_bit_mask_operation(ESP_AVRC_BIT_MASK_OP_SET, &evt_set, ESP_AVRC_RN_VOLUME_CHANGE);
  591. assert(esp_avrc_tg_set_rn_evt_cap(&evt_set) == ESP_OK);
  592. /* initialize A2DP sink */
  593. esp_a2d_register_callback(&bt_app_a2d_cb);
  594. esp_a2d_sink_register_data_callback(bt_app_a2d_data_cb);
  595. esp_a2d_sink_init();
  596. /* set discoverable and connectable mode, wait to be connected */
  597. esp_bt_gap_set_scan_mode(ESP_BT_CONNECTABLE, ESP_BT_GENERAL_DISCOVERABLE);
  598. break;
  599. }
  600. default:
  601. ESP_LOGE(BT_AV_TAG, "%s unhandled evt %d", __func__, event);
  602. break;
  603. }
  604. }