nfc.c 10 KB

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