bt.c 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. #include "bt_i.h"
  2. #include "battery_service.h"
  3. #define BT_SERVICE_TAG "BT"
  4. static void bt_draw_statusbar_callback(Canvas* canvas, void* context) {
  5. canvas_draw_icon(canvas, 0, 0, &I_Bluetooth_5x8);
  6. }
  7. static ViewPort* bt_statusbar_view_port_alloc() {
  8. ViewPort* statusbar_view_port = view_port_alloc();
  9. view_port_set_width(statusbar_view_port, 5);
  10. view_port_draw_callback_set(statusbar_view_port, bt_draw_statusbar_callback, NULL);
  11. view_port_enabled_set(statusbar_view_port, false);
  12. return statusbar_view_port;
  13. }
  14. static void bt_pin_code_show_event_handler(Bt* bt, uint32_t pin) {
  15. furi_assert(bt);
  16. string_t pin_str;
  17. string_init_printf(pin_str, "%06d", pin);
  18. dialog_message_set_text(
  19. bt->dialog_message, string_get_cstr(pin_str), 64, 32, AlignCenter, AlignCenter);
  20. dialog_message_set_buttons(bt->dialog_message, "Back", NULL, NULL);
  21. dialog_message_show(bt->dialogs, bt->dialog_message);
  22. string_clear(pin_str);
  23. }
  24. static void bt_battery_level_changed_callback(const void* _event, void* context) {
  25. furi_assert(_event);
  26. furi_assert(context);
  27. Bt* bt = context;
  28. const PowerEvent* event = _event;
  29. if(event->type == PowerEventTypeBatteryLevelChanged) {
  30. BtMessage message = {
  31. .type = BtMessageTypeUpdateBatteryLevel,
  32. .data.battery_level = event->data.battery_level};
  33. furi_check(osMessageQueuePut(bt->message_queue, &message, 0, osWaitForever) == osOK);
  34. }
  35. }
  36. Bt* bt_alloc() {
  37. Bt* bt = furi_alloc(sizeof(Bt));
  38. // Load settings
  39. if(!bt_settings_load(&bt->bt_settings)) {
  40. bt_settings_save(&bt->bt_settings);
  41. }
  42. // Alloc queue
  43. bt->message_queue = osMessageQueueNew(8, sizeof(BtMessage), NULL);
  44. // Setup statusbar view port
  45. bt->statusbar_view_port = bt_statusbar_view_port_alloc();
  46. // Gui
  47. bt->gui = furi_record_open("gui");
  48. gui_add_view_port(bt->gui, bt->statusbar_view_port, GuiLayerStatusBarLeft);
  49. // Dialogs
  50. bt->dialogs = furi_record_open("dialogs");
  51. bt->dialog_message = dialog_message_alloc();
  52. // Power
  53. bt->power = furi_record_open("power");
  54. PubSub* power_pubsub = power_get_pubsub(bt->power);
  55. subscribe_pubsub(power_pubsub, bt_battery_level_changed_callback, bt);
  56. // RPC
  57. bt->rpc = furi_record_open("rpc");
  58. bt->rpc_sem = osSemaphoreNew(1, 0, NULL);
  59. return bt;
  60. }
  61. // Called from GAP thread from Serial service
  62. static void bt_on_data_received_callback(uint8_t* data, uint16_t size, void* context) {
  63. furi_assert(context);
  64. Bt* bt = context;
  65. size_t bytes_processed = rpc_feed_bytes(bt->rpc_session, data, size, 1000);
  66. if(bytes_processed != size) {
  67. FURI_LOG_E(BT_SERVICE_TAG, "Only %d of %d bytes processed by RPC", bytes_processed, size);
  68. }
  69. }
  70. // Called from GAP thread from Serial service
  71. static void bt_on_data_sent_callback(void* context) {
  72. furi_assert(context);
  73. Bt* bt = context;
  74. osSemaphoreRelease(bt->rpc_sem);
  75. }
  76. // Called from RPC thread
  77. static void bt_rpc_send_bytes_callback(void* context, uint8_t* bytes, size_t bytes_len) {
  78. furi_assert(context);
  79. Bt* bt = context;
  80. size_t bytes_sent = 0;
  81. while(bytes_sent < bytes_len) {
  82. size_t bytes_remain = bytes_len - bytes_sent;
  83. if(bytes_remain > FURI_HAL_BT_PACKET_SIZE_MAX) {
  84. furi_hal_bt_tx(&bytes[bytes_sent], FURI_HAL_BT_PACKET_SIZE_MAX);
  85. bytes_sent += FURI_HAL_BT_PACKET_SIZE_MAX;
  86. } else {
  87. furi_hal_bt_tx(&bytes[bytes_sent], bytes_remain);
  88. bytes_sent += bytes_remain;
  89. }
  90. osSemaphoreAcquire(bt->rpc_sem, osWaitForever);
  91. }
  92. }
  93. // Called from GAP thread
  94. static void bt_on_gap_event_callback(BleEvent event, void* context) {
  95. furi_assert(context);
  96. Bt* bt = context;
  97. if(event.type == BleEventTypeConnected) {
  98. FURI_LOG_I(BT_SERVICE_TAG, "Open RPC connection");
  99. bt->rpc_session = rpc_open_session(bt->rpc);
  100. rpc_set_send_bytes_callback(bt->rpc_session, bt_rpc_send_bytes_callback, bt);
  101. furi_hal_bt_set_data_event_callbacks(
  102. bt_on_data_received_callback, bt_on_data_sent_callback, bt);
  103. // Update battery level
  104. PowerInfo info;
  105. power_get_info(bt->power, &info);
  106. BtMessage message = {
  107. .type = BtMessageTypeUpdateBatteryLevel, .data.battery_level = info.charge};
  108. furi_check(osMessageQueuePut(bt->message_queue, &message, 0, osWaitForever) == osOK);
  109. } else if(event.type == BleEventTypeDisconnected) {
  110. FURI_LOG_I(BT_SERVICE_TAG, "Close RPC connection");
  111. if(bt->rpc_session) {
  112. rpc_close_session(bt->rpc_session);
  113. bt->rpc_session = NULL;
  114. }
  115. } else if(event.type == BleEventTypeStartAdvertising || event.type == BleEventTypeStopAdvertising) {
  116. BtMessage message = {.type = BtMessageTypeUpdateStatusbar};
  117. furi_check(osMessageQueuePut(bt->message_queue, &message, 0, osWaitForever) == osOK);
  118. } else if(event.type == BleEventTypePinCodeShow) {
  119. BtMessage message = {
  120. .type = BtMessageTypePinCodeShow, .data.pin_code = event.data.pin_code};
  121. furi_check(osMessageQueuePut(bt->message_queue, &message, 0, osWaitForever) == osOK);
  122. }
  123. }
  124. int32_t bt_srv() {
  125. Bt* bt = bt_alloc();
  126. furi_record_create("bt", bt);
  127. if(!furi_hal_bt_wait_startup()) {
  128. FURI_LOG_E(BT_SERVICE_TAG, "Core2 startup failed");
  129. } else {
  130. view_port_enabled_set(bt->statusbar_view_port, true);
  131. if(furi_hal_bt_init_app(bt_on_gap_event_callback, bt)) {
  132. FURI_LOG_I(BT_SERVICE_TAG, "BLE stack started");
  133. if(bt->bt_settings.enabled) {
  134. furi_hal_bt_start_advertising();
  135. }
  136. } else {
  137. FURI_LOG_E(BT_SERVICE_TAG, "BT App start failed");
  138. }
  139. }
  140. // Update statusbar
  141. view_port_enabled_set(bt->statusbar_view_port, furi_hal_bt_is_active());
  142. BtMessage message;
  143. while(1) {
  144. furi_check(osMessageQueueGet(bt->message_queue, &message, NULL, osWaitForever) == osOK);
  145. if(message.type == BtMessageTypeUpdateStatusbar) {
  146. // Update statusbar
  147. view_port_enabled_set(bt->statusbar_view_port, furi_hal_bt_is_active());
  148. } else if(message.type == BtMessageTypeUpdateBatteryLevel) {
  149. // Update battery level
  150. if(furi_hal_bt_is_active()) {
  151. battery_svc_update_level(message.data.battery_level);
  152. }
  153. } else if(message.type == BtMessageTypePinCodeShow) {
  154. // Display PIN code
  155. bt_pin_code_show_event_handler(bt, message.data.pin_code);
  156. }
  157. }
  158. return 0;
  159. }