ble_glue.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486
  1. #include "ble_glue.h"
  2. #include "app_common.h"
  3. #include "ble_app.h"
  4. #include <ble/ble.h>
  5. #include <interface/patterns/ble_thread/tl/tl.h>
  6. #include <interface/patterns/ble_thread/shci/shci.h>
  7. #include <interface/patterns/ble_thread/tl/shci_tl.h>
  8. #include "app_debug.h"
  9. #include <furi_hal.h>
  10. #define TAG "Core2"
  11. #define BLE_GLUE_FLAG_SHCI_EVENT (1UL << 0)
  12. #define BLE_GLUE_FLAG_KILL_THREAD (1UL << 1)
  13. #define BLE_GLUE_FLAG_ALL (BLE_GLUE_FLAG_SHCI_EVENT | BLE_GLUE_FLAG_KILL_THREAD)
  14. #define POOL_SIZE \
  15. (CFG_TLBLE_EVT_QUEUE_LENGTH * 4U * \
  16. DIVC((sizeof(TL_PacketHeader_t) + TL_BLE_EVENT_FRAME_SIZE), 4U))
  17. PLACE_IN_SECTION("MB_MEM2") ALIGN(4) static uint8_t ble_glue_event_pool[POOL_SIZE];
  18. PLACE_IN_SECTION("MB_MEM2") ALIGN(4) static TL_CmdPacket_t ble_glue_system_cmd_buff;
  19. PLACE_IN_SECTION("MB_MEM2")
  20. ALIGN(4)
  21. static uint8_t ble_glue_system_spare_event_buff[sizeof(TL_PacketHeader_t) + TL_EVT_HDR_SIZE + 255U];
  22. PLACE_IN_SECTION("MB_MEM2")
  23. ALIGN(4)
  24. static uint8_t ble_glue_ble_spare_event_buff[sizeof(TL_PacketHeader_t) + TL_EVT_HDR_SIZE + 255];
  25. typedef struct {
  26. FuriMutex* shci_mtx;
  27. FuriSemaphore* shci_sem;
  28. FuriThread* thread;
  29. BleGlueStatus status;
  30. BleGlueKeyStorageChangedCallback callback;
  31. BleGlueC2Info c2_info;
  32. void* context;
  33. } BleGlue;
  34. static BleGlue* ble_glue = NULL;
  35. static int32_t ble_glue_shci_thread(void* argument);
  36. static void ble_glue_sys_status_not_callback(SHCI_TL_CmdStatus_t status);
  37. static void ble_glue_sys_user_event_callback(void* pPayload);
  38. void ble_glue_set_key_storage_changed_callback(
  39. BleGlueKeyStorageChangedCallback callback,
  40. void* context) {
  41. furi_assert(ble_glue);
  42. furi_assert(callback);
  43. ble_glue->callback = callback;
  44. ble_glue->context = context;
  45. }
  46. ///////////////////////////////////////////////////////////////////////////////
  47. /* TL hook to catch hardfaults */
  48. int32_t ble_glue_TL_SYS_SendCmd(uint8_t* buffer, uint16_t size) {
  49. if(furi_hal_bt_get_hardfault_info()) {
  50. furi_crash("ST(R) Copro(R) HardFault");
  51. }
  52. return TL_SYS_SendCmd(buffer, size);
  53. }
  54. void shci_register_io_bus(tSHciIO* fops) {
  55. /* Register IO bus services */
  56. fops->Init = TL_SYS_Init;
  57. fops->Send = ble_glue_TL_SYS_SendCmd;
  58. }
  59. ///////////////////////////////////////////////////////////////////////////////
  60. void ble_glue_init() {
  61. ble_glue = malloc(sizeof(BleGlue));
  62. ble_glue->status = BleGlueStatusStartup;
  63. #ifdef BLE_GLUE_DEBUG
  64. APPD_Init();
  65. #endif
  66. // Initialize all transport layers
  67. TL_MM_Config_t tl_mm_config;
  68. SHCI_TL_HciInitConf_t SHci_Tl_Init_Conf;
  69. // Reference table initialization
  70. TL_Init();
  71. ble_glue->shci_mtx = furi_mutex_alloc(FuriMutexTypeNormal);
  72. ble_glue->shci_sem = furi_semaphore_alloc(1, 0);
  73. // FreeRTOS system task creation
  74. ble_glue->thread = furi_thread_alloc_ex("BleShciDriver", 1024, ble_glue_shci_thread, ble_glue);
  75. furi_thread_start(ble_glue->thread);
  76. // System channel initialization
  77. SHci_Tl_Init_Conf.p_cmdbuffer = (uint8_t*)&ble_glue_system_cmd_buff;
  78. SHci_Tl_Init_Conf.StatusNotCallBack = ble_glue_sys_status_not_callback;
  79. shci_init(ble_glue_sys_user_event_callback, (void*)&SHci_Tl_Init_Conf);
  80. /**< Memory Manager channel initialization */
  81. tl_mm_config.p_BleSpareEvtBuffer = ble_glue_ble_spare_event_buff;
  82. tl_mm_config.p_SystemSpareEvtBuffer = ble_glue_system_spare_event_buff;
  83. tl_mm_config.p_AsynchEvtPool = ble_glue_event_pool;
  84. tl_mm_config.AsynchEvtPoolSize = POOL_SIZE;
  85. TL_MM_Init(&tl_mm_config);
  86. TL_Enable();
  87. /*
  88. * From now, the application is waiting for the ready event ( VS_HCI_C2_Ready )
  89. * received on the system channel before starting the Stack
  90. * This system event is received with ble_glue_sys_user_event_callback()
  91. */
  92. }
  93. const BleGlueC2Info* ble_glue_get_c2_info() {
  94. return &ble_glue->c2_info;
  95. }
  96. BleGlueStatus ble_glue_get_c2_status() {
  97. return ble_glue->status;
  98. }
  99. static const char* ble_glue_get_reltype_str(const uint8_t reltype) {
  100. static char relcode[3] = {0};
  101. switch(reltype) {
  102. case INFO_STACK_TYPE_BLE_FULL:
  103. return "F";
  104. case INFO_STACK_TYPE_BLE_HCI:
  105. return "H";
  106. case INFO_STACK_TYPE_BLE_LIGHT:
  107. return "L";
  108. case INFO_STACK_TYPE_BLE_BEACON:
  109. return "Be";
  110. case INFO_STACK_TYPE_BLE_BASIC:
  111. return "Ba";
  112. case INFO_STACK_TYPE_BLE_FULL_EXT_ADV:
  113. return "F+";
  114. case INFO_STACK_TYPE_BLE_HCI_EXT_ADV:
  115. return "H+";
  116. default:
  117. snprintf(relcode, sizeof(relcode), "%X", reltype);
  118. return relcode;
  119. }
  120. }
  121. static void ble_glue_update_c2_fw_info() {
  122. WirelessFwInfo_t wireless_info;
  123. SHCI_GetWirelessFwInfo(&wireless_info);
  124. BleGlueC2Info* local_info = &ble_glue->c2_info;
  125. local_info->VersionMajor = wireless_info.VersionMajor;
  126. local_info->VersionMinor = wireless_info.VersionMinor;
  127. local_info->VersionSub = wireless_info.VersionSub;
  128. local_info->VersionBranch = wireless_info.VersionBranch;
  129. local_info->VersionReleaseType = wireless_info.VersionReleaseType;
  130. local_info->MemorySizeSram2B = wireless_info.MemorySizeSram2B;
  131. local_info->MemorySizeSram2A = wireless_info.MemorySizeSram2A;
  132. local_info->MemorySizeSram1 = wireless_info.MemorySizeSram1;
  133. local_info->MemorySizeFlash = wireless_info.MemorySizeFlash;
  134. local_info->StackType = wireless_info.StackType;
  135. snprintf(
  136. local_info->StackTypeString,
  137. BLE_GLUE_MAX_VERSION_STRING_LEN,
  138. "%d.%d.%d:%s",
  139. local_info->VersionMajor,
  140. local_info->VersionMinor,
  141. local_info->VersionSub,
  142. ble_glue_get_reltype_str(local_info->StackType));
  143. local_info->FusVersionMajor = wireless_info.FusVersionMajor;
  144. local_info->FusVersionMinor = wireless_info.FusVersionMinor;
  145. local_info->FusVersionSub = wireless_info.FusVersionSub;
  146. local_info->FusMemorySizeSram2B = wireless_info.FusMemorySizeSram2B;
  147. local_info->FusMemorySizeSram2A = wireless_info.FusMemorySizeSram2A;
  148. local_info->FusMemorySizeFlash = wireless_info.FusMemorySizeFlash;
  149. }
  150. static void ble_glue_dump_stack_info() {
  151. const BleGlueC2Info* c2_info = &ble_glue->c2_info;
  152. FURI_LOG_I(
  153. TAG,
  154. "Core2: FUS: %d.%d.%d, mem %d/%d, flash %d pages",
  155. c2_info->FusVersionMajor,
  156. c2_info->FusVersionMinor,
  157. c2_info->FusVersionSub,
  158. c2_info->FusMemorySizeSram2B,
  159. c2_info->FusMemorySizeSram2A,
  160. c2_info->FusMemorySizeFlash);
  161. FURI_LOG_I(
  162. TAG,
  163. "Core2: Stack: %d.%d.%d, branch %d, reltype %d, stacktype %d, flash %d pages",
  164. c2_info->VersionMajor,
  165. c2_info->VersionMinor,
  166. c2_info->VersionSub,
  167. c2_info->VersionBranch,
  168. c2_info->VersionReleaseType,
  169. c2_info->StackType,
  170. c2_info->MemorySizeFlash);
  171. }
  172. bool ble_glue_wait_for_c2_start(int32_t timeout) {
  173. bool started = false;
  174. do {
  175. // TODO: use mutex?
  176. started = ble_glue->status == BleGlueStatusC2Started;
  177. if(!started) {
  178. timeout--;
  179. furi_delay_tick(1);
  180. }
  181. } while(!started && (timeout > 0));
  182. if(started) {
  183. FURI_LOG_I(
  184. TAG,
  185. "C2 boot completed, mode: %s",
  186. ble_glue->c2_info.mode == BleGlueC2ModeFUS ? "FUS" : "Stack");
  187. ble_glue_update_c2_fw_info();
  188. ble_glue_dump_stack_info();
  189. } else {
  190. FURI_LOG_E(TAG, "C2 startup failed");
  191. ble_glue->status = BleGlueStatusBroken;
  192. }
  193. return started;
  194. }
  195. bool ble_glue_start() {
  196. furi_assert(ble_glue);
  197. if(ble_glue->status != BleGlueStatusC2Started) {
  198. return false;
  199. }
  200. bool ret = false;
  201. if(ble_app_init()) {
  202. FURI_LOG_I(TAG, "Radio stack started");
  203. ble_glue->status = BleGlueStatusRadioStackRunning;
  204. ret = true;
  205. if(SHCI_C2_SetFlashActivityControl(FLASH_ACTIVITY_CONTROL_SEM7) == SHCI_Success) {
  206. FURI_LOG_I(TAG, "Flash activity control switched to SEM7");
  207. } else {
  208. FURI_LOG_E(TAG, "Failed to switch flash activity control to SEM7");
  209. }
  210. } else {
  211. FURI_LOG_E(TAG, "Radio stack startup failed");
  212. ble_glue->status = BleGlueStatusRadioStackMissing;
  213. ble_app_thread_stop();
  214. }
  215. return ret;
  216. }
  217. bool ble_glue_is_alive() {
  218. if(!ble_glue) {
  219. return false;
  220. }
  221. return ble_glue->status >= BleGlueStatusC2Started;
  222. }
  223. bool ble_glue_is_radio_stack_ready() {
  224. if(!ble_glue) {
  225. return false;
  226. }
  227. return ble_glue->status == BleGlueStatusRadioStackRunning;
  228. }
  229. BleGlueCommandResult ble_glue_force_c2_mode(BleGlueC2Mode desired_mode) {
  230. furi_check(desired_mode > BleGlueC2ModeUnknown);
  231. if(desired_mode == ble_glue->c2_info.mode) {
  232. return BleGlueCommandResultOK;
  233. }
  234. if((ble_glue->c2_info.mode == BleGlueC2ModeFUS) && (desired_mode == BleGlueC2ModeStack)) {
  235. if((ble_glue->c2_info.VersionMajor == 0) && (ble_glue->c2_info.VersionMinor == 0)) {
  236. FURI_LOG_W(TAG, "Stack isn't installed!");
  237. return BleGlueCommandResultError;
  238. }
  239. SHCI_CmdStatus_t status = SHCI_C2_FUS_StartWs();
  240. if(status) {
  241. FURI_LOG_E(TAG, "Failed to start Radio Stack with status: %02X", status);
  242. return BleGlueCommandResultError;
  243. }
  244. return BleGlueCommandResultRestartPending;
  245. }
  246. if((ble_glue->c2_info.mode == BleGlueC2ModeStack) && (desired_mode == BleGlueC2ModeFUS)) {
  247. SHCI_FUS_GetState_ErrorCode_t error_code = 0;
  248. uint8_t fus_state = SHCI_C2_FUS_GetState(&error_code);
  249. FURI_LOG_D(TAG, "FUS state: %X, error = %x", fus_state, error_code);
  250. if(fus_state == SHCI_FUS_CMD_NOT_SUPPORTED) {
  251. // Second call to SHCI_C2_FUS_GetState() restarts whole MCU & boots FUS
  252. fus_state = SHCI_C2_FUS_GetState(&error_code);
  253. FURI_LOG_D(TAG, "FUS state#2: %X, error = %x", fus_state, error_code);
  254. return BleGlueCommandResultRestartPending;
  255. }
  256. return BleGlueCommandResultOK;
  257. }
  258. return BleGlueCommandResultError;
  259. }
  260. static void ble_glue_sys_status_not_callback(SHCI_TL_CmdStatus_t status) {
  261. switch(status) {
  262. case SHCI_TL_CmdBusy:
  263. furi_mutex_acquire(ble_glue->shci_mtx, FuriWaitForever);
  264. break;
  265. case SHCI_TL_CmdAvailable:
  266. furi_mutex_release(ble_glue->shci_mtx);
  267. break;
  268. default:
  269. break;
  270. }
  271. }
  272. /*
  273. * The type of the payload for a system user event is tSHCI_UserEvtRxParam
  274. * When the system event is both :
  275. * - a ready event (subevtcode = SHCI_SUB_EVT_CODE_READY)
  276. * - reported by the FUS (sysevt_ready_rsp == FUS_FW_RUNNING)
  277. * The buffer shall not be released
  278. * ( eg ((tSHCI_UserEvtRxParam*)pPayload)->status shall be set to SHCI_TL_UserEventFlow_Disable )
  279. * When the status is not filled, the buffer is released by default
  280. */
  281. static void ble_glue_sys_user_event_callback(void* pPayload) {
  282. UNUSED(pPayload);
  283. #ifdef BLE_GLUE_DEBUG
  284. APPD_EnableCPU2();
  285. #endif
  286. TL_AsynchEvt_t* p_sys_event =
  287. (TL_AsynchEvt_t*)(((tSHCI_UserEvtRxParam*)pPayload)->pckt->evtserial.evt.payload);
  288. if(p_sys_event->subevtcode == SHCI_SUB_EVT_CODE_READY) {
  289. FURI_LOG_I(TAG, "Core2 started");
  290. SHCI_C2_Ready_Evt_t* p_c2_ready_evt = (SHCI_C2_Ready_Evt_t*)p_sys_event->payload;
  291. if(p_c2_ready_evt->sysevt_ready_rsp == WIRELESS_FW_RUNNING) {
  292. ble_glue->c2_info.mode = BleGlueC2ModeStack;
  293. } else if(p_c2_ready_evt->sysevt_ready_rsp == FUS_FW_RUNNING) {
  294. ble_glue->c2_info.mode = BleGlueC2ModeFUS;
  295. }
  296. ble_glue->status = BleGlueStatusC2Started;
  297. } else if(p_sys_event->subevtcode == SHCI_SUB_EVT_ERROR_NOTIF) {
  298. FURI_LOG_E(TAG, "Error during initialization");
  299. } else if(p_sys_event->subevtcode == SHCI_SUB_EVT_BLE_NVM_RAM_UPDATE) {
  300. SHCI_C2_BleNvmRamUpdate_Evt_t* p_sys_ble_nvm_ram_update_event =
  301. (SHCI_C2_BleNvmRamUpdate_Evt_t*)p_sys_event->payload;
  302. if(ble_glue->callback) {
  303. ble_glue->callback(
  304. (uint8_t*)p_sys_ble_nvm_ram_update_event->StartAddress,
  305. p_sys_ble_nvm_ram_update_event->Size,
  306. ble_glue->context);
  307. }
  308. }
  309. }
  310. static void ble_glue_clear_shared_memory() {
  311. memset(ble_glue_event_pool, 0, sizeof(ble_glue_event_pool));
  312. memset(&ble_glue_system_cmd_buff, 0, sizeof(ble_glue_system_cmd_buff));
  313. memset(ble_glue_system_spare_event_buff, 0, sizeof(ble_glue_system_spare_event_buff));
  314. memset(ble_glue_ble_spare_event_buff, 0, sizeof(ble_glue_ble_spare_event_buff));
  315. }
  316. void ble_glue_thread_stop() {
  317. if(ble_glue) {
  318. FuriThreadId thread_id = furi_thread_get_id(ble_glue->thread);
  319. furi_assert(thread_id);
  320. furi_thread_flags_set(thread_id, BLE_GLUE_FLAG_KILL_THREAD);
  321. furi_thread_join(ble_glue->thread);
  322. furi_thread_free(ble_glue->thread);
  323. // Free resources
  324. furi_mutex_free(ble_glue->shci_mtx);
  325. furi_semaphore_free(ble_glue->shci_sem);
  326. ble_glue_clear_shared_memory();
  327. free(ble_glue);
  328. ble_glue = NULL;
  329. }
  330. }
  331. // Wrap functions
  332. static int32_t ble_glue_shci_thread(void* context) {
  333. UNUSED(context);
  334. uint32_t flags = 0;
  335. while(true) {
  336. flags = furi_thread_flags_wait(BLE_GLUE_FLAG_ALL, FuriFlagWaitAny, FuriWaitForever);
  337. if(flags & BLE_GLUE_FLAG_SHCI_EVENT) {
  338. shci_user_evt_proc();
  339. }
  340. if(flags & BLE_GLUE_FLAG_KILL_THREAD) {
  341. break;
  342. }
  343. }
  344. return 0;
  345. }
  346. void shci_notify_asynch_evt(void* pdata) {
  347. UNUSED(pdata);
  348. if(ble_glue) {
  349. FuriThreadId thread_id = furi_thread_get_id(ble_glue->thread);
  350. furi_assert(thread_id);
  351. furi_thread_flags_set(thread_id, BLE_GLUE_FLAG_SHCI_EVENT);
  352. }
  353. }
  354. void shci_cmd_resp_release(uint32_t flag) {
  355. UNUSED(flag);
  356. if(ble_glue) {
  357. furi_semaphore_release(ble_glue->shci_sem);
  358. }
  359. }
  360. void shci_cmd_resp_wait(uint32_t timeout) {
  361. UNUSED(timeout);
  362. if(ble_glue) {
  363. furi_hal_power_insomnia_enter();
  364. furi_semaphore_acquire(ble_glue->shci_sem, FuriWaitForever);
  365. furi_hal_power_insomnia_exit();
  366. }
  367. }
  368. bool ble_glue_reinit_c2() {
  369. return SHCI_C2_Reinit() == SHCI_Success;
  370. }
  371. BleGlueCommandResult ble_glue_fus_stack_delete() {
  372. FURI_LOG_I(TAG, "Erasing stack");
  373. SHCI_CmdStatus_t erase_stat = SHCI_C2_FUS_FwDelete();
  374. FURI_LOG_I(TAG, "Cmd res = %x", erase_stat);
  375. if(erase_stat == SHCI_Success) {
  376. return BleGlueCommandResultOperationOngoing;
  377. }
  378. ble_glue_fus_get_status();
  379. return BleGlueCommandResultError;
  380. }
  381. BleGlueCommandResult ble_glue_fus_stack_install(uint32_t src_addr, uint32_t dst_addr) {
  382. FURI_LOG_I(TAG, "Installing stack");
  383. SHCI_CmdStatus_t write_stat = SHCI_C2_FUS_FwUpgrade(src_addr, dst_addr);
  384. FURI_LOG_I(TAG, "Cmd res = %x", write_stat);
  385. if(write_stat == SHCI_Success) {
  386. return BleGlueCommandResultOperationOngoing;
  387. }
  388. ble_glue_fus_get_status();
  389. return BleGlueCommandResultError;
  390. }
  391. BleGlueCommandResult ble_glue_fus_get_status() {
  392. furi_check(ble_glue->c2_info.mode == BleGlueC2ModeFUS);
  393. SHCI_FUS_GetState_ErrorCode_t error_code = 0;
  394. uint8_t fus_state = SHCI_C2_FUS_GetState(&error_code);
  395. FURI_LOG_I(TAG, "FUS state: %x, error: %x", fus_state, error_code);
  396. if((error_code != 0) || (fus_state == FUS_STATE_VALUE_ERROR)) {
  397. return BleGlueCommandResultError;
  398. } else if(
  399. (fus_state >= FUS_STATE_VALUE_FW_UPGRD_ONGOING) &&
  400. (fus_state <= FUS_STATE_VALUE_SERVICE_ONGOING_END)) {
  401. return BleGlueCommandResultOperationOngoing;
  402. }
  403. return BleGlueCommandResultOK;
  404. }
  405. BleGlueCommandResult ble_glue_fus_wait_operation() {
  406. furi_check(ble_glue->c2_info.mode == BleGlueC2ModeFUS);
  407. while(true) {
  408. BleGlueCommandResult fus_status = ble_glue_fus_get_status();
  409. if(fus_status == BleGlueCommandResultOperationOngoing) {
  410. furi_delay_ms(20);
  411. } else if(fus_status == BleGlueCommandResultError) {
  412. return BleGlueCommandResultError;
  413. } else {
  414. return BleGlueCommandResultOK;
  415. }
  416. }
  417. }