uhf_main.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364
  1. #include <furi.h>
  2. #include <furi_hal.h>
  3. #include <gui/gui.h>
  4. #include <gui/view_dispatcher.h>
  5. #include <gui/scene_manager.h>
  6. #include <gui/modules/widget.h>
  7. #include <gui/modules/submenu.h>
  8. #include <gui/modules/text_input.h>
  9. #include <lib/toolbox/stream/file_stream.h>
  10. #include "cmd.h"
  11. #include <stdlib.h>
  12. #define UHF_APPS_DATA_FOLDER EXT_PATH("apps_data")
  13. #define UHF_LOG_FOLDER \
  14. UHF_APPS_DATA_FOLDER "/" \
  15. "uhf"
  16. #define UHF_LOG_FILE_NAME "log.txt"
  17. #define UHF_LOG_PATH UHF_LOG_FOLDER "/" UHF_LOG_FILE_NAME
  18. #define SCENE_COUNT 2
  19. bool logged = false;
  20. // volatile int array_ptr = 0;
  21. // uint8_t array_log[] = {0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
  22. // 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF};
  23. char* convertToHexString(const uint8_t* array, size_t length) {
  24. if(array == NULL || length == 0) {
  25. return NULL;
  26. }
  27. // Each byte takes 3 characters in the hex representation (2 characters + space), plus 1 for the null terminator
  28. size_t hexLength = (length * 3) + 1;
  29. char* hexArray = (char*)malloc(hexLength * sizeof(char));
  30. if(hexArray == NULL) {
  31. return NULL;
  32. }
  33. size_t index = 0;
  34. for(size_t i = 0; i < length; i++) {
  35. index += snprintf(&hexArray[index], hexLength - index, "%02x ", array[i]);
  36. }
  37. hexArray[hexLength - 1] = '\0';
  38. return hexArray;
  39. }
  40. void store_message(char* message) {
  41. Storage* storage = furi_record_open(RECORD_STORAGE);
  42. if(!storage_dir_exists(storage, UHF_APPS_DATA_FOLDER)) {
  43. storage_simply_mkdir(storage, UHF_APPS_DATA_FOLDER);
  44. }
  45. if(!storage_dir_exists(storage, UHF_LOG_FOLDER)) {
  46. storage_simply_mkdir(storage, UHF_LOG_FOLDER);
  47. }
  48. File* file = storage_file_alloc(storage);
  49. if(storage_file_open(file, UHF_LOG_PATH, FSAM_WRITE, FSOM_OPEN_APPEND)) {
  50. if(!logged) {
  51. storage_file_seek(file, 0, true);
  52. storage_file_truncate(file);
  53. logged = true;
  54. }
  55. storage_file_write(file, message, strlen(message));
  56. storage_file_write(file, "\n", 1);
  57. storage_file_close(file);
  58. storage_file_free(file);
  59. }
  60. furi_record_close(RECORD_STORAGE);
  61. }
  62. typedef enum { STARTED, RUNNING, STOPPED } UHFRFIDWorkerState;
  63. typedef enum { SEND_READ_CMD, SEND_WRITE_CMD } UHFRFIDWorkerCMD;
  64. // typedef void (*UHFRFIDWorkerCallback)();
  65. typedef struct UHFReturnData {
  66. uint8_t* data;
  67. size_t length;
  68. } UHFReturnData;
  69. UHFReturnData* uhf_return_data_alloc() {
  70. UHFReturnData* uhf_data = (UHFReturnData*)malloc(sizeof(UHFReturnData));
  71. uhf_data->data = (uint8_t*)malloc(24 * sizeof(uint8_t));
  72. uhf_data->length = 0;
  73. return uhf_data;
  74. }
  75. void uhf_return_data_free(UHFReturnData* uhf_data) {
  76. free(uhf_data->data);
  77. }
  78. typedef struct UHFRFIDWorker {
  79. FuriThread* thread;
  80. UHFReturnData* uhf_data;
  81. // UHFRFIDWorkerCMD cmd;
  82. // Storage* storage;
  83. // UHFRFIDWorkerCallback callback;
  84. } UHFRFIDWorker;
  85. typedef enum { MainMenuRead, MainMenuSaved } MainMenuSceneIndex;
  86. typedef enum { MainMenuReadEvent, MainMenuSavedEvent } MainMenuEvent;
  87. void uhf_rfid_worker_enable_field() {
  88. }
  89. void uhf_rfid_worker_disable_field() {
  90. }
  91. // uint8_t* uhf_rfid_read() {
  92. // return 0;
  93. // }
  94. void rx_callback(UartIrqEvent event, uint8_t data, void* ctx) {
  95. UNUSED(event);
  96. UHFReturnData* uhf_data = ctx;
  97. uhf_data->data[uhf_data->length++] = data;
  98. // UNUSED(ctx);
  99. // UNUSED(data);
  100. // array_log[array_ptr++] = data;
  101. }
  102. int32_t uhf_rfid_worker_task(void* ctx) {
  103. UHFRFIDWorker* worker = ctx;
  104. // UHFRFIDWorker* worker = ctx;
  105. // uhf_rfid_worker_enable_field();
  106. // array_ptr = 0;
  107. furi_hal_console_disable();
  108. furi_hal_uart_set_br(FuriHalUartIdUSART1, 115200);
  109. furi_hal_uart_set_irq_cb(FuriHalUartIdUSART1, rx_callback, worker->uhf_data);
  110. // ArrayList* data_list = worker->context;
  111. // int prev_size = data_list->size;
  112. // switch(worker->cmd) {
  113. // case SEND_READ_CMD:
  114. // // furi_hal_console_enable();
  115. // // FURI_LOG_E("TAG", "SENDING DATA");
  116. // // FURI_LOG_E(
  117. // // "TAG",
  118. // // "%02x%02x%02x%02x%02x%02x%02x",
  119. // // CMD_SINGLE_POLLING.cmd[0],
  120. // // CMD_SINGLE_POLLING.cmd[1],
  121. // // CMD_SINGLE_POLLING.cmd[2],
  122. // // CMD_SINGLE_POLLING.cmd[3],
  123. // // CMD_SINGLE_POLLING.cmd[4],
  124. // // CMD_SINGLE_POLLING.cmd[5],
  125. // // CMD_SINGLE_POLLING.cmd[6]);
  126. // // furi_hal_console_disable();
  127. // break;
  128. // default:
  129. // break;
  130. // }
  131. furi_hal_uart_tx(FuriHalUartIdUSART1, CMD_SINGLE_POLLING.cmd, CMD_SINGLE_POLLING.length);
  132. // while(prev_size == 0 || !(prev_size != data_list->size)) {
  133. // furi_delay_ms(50);
  134. // }
  135. // furi_hal_console_enable();
  136. // FURI_LOG_E("TAG", "DATA READ %s", array_list_to_string(data_list));
  137. // furi_delay_ms(2000);
  138. return 0;
  139. }
  140. UHFRFIDWorker* uhf_rfid_worker_alloc() {
  141. UHFRFIDWorker* uhf_rfid_worker = malloc(sizeof(UHFRFIDWorker));
  142. uhf_rfid_worker->thread =
  143. furi_thread_alloc_ex("UHFRFIDWorker", 8 * 1024, uhf_rfid_worker_task, uhf_rfid_worker);
  144. // uhf_rfid_worker->cmd = SEND_READ_CMD;
  145. uhf_rfid_worker->uhf_data = uhf_return_data_alloc();
  146. return uhf_rfid_worker;
  147. }
  148. void uhf_rfid_worker_free(UHFRFIDWorker* uhf_rfid_worker) {
  149. furi_assert(uhf_rfid_worker);
  150. furi_thread_free(uhf_rfid_worker->thread);
  151. // ArrayList* data_list = uhf_rfid_worker->context;
  152. // array_list_free(data_list);
  153. // furi_record_close(RECORD_STORAGE);
  154. free(uhf_rfid_worker);
  155. }
  156. // void uhf_rfid_worker_change_state(UHFRFIDWorker* worker, UHFRFIDWorkerState state) {
  157. // worker->state = state;
  158. // }
  159. void uhf_rfid_worker_start(UHFRFIDWorker* uhf_rfid_worker, UHFRFIDWorkerState state, void* ctx) {
  160. UNUSED(state);
  161. UNUSED(ctx);
  162. // uhf_rfid_worker_change_state(uhf_rfid_worker, state);
  163. furi_thread_start(uhf_rfid_worker->thread);
  164. }
  165. void uhf_rfid_worker_stop(UHFRFIDWorker* uhf_rfid_worker) {
  166. furi_assert(uhf_rfid_worker);
  167. furi_assert(uhf_rfid_worker->thread);
  168. if(furi_thread_get_state(uhf_rfid_worker->thread) != FuriThreadStateStopped) {
  169. // uhf_rfid_worker_change_state(uhf_rfid_worker, STOPPED);
  170. furi_thread_join(uhf_rfid_worker->thread);
  171. }
  172. }
  173. typedef enum { MainMenuScene, WaitingReadScene, ReadSuccessScene } UHF_RFID_Scene;
  174. typedef enum { SubmenuView, WidgetView, TextInputView } UHF_RFID_Scene_View;
  175. typedef struct UHFRFIDApp {
  176. SceneManager* scene_manager;
  177. ViewDispatcher* view_dispatcher;
  178. Submenu* submenu;
  179. Widget* widget;
  180. TextInput* text_input;
  181. } UHFRFIDApp;
  182. void scenes_menu_callback(void* ctx, uint32_t index) {
  183. UHFRFIDApp* app = ctx;
  184. switch(index) {
  185. case MainMenuRead:
  186. scene_manager_handle_custom_event(app->scene_manager, MainMenuReadEvent);
  187. break;
  188. case MainMenuSaved:
  189. // not implemented yet
  190. furi_hal_console_enable();
  191. FURI_LOG_E("TAG", "Saved was pressed");
  192. furi_hal_console_disable();
  193. break;
  194. }
  195. }
  196. void scene_main_menu_on_enter(void* ctx) {
  197. UHFRFIDApp* app = ctx;
  198. submenu_reset(app->submenu);
  199. submenu_set_header(app->submenu, "[UHF]RFID App");
  200. submenu_add_item(app->submenu, "Read", MainMenuRead, scenes_menu_callback, app);
  201. submenu_add_item(app->submenu, "Saved", MainMenuSaved, scenes_menu_callback, app);
  202. view_dispatcher_switch_to_view(app->view_dispatcher, SubmenuView);
  203. }
  204. bool scene_main_menu_on_event(void* ctx, SceneManagerEvent event) {
  205. UHFRFIDApp* app = ctx;
  206. bool consumed = false;
  207. switch(event.type) {
  208. case SceneManagerEventTypeCustom:
  209. switch(event.event) {
  210. case MainMenuReadEvent:
  211. scene_manager_next_scene(app->scene_manager, WaitingReadScene);
  212. consumed = true;
  213. break;
  214. case MainMenuSavedEvent:
  215. // TODO : not implemented yet
  216. break;
  217. }
  218. default:
  219. break;
  220. }
  221. return consumed;
  222. }
  223. void scene_main_menu_on_exit(void* ctx) {
  224. UHFRFIDApp* app = ctx;
  225. submenu_reset(app->submenu);
  226. }
  227. void scene_read_on_enter(void* ctx) {
  228. UHFRFIDApp* app = ctx;
  229. widget_reset(app->widget);
  230. UHFRFIDWorker* worker = uhf_rfid_worker_alloc();
  231. UHFReturnData* uhf_data = worker->uhf_data;
  232. uhf_rfid_worker_start(worker, RUNNING, NULL);
  233. furi_delay_ms(1000);
  234. // UNUSED(uhf_data);
  235. char* values = convertToHexString(uhf_data->data, uhf_data->length);
  236. // UNUSED(values);
  237. if(values != NULL) {
  238. store_message(values);
  239. widget_add_string_element(
  240. app->widget, 25, 15, AlignCenter, AlignCenter, FontPrimary, values);
  241. } else {
  242. store_message("No data found");
  243. widget_add_string_element(
  244. app->widget, 25, 15, AlignCenter, AlignCenter, FontPrimary, "No data found");
  245. }
  246. // furi_hal_console_enable();
  247. // FURI_LOG_E("TAG", "data %s", values);
  248. view_dispatcher_switch_to_view(app->view_dispatcher, WidgetView);
  249. // furi_delay_ms(2000);
  250. uhf_rfid_worker_stop(worker);
  251. uhf_return_data_free(worker->uhf_data);
  252. uhf_rfid_worker_free(worker);
  253. }
  254. bool scene_read_on_event(void* ctx, SceneManagerEvent event) {
  255. UNUSED(ctx);
  256. UNUSED(event);
  257. return false; // event not handled.
  258. }
  259. void scene_read_on_exit(void* ctx) {
  260. UNUSED(ctx);
  261. }
  262. bool scene_custom_callback(void* ctx, uint32_t custom_event) {
  263. furi_assert(ctx);
  264. UHFRFIDApp* app = ctx;
  265. return scene_manager_handle_custom_event(app->scene_manager, custom_event);
  266. };
  267. bool scene_back_event_callback(void* ctx) {
  268. furi_assert(ctx);
  269. UHFRFIDApp* app = ctx;
  270. return scene_manager_handle_back_event(app->scene_manager);
  271. }
  272. void (*const scene_on_enter_handlers[])(void*) = {scene_main_menu_on_enter, scene_read_on_enter};
  273. bool (*const scene_on_event_handlers[])(void*, SceneManagerEvent) = {
  274. scene_main_menu_on_event,
  275. scene_read_on_event};
  276. void (*const scene_on_exit_handlers[])(void*) = {scene_main_menu_on_exit, scene_read_on_exit};
  277. const SceneManagerHandlers scene_manager_handlers = {
  278. .on_enter_handlers = scene_on_enter_handlers,
  279. .on_event_handlers = scene_on_event_handlers,
  280. .on_exit_handlers = scene_on_exit_handlers,
  281. .scene_num = SCENE_COUNT};
  282. UHFRFIDApp* uhf_rfid_app_alloc() {
  283. UHFRFIDApp* app = malloc(sizeof(UHFRFIDApp));
  284. app->scene_manager = scene_manager_alloc(&scene_manager_handlers, app);
  285. app->view_dispatcher = view_dispatcher_alloc();
  286. view_dispatcher_enable_queue(app->view_dispatcher);
  287. view_dispatcher_set_event_callback_context(app->view_dispatcher, app);
  288. view_dispatcher_set_custom_event_callback(app->view_dispatcher, scene_custom_callback);
  289. view_dispatcher_set_navigation_event_callback(app->view_dispatcher, scene_back_event_callback);
  290. app->submenu = submenu_alloc();
  291. view_dispatcher_add_view(app->view_dispatcher, SubmenuView, submenu_get_view(app->submenu));
  292. app->widget = widget_alloc();
  293. view_dispatcher_add_view(app->view_dispatcher, WidgetView, widget_get_view(app->widget));
  294. return app;
  295. }
  296. void uhf_rfid_app_free(UHFRFIDApp* app) {
  297. furi_assert(app);
  298. view_dispatcher_remove_view(app->view_dispatcher, SubmenuView);
  299. view_dispatcher_remove_view(app->view_dispatcher, WidgetView);
  300. scene_manager_free(app->scene_manager);
  301. view_dispatcher_free(app->view_dispatcher);
  302. submenu_free(app->submenu);
  303. widget_free(app->widget);
  304. free(app);
  305. }
  306. int32_t uhf_rfid_app2(void* p) {
  307. UNUSED(p);
  308. UHFRFIDApp* app = uhf_rfid_app_alloc();
  309. Gui* gui = furi_record_open(RECORD_GUI);
  310. view_dispatcher_attach_to_gui(app->view_dispatcher, gui, ViewDispatcherTypeFullscreen);
  311. scene_manager_next_scene(app->scene_manager, MainMenuScene);
  312. view_dispatcher_run(app->view_dispatcher);
  313. uhf_rfid_app_free(app);
  314. return 0;
  315. }