nfc.c 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228
  1. #include "nfc_i.h"
  2. #include "api-hal-nfc.h"
  3. osMessageQueueId_t message_queue = NULL;
  4. uint32_t nfc_view_stop(void* context) {
  5. furi_assert(message_queue);
  6. NfcMessage message;
  7. message.type = NfcMessageTypeStop;
  8. furi_check(osMessageQueuePut(message_queue, &message, 0, osWaitForever) == osOK);
  9. return NfcViewMenu;
  10. }
  11. uint32_t nfc_view_exit(void* context) {
  12. furi_assert(message_queue);
  13. NfcMessage message;
  14. message.type = NfcMessageTypeExit;
  15. furi_check(osMessageQueuePut(message_queue, &message, 0, osWaitForever) == osOK);
  16. return VIEW_NONE;
  17. }
  18. void nfc_menu_callback(void* context, uint32_t index) {
  19. furi_assert(message_queue);
  20. NfcMessage message;
  21. if(index == 0) {
  22. message.type = NfcMessageTypeDetect;
  23. } else if(index == 1) {
  24. message.type = NfcMessageTypeEmulate;
  25. } else if(index == 2) {
  26. message.type = NfcMessageTypeField;
  27. }
  28. furi_check(osMessageQueuePut(message_queue, &message, 0, osWaitForever) == osOK);
  29. }
  30. Nfc* nfc_alloc() {
  31. Nfc* nfc = furi_alloc(sizeof(Nfc));
  32. message_queue = osMessageQueueNew(8, sizeof(NfcMessage), NULL);
  33. nfc->worker = nfc_worker_alloc(message_queue);
  34. // Open GUI record
  35. nfc->gui = furi_record_open("gui");
  36. // View Dispatcher
  37. nfc->view_dispatcher = view_dispatcher_alloc();
  38. view_dispatcher_attach_to_gui(nfc->view_dispatcher, nfc->gui, ViewDispatcherTypeFullscreen);
  39. // Menu
  40. nfc->submenu = submenu_alloc();
  41. submenu_add_item(nfc->submenu, "Detect", 0, nfc_menu_callback, nfc);
  42. submenu_add_item(nfc->submenu, "Emulate", 1, nfc_menu_callback, nfc);
  43. submenu_add_item(nfc->submenu, "Field", 2, nfc_menu_callback, nfc);
  44. View* submenu_view = submenu_get_view(nfc->submenu);
  45. view_set_previous_callback(submenu_view, nfc_view_exit);
  46. view_dispatcher_add_view(nfc->view_dispatcher, NfcViewMenu, submenu_view);
  47. // Detect
  48. nfc->view_detect = view_alloc();
  49. view_set_context(nfc->view_detect, nfc);
  50. view_set_draw_callback(nfc->view_detect, nfc_view_read_draw);
  51. view_set_previous_callback(nfc->view_detect, nfc_view_stop);
  52. view_allocate_model(nfc->view_detect, ViewModelTypeLocking, sizeof(NfcViewReadModel));
  53. view_dispatcher_add_view(nfc->view_dispatcher, NfcViewRead, nfc->view_detect);
  54. // Emulate
  55. nfc->view_emulate = view_alloc();
  56. view_set_context(nfc->view_emulate, nfc);
  57. view_set_draw_callback(nfc->view_emulate, nfc_view_emulate_draw);
  58. view_set_previous_callback(nfc->view_emulate, nfc_view_stop);
  59. view_dispatcher_add_view(nfc->view_dispatcher, NfcViewEmulate, nfc->view_emulate);
  60. // Field
  61. nfc->view_field = view_alloc();
  62. view_set_context(nfc->view_field, nfc);
  63. view_set_draw_callback(nfc->view_field, nfc_view_field_draw);
  64. view_set_previous_callback(nfc->view_field, nfc_view_stop);
  65. view_dispatcher_add_view(nfc->view_dispatcher, NfcViewField, nfc->view_field);
  66. // Error
  67. nfc->view_error = view_alloc();
  68. view_set_context(nfc->view_error, nfc);
  69. view_set_draw_callback(nfc->view_error, nfc_view_error_draw);
  70. view_set_previous_callback(nfc->view_error, nfc_view_stop);
  71. view_allocate_model(nfc->view_error, ViewModelTypeLockFree, sizeof(NfcViewErrorModel));
  72. view_dispatcher_add_view(nfc->view_dispatcher, NfcViewError, nfc->view_error);
  73. // Switch to menu
  74. view_dispatcher_switch_to_view(nfc->view_dispatcher, NfcViewMenu);
  75. return nfc;
  76. }
  77. void nfc_free(Nfc* nfc) {
  78. // Free nfc worker
  79. nfc_worker_free(nfc->worker);
  80. // Free allocated queue
  81. osMessageQueueDelete(message_queue);
  82. message_queue = NULL;
  83. // Free allocated views
  84. // Menu
  85. view_dispatcher_remove_view(nfc->view_dispatcher, NfcViewMenu);
  86. submenu_free(nfc->submenu);
  87. // Detect
  88. view_dispatcher_remove_view(nfc->view_dispatcher, NfcViewRead);
  89. view_free(nfc->view_detect);
  90. // Emulate
  91. view_dispatcher_remove_view(nfc->view_dispatcher, NfcViewEmulate);
  92. view_free(nfc->view_emulate);
  93. // Field
  94. view_dispatcher_remove_view(nfc->view_dispatcher, NfcViewField);
  95. view_free(nfc->view_field);
  96. // Error
  97. view_dispatcher_remove_view(nfc->view_dispatcher, NfcViewError);
  98. view_free(nfc->view_error);
  99. // Free View Dispatcher
  100. view_dispatcher_free(nfc->view_dispatcher);
  101. // Close all opened records
  102. furi_record_close("gui");
  103. nfc->gui = NULL;
  104. // Free nfc object
  105. free(nfc);
  106. }
  107. void nfc_cli_detect(string_t args, void* context) {
  108. furi_assert(context);
  109. Cli* cli = context;
  110. // Check if nfc worker is not busy
  111. if(api_hal_nfc_is_busy()) {
  112. printf("Nfc is busy");
  113. return;
  114. }
  115. rfalNfcDevice* dev_list;
  116. uint8_t dev_cnt = 0;
  117. bool cmd_exit = false;
  118. api_hal_nfc_init();
  119. printf("Detecting nfc...\r\nPress Ctrl+C to abort\r\n");
  120. while(!cmd_exit) {
  121. cmd_exit |= cli_cmd_interrupt_received(cli);
  122. cmd_exit |= api_hal_nfc_detect(&dev_list, &dev_cnt, 100);
  123. if(dev_cnt > 0) {
  124. printf("Found %d devices\r\n", dev_cnt);
  125. for(uint8_t i = 0; i < dev_cnt; i++) {
  126. printf("%d found: %s ", i, nfc_get_dev_type(dev_list[i].type));
  127. if(dev_list[i].type == RFAL_NFC_LISTEN_TYPE_NFCA) {
  128. printf("type: %s, ", nfc_get_nfca_type(dev_list[i].dev.nfca.type));
  129. }
  130. printf("UID length: %d, UID:", dev_list[i].nfcidLen);
  131. for(uint8_t j = 0; j < dev_list[i].nfcidLen; j++) {
  132. printf("%02X", dev_list[i].nfcid[j]);
  133. }
  134. printf("\r\n");
  135. }
  136. }
  137. osDelay(50);
  138. }
  139. }
  140. void nfc_cli_init() {
  141. Cli* cli = furi_record_open("cli");
  142. cli_add_command(cli, "nfc_detect", nfc_cli_detect, cli);
  143. furi_record_close("cli");
  144. }
  145. void nfc_start(Nfc* nfc, NfcView view_id, NfcWorkerState worker_state) {
  146. NfcWorkerState state = nfc_worker_get_state(nfc->worker);
  147. if(state == NfcWorkerStateBroken) {
  148. with_view_model(
  149. nfc->view_error, (NfcViewErrorModel * model) {
  150. model->error = nfc_worker_get_error(nfc->worker);
  151. return true;
  152. });
  153. view_dispatcher_switch_to_view(nfc->view_dispatcher, NfcViewError);
  154. } else if(state == NfcWorkerStateReady) {
  155. view_dispatcher_switch_to_view(nfc->view_dispatcher, view_id);
  156. nfc_worker_start(nfc->worker, worker_state);
  157. }
  158. }
  159. int32_t nfc_task(void* p) {
  160. Nfc* nfc = nfc_alloc();
  161. NfcMessage message;
  162. while(1) {
  163. furi_check(osMessageQueueGet(message_queue, &message, NULL, osWaitForever) == osOK);
  164. if(message.type == NfcMessageTypeDetect) {
  165. with_view_model(
  166. nfc->view_detect, (NfcViewReadModel * model) {
  167. model->found = false;
  168. return true;
  169. });
  170. nfc_start(nfc, NfcViewRead, NfcWorkerStatePoll);
  171. } else if(message.type == NfcMessageTypeEmulate) {
  172. nfc_start(nfc, NfcViewEmulate, NfcWorkerStateEmulate);
  173. } else if(message.type == NfcMessageTypeField) {
  174. nfc_start(nfc, NfcViewField, NfcWorkerStateField);
  175. } else if(message.type == NfcMessageTypeStop) {
  176. nfc_worker_stop(nfc->worker);
  177. } else if(message.type == NfcMessageTypeDeviceFound) {
  178. with_view_model(
  179. nfc->view_detect, (NfcViewReadModel * model) {
  180. model->found = true;
  181. model->device = message.device;
  182. return true;
  183. });
  184. } else if(message.type == NfcMessageTypeDeviceNotFound) {
  185. with_view_model(
  186. nfc->view_detect, (NfcViewReadModel * model) {
  187. model->found = false;
  188. return true;
  189. });
  190. } else if(message.type == NfcMessageTypeExit) {
  191. nfc_free(nfc);
  192. break;
  193. }
  194. }
  195. return 0;
  196. }