seader_worker.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382
  1. #include "seader_worker_i.h"
  2. #include <flipper_format/flipper_format.h>
  3. #include <lib/bit_lib/bit_lib.h>
  4. #define TAG "SeaderWorker"
  5. #define APDU_HEADER_LEN 5
  6. #define ASN1_PREFIX 6
  7. #define ASN1_DEBUG true
  8. #define RFAL_PICOPASS_TXRX_FLAGS \
  9. (FURI_HAL_NFC_LL_TXRX_FLAGS_CRC_TX_MANUAL | FURI_HAL_NFC_LL_TXRX_FLAGS_AGC_ON | \
  10. FURI_HAL_NFC_LL_TXRX_FLAGS_PAR_RX_REMV | FURI_HAL_NFC_LL_TXRX_FLAGS_CRC_RX_KEEP)
  11. char display[SEADER_UART_RX_BUF_SIZE * 2 + 1] = {0};
  12. // Forward declaration
  13. void seader_send_card_detected(SeaderUartBridge* seader_uart, CardDetails_t* cardDetails);
  14. /***************************** Seader Worker API *******************************/
  15. SeaderWorker* seader_worker_alloc() {
  16. SeaderWorker* seader_worker = malloc(sizeof(SeaderWorker));
  17. // Worker thread attributes
  18. seader_worker->thread =
  19. furi_thread_alloc_ex("SeaderWorker", 8192, seader_worker_task, seader_worker);
  20. seader_worker->messages = furi_message_queue_alloc(3, sizeof(SeaderAPDU));
  21. seader_worker->mq_mutex = furi_mutex_alloc(FuriMutexTypeNormal);
  22. seader_worker->callback = NULL;
  23. seader_worker->context = NULL;
  24. seader_worker->storage = furi_record_open(RECORD_STORAGE);
  25. memset(seader_worker->sam_version, 0, sizeof(seader_worker->sam_version));
  26. seader_worker_change_state(seader_worker, SeaderWorkerStateReady);
  27. return seader_worker;
  28. }
  29. void seader_worker_free(SeaderWorker* seader_worker) {
  30. furi_assert(seader_worker);
  31. furi_thread_free(seader_worker->thread);
  32. furi_message_queue_free(seader_worker->messages);
  33. furi_mutex_free(seader_worker->mq_mutex);
  34. furi_record_close(RECORD_STORAGE);
  35. free(seader_worker);
  36. }
  37. SeaderWorkerState seader_worker_get_state(SeaderWorker* seader_worker) {
  38. return seader_worker->state;
  39. }
  40. void seader_worker_start(
  41. SeaderWorker* seader_worker,
  42. SeaderWorkerState state,
  43. SeaderUartBridge* uart,
  44. SeaderWorkerCallback callback,
  45. void* context) {
  46. furi_assert(seader_worker);
  47. furi_assert(uart);
  48. seader_worker->stage = SeaderPollerEventTypeCardDetect;
  49. seader_worker->callback = callback;
  50. seader_worker->context = context;
  51. seader_worker->uart = uart;
  52. seader_worker_change_state(seader_worker, state);
  53. furi_thread_start(seader_worker->thread);
  54. }
  55. void seader_worker_stop(SeaderWorker* seader_worker) {
  56. furi_assert(seader_worker);
  57. if(seader_worker->state == SeaderWorkerStateBroken ||
  58. seader_worker->state == SeaderWorkerStateReady) {
  59. return;
  60. }
  61. seader_worker_change_state(seader_worker, SeaderWorkerStateStop);
  62. furi_thread_join(seader_worker->thread);
  63. }
  64. void seader_worker_change_state(SeaderWorker* seader_worker, SeaderWorkerState state) {
  65. seader_worker->state = state;
  66. }
  67. /***************************** Seader Worker Thread *******************************/
  68. bool seader_process_success_response(Seader* seader, uint8_t* apdu, size_t len) {
  69. SeaderWorker* seader_worker = seader->worker;
  70. if(seader_process_success_response_i(seader, apdu, len, false, NULL)) {
  71. // no-op, message was processed
  72. } else {
  73. FURI_LOG_I(TAG, "Enqueue SAM message, %d bytes", len);
  74. uint32_t space = furi_message_queue_get_space(seader_worker->messages);
  75. if(space > 0) {
  76. SeaderAPDU seaderApdu = {};
  77. seaderApdu.len = len;
  78. memcpy(seaderApdu.buf, apdu, len);
  79. if(furi_mutex_acquire(seader_worker->mq_mutex, FuriWaitForever) == FuriStatusOk) {
  80. furi_message_queue_put(seader_worker->messages, &seaderApdu, FuriWaitForever);
  81. furi_mutex_release(seader_worker->mq_mutex);
  82. }
  83. }
  84. }
  85. return true;
  86. }
  87. bool seader_worker_process_sam_message(Seader* seader, uint8_t* apdu, uint32_t len) {
  88. SeaderWorker* seader_worker = seader->worker;
  89. SeaderUartBridge* seader_uart = seader_worker->uart;
  90. if(len < 2) {
  91. return false;
  92. }
  93. memset(display, 0, sizeof(display));
  94. for(uint8_t i = 0; i < len; i++) {
  95. snprintf(display + (i * 2), sizeof(display), "%02x", apdu[i]);
  96. }
  97. FURI_LOG_I(TAG, "APDU: %s", display);
  98. uint8_t SW1 = apdu[len - 2];
  99. uint8_t SW2 = apdu[len - 1];
  100. uint8_t GET_RESPONSE[] = {0x00, 0xc0, 0x00, 0x00, 0xff};
  101. switch(SW1) {
  102. case 0x61:
  103. // FURI_LOG_I(TAG, "Request %d bytes", SW2);
  104. GET_RESPONSE[4] = SW2;
  105. seader_ccid_XfrBlock(seader_uart, GET_RESPONSE, sizeof(GET_RESPONSE));
  106. return true;
  107. break;
  108. case 0x90:
  109. if(SW2 == 0x00) {
  110. if(len > 2) {
  111. return seader_process_success_response(seader, apdu, len - 2);
  112. }
  113. }
  114. break;
  115. default:
  116. FURI_LOG_W(TAG, "Unknown SW %02x%02x", SW1, SW2);
  117. break;
  118. }
  119. return false;
  120. }
  121. void seader_worker_virtual_credential(Seader* seader) {
  122. SeaderWorker* seader_worker = seader->worker;
  123. // Detect card
  124. seader_worker_card_detect(
  125. seader, 0, NULL, seader->credential->diversifier, sizeof(PicopassSerialNum), NULL, 0);
  126. bool running = true;
  127. // Max times the loop will run with no message to process
  128. uint8_t dead_loops = 20;
  129. while(running) {
  130. if(furi_mutex_acquire(seader_worker->mq_mutex, 0) == FuriStatusOk) {
  131. uint32_t count = furi_message_queue_get_count(seader_worker->messages);
  132. if(count > 0) {
  133. FURI_LOG_I(TAG, "Dequeue SAM message [%ld messages]", count);
  134. SeaderAPDU seaderApdu = {};
  135. FuriStatus status =
  136. furi_message_queue_get(seader_worker->messages, &seaderApdu, FuriWaitForever);
  137. if(status != FuriStatusOk) {
  138. FURI_LOG_W(TAG, "furi_message_queue_get fail %d", status);
  139. view_dispatcher_send_custom_event(
  140. seader->view_dispatcher, SeaderCustomEventWorkerExit);
  141. }
  142. if(seader_process_success_response_i(
  143. seader, seaderApdu.buf, seaderApdu.len, true, NULL)) {
  144. // no-op
  145. } else {
  146. FURI_LOG_I(TAG, "Response false");
  147. running = false;
  148. }
  149. }
  150. furi_mutex_release(seader_worker->mq_mutex);
  151. } else {
  152. dead_loops--;
  153. running = (dead_loops > 0);
  154. FURI_LOG_D(
  155. TAG, "Dead loops: %d -> Running: %s", dead_loops, running ? "true" : "false");
  156. }
  157. running = (seader_worker->stage != SeaderPollerEventTypeComplete);
  158. }
  159. if(dead_loops > 0) {
  160. FURI_LOG_D(TAG, "Final dead loops: %d", dead_loops);
  161. } else {
  162. view_dispatcher_send_custom_event(seader->view_dispatcher, SeaderCustomEventWorkerExit);
  163. }
  164. }
  165. int32_t seader_worker_task(void* context) {
  166. SeaderWorker* seader_worker = context;
  167. Seader* seader = seader_worker->context;
  168. SeaderUartBridge* seader_uart = seader_worker->uart;
  169. if(seader_worker->state == SeaderWorkerStateCheckSam) {
  170. FURI_LOG_D(TAG, "Check for SAM");
  171. seader_ccid_check_for_sam(seader_uart);
  172. } else if(seader_worker->state == SeaderWorkerStateVirtualCredential) {
  173. FURI_LOG_D(TAG, "Virtual Credential");
  174. seader_worker_virtual_credential(seader);
  175. }
  176. seader_worker_change_state(seader_worker, SeaderWorkerStateReady);
  177. return 0;
  178. }
  179. void seader_worker_poller_conversation(Seader* seader, SeaderPollerContainer* spc) {
  180. SeaderWorker* seader_worker = seader->worker;
  181. if(furi_mutex_acquire(seader_worker->mq_mutex, 0) == FuriStatusOk) {
  182. furi_thread_set_current_priority(FuriThreadPriorityHighest);
  183. uint32_t count = furi_message_queue_get_count(seader_worker->messages);
  184. if(count > 0) {
  185. FURI_LOG_I(TAG, "Dequeue SAM message [%ld messages]", count);
  186. SeaderAPDU seaderApdu = {};
  187. FuriStatus status =
  188. furi_message_queue_get(seader_worker->messages, &seaderApdu, FuriWaitForever);
  189. if(status != FuriStatusOk) {
  190. FURI_LOG_W(TAG, "furi_message_queue_get fail %d", status);
  191. seader_worker->stage = SeaderPollerEventTypeComplete;
  192. view_dispatcher_send_custom_event(
  193. seader->view_dispatcher, SeaderCustomEventWorkerExit);
  194. }
  195. if(seader_process_success_response_i(
  196. seader, seaderApdu.buf, seaderApdu.len, true, spc)) {
  197. // no-op
  198. } else {
  199. FURI_LOG_I(TAG, "Response false");
  200. view_dispatcher_send_custom_event(
  201. seader->view_dispatcher, SeaderCustomEventWorkerExit);
  202. seader_worker->stage = SeaderPollerEventTypeComplete;
  203. }
  204. }
  205. furi_mutex_release(seader_worker->mq_mutex);
  206. } else {
  207. furi_thread_set_current_priority(FuriThreadPriorityLowest);
  208. }
  209. }
  210. NfcCommand seader_worker_poller_callback_iso14443_4a(NfcGenericEvent event, void* context) {
  211. furi_assert(event.protocol == NfcProtocolIso14443_4a);
  212. NfcCommand ret = NfcCommandContinue;
  213. Seader* seader = context;
  214. SeaderWorker* seader_worker = seader->worker;
  215. const Iso14443_4aPollerEvent* iso14443_4a_event = event.event_data;
  216. SeaderPollerContainer spc = {.iso14443_4a_poller = event.instance};
  217. if(iso14443_4a_event->type == Iso14443_4aPollerEventTypeReady) {
  218. if(seader_worker->stage == SeaderPollerEventTypeCardDetect) {
  219. view_dispatcher_send_custom_event(
  220. seader->view_dispatcher, SeaderCustomEventPollerDetect);
  221. nfc_device_set_data(
  222. seader->nfc_device, NfcProtocolIso14443_4a, nfc_poller_get_data(seader->poller));
  223. size_t uid_len;
  224. const uint8_t* uid = nfc_device_get_uid(seader->nfc_device, &uid_len);
  225. const Iso14443_3aData* iso14443_3a_data =
  226. nfc_device_get_data(seader->nfc_device, NfcProtocolIso14443_3a);
  227. uint8_t sak = iso14443_3a_get_sak(iso14443_3a_data);
  228. seader_worker_card_detect(
  229. seader, sak, (uint8_t*)iso14443_3a_data->atqa, uid, uid_len, NULL, 0);
  230. // nfc_set_fdt_poll_fc(event.instance, SEADER_POLLER_MAX_FWT);
  231. furi_thread_set_current_priority(FuriThreadPriorityLowest);
  232. seader_worker->stage = SeaderPollerEventTypeConversation;
  233. } else if(seader_worker->stage == SeaderPollerEventTypeConversation) {
  234. seader_worker_poller_conversation(seader, &spc);
  235. } else if(seader_worker->stage == SeaderPollerEventTypeComplete) {
  236. ret = NfcCommandStop;
  237. }
  238. } else if(iso14443_4a_event->type == Iso14443_4aPollerEventTypeError) {
  239. Iso14443_4aPollerEventData* data = iso14443_4a_event->data;
  240. Iso14443_4aError error = data->error;
  241. FURI_LOG_W(TAG, "Iso14443_4aError %i", error);
  242. // I was hoping to catch MFC here, but it seems to be treated the same (None) as no card being present.
  243. switch(error) {
  244. case Iso14443_4aErrorNone:
  245. break;
  246. case Iso14443_4aErrorNotPresent:
  247. break;
  248. case Iso14443_4aErrorProtocol:
  249. ret = NfcCommandStop;
  250. break;
  251. case Iso14443_4aErrorTimeout:
  252. break;
  253. case Iso14443_4aErrorSendExtra:
  254. break;
  255. }
  256. }
  257. return ret;
  258. }
  259. NfcCommand seader_worker_poller_callback_mfc(NfcGenericEvent event, void* context) {
  260. furi_assert(event.protocol == NfcProtocolMfClassic);
  261. NfcCommand ret = NfcCommandContinue;
  262. Seader* seader = context;
  263. SeaderWorker* seader_worker = seader->worker;
  264. MfClassicPollerEvent* mfc_event = event.event_data;
  265. SeaderPollerContainer spc = {.mfc_poller = event.instance};
  266. if(mfc_event->type == MfClassicPollerEventTypeSuccess) {
  267. if(seader_worker->stage == SeaderPollerEventTypeCardDetect) {
  268. view_dispatcher_send_custom_event(
  269. seader->view_dispatcher, SeaderCustomEventPollerDetect);
  270. const MfClassicData* mfc_data = nfc_poller_get_data(seader->poller);
  271. uint8_t sak = iso14443_3a_get_sak(mfc_data->iso14443_3a_data);
  272. size_t uid_len = 0;
  273. const uint8_t* uid = mf_classic_get_uid(mfc_data, &uid_len);
  274. seader_worker_card_detect(seader, sak, NULL, uid, uid_len, NULL, 0);
  275. furi_thread_set_current_priority(FuriThreadPriorityLowest);
  276. seader_worker->stage = SeaderPollerEventTypeConversation;
  277. } else if(seader_worker->stage == SeaderPollerEventTypeConversation) {
  278. seader_worker_poller_conversation(seader, &spc);
  279. } else if(seader_worker->stage == SeaderPollerEventTypeComplete) {
  280. ret = NfcCommandStop;
  281. } else if(seader_worker->stage == SeaderPollerEventTypeFail) {
  282. ret = NfcCommandStop;
  283. }
  284. } else if(mfc_event->type == MfClassicPollerEventTypeFail) {
  285. ret = NfcCommandStop;
  286. }
  287. return ret;
  288. }
  289. NfcCommand seader_worker_poller_callback_picopass(PicopassPollerEvent event, void* context) {
  290. furi_assert(context);
  291. NfcCommand ret = NfcCommandContinue;
  292. Seader* seader = context;
  293. SeaderWorker* seader_worker = seader->worker;
  294. // I know this is is passing the same thing that is on seader all the way down, but I prefer the symmetry between the 15a and iso15 stuff
  295. PicopassPoller* instance = seader->picopass_poller;
  296. SeaderPollerContainer spc = {.picopass_poller = instance};
  297. if(event.type == PicopassPollerEventTypeCardDetected) {
  298. seader_worker->stage = SeaderPollerEventTypeCardDetect;
  299. } else if(event.type == PicopassPollerEventTypeSuccess) {
  300. if(seader_worker->stage == SeaderPollerEventTypeCardDetect) {
  301. view_dispatcher_send_custom_event(
  302. seader->view_dispatcher, SeaderCustomEventPollerDetect);
  303. uint8_t* csn = picopass_poller_get_csn(instance);
  304. seader_worker_card_detect(seader, 0, NULL, csn, sizeof(PicopassSerialNum), NULL, 0);
  305. furi_thread_set_current_priority(FuriThreadPriorityLowest);
  306. seader_worker->stage = SeaderPollerEventTypeConversation;
  307. } else if(seader_worker->stage == SeaderPollerEventTypeConversation) {
  308. seader_worker_poller_conversation(seader, &spc);
  309. } else if(seader_worker->stage == SeaderPollerEventTypeComplete) {
  310. ret = NfcCommandStop;
  311. }
  312. } else if(event.type == PicopassPollerEventTypeFail) {
  313. ret = NfcCommandStop;
  314. FURI_LOG_W(TAG, "PicopassPollerEventTypeFail");
  315. } else {
  316. FURI_LOG_D(TAG, "picopass event type %x", event.type);
  317. }
  318. return ret;
  319. }