output_bt.c 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238
  1. /*
  2. * Squeezelite for esp32
  3. *
  4. * (c) Sebastien 2019
  5. * Philippe G. 2019, philippe_44@outlook.com
  6. *
  7. * This software is released under the MIT License.
  8. * https://opensource.org/licenses/MIT
  9. *
  10. */
  11. #include <assert.h>
  12. #include "driver/gpio.h"
  13. #include "squeezelite.h"
  14. #include "equalizer.h"
  15. #include "perf_trace.h"
  16. #include "services.h"
  17. #include "led.h"
  18. #include "Config.h"
  19. extern log_level log_level_from_sys_level(sys_squeezelite_debug_levels level);
  20. extern sys_squeezelite_config* get_profile(const char* name);
  21. static sys_squeezelite_config * config = NULL;
  22. extern struct outputstate output;
  23. extern struct buffer *outputbuf;
  24. extern struct buffer *streambuf;
  25. extern u8_t *silencebuf;
  26. #define LOCK mutex_lock(outputbuf->mutex)
  27. #define UNLOCK mutex_unlock(outputbuf->mutex)
  28. #define LOCK_S mutex_lock(streambuf->mutex)
  29. #define UNLOCK_S mutex_unlock(streambuf->mutex)
  30. #define FRAME_BLOCK MAX_SILENCE_FRAMES
  31. #define STATS_REPORT_DELAY_MS 15000
  32. extern void hal_bluetooth_init();
  33. extern void hal_bluetooth_stop(void);
  34. extern u8_t config_spdif_gpio;
  35. static log_level loglevel;
  36. static bool running = false;
  37. static uint8_t *btout;
  38. static frames_t oframes;
  39. static bool stats;
  40. static uint32_t bt_idle_since;
  41. static int _write_frames(frames_t out_frames, bool silence, s32_t gainL, s32_t gainR, u8_t flags,
  42. s32_t cross_gain_in, s32_t cross_gain_out, ISAMPLE_T **cross_ptr);
  43. #define DECLARE_ALL_MIN_MAX \
  44. DECLARE_MIN_MAX(req);\
  45. DECLARE_MIN_MAX(rec);\
  46. DECLARE_MIN_MAX(bt);\
  47. DECLARE_MIN_MAX(under);\
  48. DECLARE_MIN_MAX(stream_buf);\
  49. DECLARE_MIN_MAX_DURATION(lock_out_time)
  50. #define RESET_ALL_MIN_MAX \
  51. RESET_MIN_MAX(bt); \
  52. RESET_MIN_MAX(req); \
  53. RESET_MIN_MAX(rec); \
  54. RESET_MIN_MAX(under); \
  55. RESET_MIN_MAX(stream_buf); \
  56. RESET_MIN_MAX_DURATION(lock_out_time)
  57. DECLARE_ALL_MIN_MAX;
  58. /****************************************************************************************
  59. * Get inactivity callback
  60. */
  61. static uint32_t bt_idle_callback(void) {
  62. return output.state <= OUTPUT_STOPPED ? pdTICKS_TO_MS(xTaskGetTickCount()) - bt_idle_since : 0;
  63. }
  64. /****************************************************************************************
  65. * Init BT sink
  66. */
  67. void output_init_bt() {
  68. config = get_profile(NULL); // get the active profile
  69. if(!config) return;
  70. loglevel = log_level_from_sys_level(config->log.output);
  71. bt_idle_since = pdTICKS_TO_MS(xTaskGetTickCount());
  72. services_sleep_setsleeper(bt_idle_callback);
  73. // even BT has a right to use led :-)
  74. led_blink(LED_GREEN, 200, 1000);
  75. running = true;
  76. output.write_cb = &_write_frames;
  77. hal_bluetooth_init();
  78. stats = platform->services.statistics;
  79. equalizer_set_samplerate(output.current_sample_rate);
  80. }
  81. /****************************************************************************************
  82. * Close BT sink
  83. */
  84. void output_close_bt(void) {
  85. LOCK;
  86. running = false;
  87. UNLOCK;
  88. hal_bluetooth_stop();
  89. equalizer_close();
  90. }
  91. /****************************************************************************************
  92. * Data framing callback
  93. */
  94. static int _write_frames(frames_t out_frames, bool silence, s32_t gainL, s32_t gainR, u8_t flags,
  95. s32_t cross_gain_in, s32_t cross_gain_out, ISAMPLE_T **cross_ptr) {
  96. assert(btout != NULL);
  97. if (!silence ) {
  98. if (output.fade == FADE_ACTIVE && output.fade_dir == FADE_CROSS && *cross_ptr) {
  99. _apply_cross(outputbuf, out_frames, cross_gain_in, cross_gain_out, cross_ptr);
  100. }
  101. _apply_gain(outputbuf, out_frames, gainL, gainR, flags);
  102. #if BYTES_PER_FRAME == 4
  103. memcpy(btout + oframes * BYTES_PER_FRAME, outputbuf->readp, out_frames * BYTES_PER_FRAME);
  104. #else
  105. {
  106. frames_t count = out_frames;
  107. s32_t *_iptr = (s32_t*) outputbuf->readp;
  108. s16_t *_optr = (s16_t*) (btout + oframes * BYTES_PER_FRAME);
  109. while (count--) {
  110. *_optr++ = *_iptr++ >> 16;
  111. *_optr++ = *_iptr++ >> 16;
  112. }
  113. }
  114. #endif
  115. } else {
  116. u8_t *buf = silencebuf;
  117. memcpy(btout + oframes * BYTES_PER_FRAME, buf, out_frames * BYTES_PER_FRAME);
  118. }
  119. output_visu_export(btout + oframes * BYTES_PER_FRAME, out_frames, output.current_sample_rate, silence, (gainL + gainR) / 2);
  120. oframes += out_frames;
  121. return (int)out_frames;
  122. }
  123. /****************************************************************************************
  124. * Data callback for BT stack
  125. */
  126. int32_t output_bt_data(uint8_t *data, int32_t len) {
  127. int32_t iframes = len / BYTES_PER_FRAME, start_timer = 0;
  128. if (iframes <= 0 || data == NULL || !running) {
  129. return 0;
  130. }
  131. btout = data;
  132. oframes = 0;
  133. // This is how the BTC layer calculates the number of bytes to
  134. // for us to send. (BTC_SBC_DEC_PCM_DATA_LEN * sizeof(OI_INT16) - availPcmBytes
  135. SET_MIN_MAX(len,req);
  136. TIME_MEASUREMENT_START(start_timer);
  137. LOCK;
  138. SET_MIN_MAX_SIZED(_buf_used(outputbuf),bt,outputbuf->size);
  139. output.device_frames = 0;
  140. output.updated = gettime_ms();
  141. output.frames_played_dmp = output.frames_played;
  142. _output_frames(iframes);
  143. output.frames_in_process = oframes;
  144. UNLOCK;
  145. equalizer_process(data, oframes * BYTES_PER_FRAME);
  146. SET_MIN_MAX(TIME_MEASUREMENT_GET(start_timer),lock_out_time);
  147. SET_MIN_MAX((len-oframes*BYTES_PER_FRAME), rec);
  148. TIME_MEASUREMENT_START(start_timer);
  149. return oframes * BYTES_PER_FRAME;
  150. }
  151. /****************************************************************************************
  152. * Tick for BT
  153. */
  154. void output_bt_tick(void) {
  155. static time_t lastTime=0;
  156. if (!running) return;
  157. LOCK_S;
  158. SET_MIN_MAX_SIZED(_buf_used(streambuf), stream_buf, streambuf->size);
  159. UNLOCK_S;
  160. if (stats && lastTime <= gettime_ms() )
  161. {
  162. lastTime = gettime_ms() + STATS_REPORT_DELAY_MS;
  163. LOG_INFO("Statistics over %u secs. " , STATS_REPORT_DELAY_MS/1000);
  164. LOG_INFO(" +==========+==========+================+=====+================+");
  165. LOG_INFO(" | max | min | average | avg | count |");
  166. LOG_INFO(" | (bytes) | (bytes) | (bytes) | pct | |");
  167. LOG_INFO(" +==========+==========+================+=====+================+");
  168. LOG_INFO(LINE_MIN_MAX_FORMAT,LINE_MIN_MAX("stream avl",stream_buf));
  169. LOG_INFO(LINE_MIN_MAX_FORMAT,LINE_MIN_MAX("output avl",bt));
  170. LOG_INFO(LINE_MIN_MAX_FORMAT,LINE_MIN_MAX("requested",req));
  171. LOG_INFO(LINE_MIN_MAX_FORMAT,LINE_MIN_MAX("received",rec));
  172. LOG_INFO(LINE_MIN_MAX_FORMAT,LINE_MIN_MAX("underrun",under));
  173. LOG_INFO( " +==========+==========+================+=====+================+");
  174. LOG_INFO("\n");
  175. LOG_INFO(" ==========+==========+===========+===========+ ");
  176. LOG_INFO(" max (us) | min (us) | avg(us) | count | ");
  177. LOG_INFO(" ==========+==========+===========+===========+ ");
  178. LOG_INFO(LINE_MIN_MAX_DURATION_FORMAT,LINE_MIN_MAX_DURATION("Out Buf Lock",lock_out_time));
  179. LOG_INFO(" ==========+==========+===========+===========+");
  180. RESET_ALL_MIN_MAX;
  181. }
  182. }
  183. /****************************************************************************************
  184. * BT playback stop
  185. */
  186. void output_bt_stop(void) {
  187. led_blink(LED_GREEN, 200, 1000);
  188. bt_idle_since = pdTICKS_TO_MS(xTaskGetTickCount());
  189. }
  190. /****************************************************************************************
  191. * BT playback start
  192. */
  193. void output_bt_start(void) {
  194. led_on(LED_GREEN);
  195. bt_idle_since = pdTICKS_TO_MS(xTaskGetTickCount());
  196. }