gen4_poller.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608
  1. #include "core/log.h"
  2. #include "gen4_poller_i.h"
  3. #include <nfc/protocols/iso14443_3a/iso14443_3a.h>
  4. #include <nfc/protocols/iso14443_3a/iso14443_3a_poller.h>
  5. #include <nfc/helpers/nfc_util.h>
  6. #include <nfc/nfc_poller.h>
  7. #include <furi/furi.h>
  8. #define GEN4_POLLER_THREAD_FLAG_DETECTED (1U << 0)
  9. typedef NfcCommand (*Gen4PollerStateHandler)(Gen4Poller* instance);
  10. typedef struct {
  11. NfcPoller* poller;
  12. uint32_t password;
  13. BitBuffer* tx_buffer;
  14. BitBuffer* rx_buffer;
  15. FuriThreadId thread_id;
  16. Gen4PollerError error;
  17. } Gen4PollerDetectContext;
  18. static const uint8_t gen4_poller_default_config[] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02,
  19. 0x00, 0x09, 0x78, 0x00, 0x91, 0x02, 0xDA,
  20. 0xBC, 0x19, 0x10, 0x10, 0x11, 0x12, 0x13,
  21. 0x14, 0x15, 0x16, 0x04, 0x00, 0x08, 0x00};
  22. static const uint8_t gen4_poller_default_block_0[GEN4_POLLER_BLOCK_SIZE] =
  23. {0x00, 0x01, 0x02, 0x03, 0x04, 0x04, 0x08, 0x04, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
  24. static const uint8_t gen4_poller_default_empty_block[GEN4_POLLER_BLOCK_SIZE] = {0};
  25. static const uint8_t gen4_poller_default_sector_trailer_block[GEN4_POLLER_BLOCK_SIZE] =
  26. {0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x07, 0x80, 0x69, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF};
  27. static bool gen4_poller_is_sector_trailer(uint8_t block_num) {
  28. uint8_t sec_tr_block_num = 0;
  29. if(block_num < 128) {
  30. sec_tr_block_num = block_num | 0x03;
  31. } else {
  32. sec_tr_block_num = block_num | 0x0f;
  33. }
  34. return block_num == sec_tr_block_num;
  35. }
  36. Gen4Poller* gen4_poller_alloc(Nfc* nfc) {
  37. furi_assert(nfc);
  38. Gen4Poller* instance = malloc(sizeof(Gen4Poller));
  39. instance->poller = nfc_poller_alloc(nfc, NfcProtocolIso14443_3a);
  40. instance->gen4_event.data = &instance->gen4_event_data;
  41. instance->tx_buffer = bit_buffer_alloc(GEN4_POLLER_MAX_BUFFER_SIZE);
  42. instance->rx_buffer = bit_buffer_alloc(GEN4_POLLER_MAX_BUFFER_SIZE);
  43. return instance;
  44. }
  45. void gen4_poller_free(Gen4Poller* instance) {
  46. furi_assert(instance);
  47. nfc_poller_free(instance->poller);
  48. bit_buffer_free(instance->tx_buffer);
  49. bit_buffer_free(instance->rx_buffer);
  50. free(instance);
  51. }
  52. void gen4_poller_set_password(Gen4Poller* instance, uint32_t password) {
  53. furi_assert(instance);
  54. instance->password = password;
  55. }
  56. NfcCommand gen4_poller_detect_callback(NfcGenericEvent event, void* context) {
  57. furi_assert(context);
  58. furi_assert(event.protocol == NfcProtocolIso14443_3a);
  59. furi_assert(event.instance);
  60. furi_assert(event.event_data);
  61. NfcCommand command = NfcCommandStop;
  62. Gen4PollerDetectContext* gen4_poller_detect_ctx = context;
  63. Iso14443_3aPoller* iso3_poller = event.instance;
  64. Iso14443_3aPollerEvent* iso3_event = event.event_data;
  65. gen4_poller_detect_ctx->error = Gen4PollerErrorTimeout;
  66. if(iso3_event->type == Iso14443_3aPollerEventTypeReady) {
  67. do {
  68. bit_buffer_append_byte(gen4_poller_detect_ctx->tx_buffer, GEN4_CMD_PREFIX);
  69. uint8_t pwd_arr[4] = {};
  70. nfc_util_num2bytes(gen4_poller_detect_ctx->password, COUNT_OF(pwd_arr), pwd_arr);
  71. bit_buffer_append_bytes(gen4_poller_detect_ctx->tx_buffer, pwd_arr, COUNT_OF(pwd_arr));
  72. bit_buffer_append_byte(gen4_poller_detect_ctx->tx_buffer, GEN4_CMD_GET_CFG);
  73. Iso14443_3aError error = iso14443_3a_poller_send_standard_frame(
  74. iso3_poller,
  75. gen4_poller_detect_ctx->tx_buffer,
  76. gen4_poller_detect_ctx->rx_buffer,
  77. GEN4_POLLER_MAX_FWT);
  78. if(error != Iso14443_3aErrorNone) {
  79. gen4_poller_detect_ctx->error = Gen4PollerErrorProtocol;
  80. break;
  81. }
  82. size_t rx_bytes = bit_buffer_get_size_bytes(gen4_poller_detect_ctx->rx_buffer);
  83. if((rx_bytes != 30) && (rx_bytes != 32)) {
  84. gen4_poller_detect_ctx->error = Gen4PollerErrorProtocol;
  85. break;
  86. }
  87. gen4_poller_detect_ctx->error = Gen4PollerErrorNone;
  88. } while(false);
  89. } else if(iso3_event->type == Iso14443_3aPollerEventTypeError) {
  90. gen4_poller_detect_ctx->error = Gen4PollerErrorTimeout;
  91. }
  92. furi_thread_flags_set(gen4_poller_detect_ctx->thread_id, GEN4_POLLER_THREAD_FLAG_DETECTED);
  93. return command;
  94. }
  95. Gen4PollerError gen4_poller_detect(Nfc* nfc, uint32_t password) {
  96. furi_assert(nfc);
  97. Gen4PollerDetectContext gen4_poller_detect_ctx = {};
  98. gen4_poller_detect_ctx.poller = nfc_poller_alloc(nfc, NfcProtocolIso14443_3a);
  99. gen4_poller_detect_ctx.password = password;
  100. gen4_poller_detect_ctx.tx_buffer = bit_buffer_alloc(GEN4_POLLER_MAX_BUFFER_SIZE);
  101. gen4_poller_detect_ctx.rx_buffer = bit_buffer_alloc(GEN4_POLLER_MAX_BUFFER_SIZE);
  102. gen4_poller_detect_ctx.thread_id = furi_thread_get_current_id();
  103. gen4_poller_detect_ctx.error = Gen4PollerErrorNone;
  104. nfc_poller_start(
  105. gen4_poller_detect_ctx.poller, gen4_poller_detect_callback, &gen4_poller_detect_ctx);
  106. uint32_t flags =
  107. furi_thread_flags_wait(GEN4_POLLER_THREAD_FLAG_DETECTED, FuriFlagWaitAny, FuriWaitForever);
  108. if(flags & GEN4_POLLER_THREAD_FLAG_DETECTED) {
  109. furi_thread_flags_clear(GEN4_POLLER_THREAD_FLAG_DETECTED);
  110. }
  111. nfc_poller_stop(gen4_poller_detect_ctx.poller);
  112. nfc_poller_free(gen4_poller_detect_ctx.poller);
  113. bit_buffer_free(gen4_poller_detect_ctx.tx_buffer);
  114. bit_buffer_free(gen4_poller_detect_ctx.rx_buffer);
  115. return gen4_poller_detect_ctx.error;
  116. }
  117. NfcCommand gen4_poller_idle_handler(Gen4Poller* instance) {
  118. NfcCommand command = NfcCommandContinue;
  119. instance->current_block = 0;
  120. memset(instance->config, 0, sizeof(instance->config));
  121. instance->gen4_event.type = Gen4PollerEventTypeCardDetected;
  122. command = instance->callback(instance->gen4_event, instance->context);
  123. instance->state = Gen4PollerStateRequestMode;
  124. return command;
  125. }
  126. NfcCommand gen4_poller_request_mode_handler(Gen4Poller* instance) {
  127. NfcCommand command = NfcCommandContinue;
  128. instance->gen4_event.type = Gen4PollerEventTypeRequestMode;
  129. command = instance->callback(instance->gen4_event, instance->context);
  130. if(instance->gen4_event_data.request_mode.mode == Gen4PollerModeWipe) {
  131. instance->state = Gen4PollerStateWipe;
  132. } else if(instance->gen4_event_data.request_mode.mode == Gen4PollerModeWrite) {
  133. instance->state = Gen4PollerStateRequestWriteData;
  134. } else if(instance->gen4_event_data.request_mode.mode == Gen4PollerModeSetPassword) {
  135. instance->state = Gen4PollerStateChangePassword;
  136. } else if(instance->gen4_event_data.request_mode.mode == Gen4PollerModeSetDefaultCFG) {
  137. instance->state = Gen4PollerStateSetDefaultConfig;
  138. } else if(instance->gen4_event_data.request_mode.mode == Gen4PollerModeGetCFG) {
  139. instance->state = Gen4PollerStateGetCurrentConfig;
  140. } else if(instance->gen4_event_data.request_mode.mode == Gen4PollerModeGetRevision) {
  141. instance->state = Gen4PollerStateGetRevision;
  142. } else {
  143. instance->state = Gen4PollerStateFail;
  144. }
  145. return command;
  146. }
  147. NfcCommand gen4_poller_wipe_handler(Gen4Poller* instance) {
  148. NfcCommand command = NfcCommandContinue;
  149. do {
  150. Gen4PollerError error = Gen4PollerErrorNone;
  151. if(instance->current_block == 0) {
  152. error = gen4_poller_set_config(
  153. instance,
  154. instance->password,
  155. gen4_poller_default_config,
  156. sizeof(gen4_poller_default_config),
  157. false);
  158. if(error != Gen4PollerErrorNone) {
  159. FURI_LOG_D(TAG, "Failed to set default config: %d", error);
  160. instance->state = Gen4PollerStateFail;
  161. break;
  162. }
  163. instance->password = 0;
  164. error = gen4_poller_write_block(
  165. instance, instance->password, instance->current_block, gen4_poller_default_block_0);
  166. if(error != Gen4PollerErrorNone) {
  167. FURI_LOG_D(TAG, "Failed to write 0 block: %d", error);
  168. instance->state = Gen4PollerStateFail;
  169. break;
  170. }
  171. } else if(instance->current_block < GEN4_POLLER_BLOCKS_TOTAL) {
  172. const uint8_t* block = gen4_poller_is_sector_trailer(instance->current_block) ?
  173. gen4_poller_default_sector_trailer_block :
  174. gen4_poller_default_empty_block;
  175. error = gen4_poller_write_block(
  176. instance, instance->password, instance->current_block, block);
  177. if(error != Gen4PollerErrorNone) {
  178. FURI_LOG_D(TAG, "Failed to write %d block: %d", instance->current_block, error);
  179. instance->state = Gen4PollerStateFail;
  180. break;
  181. }
  182. } else {
  183. instance->state = Gen4PollerStateSuccess;
  184. break;
  185. }
  186. instance->current_block++;
  187. } while(false);
  188. return command;
  189. }
  190. NfcCommand gen4_poller_request_write_data_handler(Gen4Poller* instance) {
  191. NfcCommand command = NfcCommandContinue;
  192. instance->gen4_event.type = Gen4PollerEventTypeRequestDataToWrite;
  193. command = instance->callback(instance->gen4_event, instance->context);
  194. instance->protocol = instance->gen4_event_data.request_data.protocol;
  195. instance->data = instance->gen4_event_data.request_data.data;
  196. if((instance->protocol == NfcProtocolMfClassic) ||
  197. (instance->protocol == NfcProtocolMfUltralight)) {
  198. instance->state = Gen4PollerStateWrite;
  199. } else {
  200. FURI_LOG_E(TAG, "Unsupported protocol");
  201. instance->state = Gen4PollerStateFail;
  202. }
  203. return command;
  204. }
  205. static NfcCommand gen4_poller_write_mf_classic(Gen4Poller* instance) {
  206. NfcCommand command = NfcCommandContinue;
  207. do {
  208. const MfClassicData* mfc_data = instance->data;
  209. const Iso14443_3aData* iso3_data = mfc_data->iso14443_3a_data;
  210. if(instance->current_block == 0) {
  211. instance->config[0] = 0x00;
  212. instance->total_blocks = mf_classic_get_total_block_num(mfc_data->type);
  213. if(iso3_data->uid_len == 4) {
  214. instance->config[1] = Gen4PollerUIDLengthSingle;
  215. } else if(iso3_data->uid_len == 7) {
  216. instance->config[1] = Gen4PollerUIDLengthDouble;
  217. } else {
  218. FURI_LOG_E(TAG, "Unsupported UID len: %d", iso3_data->uid_len);
  219. instance->state = Gen4PollerStateFail;
  220. break;
  221. }
  222. instance->config[6] = Gen4PollerShadowModeIgnore;
  223. instance->config[24] = iso3_data->atqa[0];
  224. instance->config[25] = iso3_data->atqa[1];
  225. instance->config[26] = iso3_data->sak;
  226. instance->config[27] = 0x00;
  227. instance->config[28] = instance->total_blocks;
  228. instance->config[29] = 0x01;
  229. Gen4PollerError error = gen4_poller_set_config(
  230. instance, instance->password, instance->config, sizeof(instance->config), false);
  231. if(error != Gen4PollerErrorNone) {
  232. FURI_LOG_D(TAG, "Failed to write config: %d", error);
  233. instance->state = Gen4PollerStateFail;
  234. break;
  235. }
  236. }
  237. if(instance->current_block < instance->total_blocks) {
  238. FURI_LOG_D(TAG, "Writing block %d", instance->current_block);
  239. Gen4PollerError error = gen4_poller_write_block(
  240. instance,
  241. instance->password,
  242. instance->current_block,
  243. mfc_data->block[instance->current_block].data);
  244. if(error != Gen4PollerErrorNone) {
  245. FURI_LOG_D(TAG, "Failed to write %d block: %d", instance->current_block, error);
  246. instance->state = Gen4PollerStateFail;
  247. break;
  248. }
  249. } else {
  250. instance->state = Gen4PollerStateSuccess;
  251. break;
  252. }
  253. instance->current_block++;
  254. } while(false);
  255. return command;
  256. }
  257. static NfcCommand gen4_poller_write_mf_ultralight(Gen4Poller* instance) {
  258. NfcCommand command = NfcCommandContinue;
  259. do {
  260. const MfUltralightData* mfu_data = instance->data;
  261. const Iso14443_3aData* iso3_data = mfu_data->iso14443_3a_data;
  262. if(instance->current_block == 0) {
  263. instance->total_blocks = 64;
  264. instance->config[0] = 0x01;
  265. switch(mfu_data->type) {
  266. case MfUltralightTypeNTAG203:
  267. case MfUltralightTypeNTAG213:
  268. case MfUltralightTypeNTAG215:
  269. case MfUltralightTypeNTAG216:
  270. case MfUltralightTypeNTAGI2C1K:
  271. case MfUltralightTypeNTAGI2C2K:
  272. case MfUltralightTypeNTAGI2CPlus1K:
  273. case MfUltralightTypeNTAGI2CPlus2K:
  274. instance->config[27] = Gen4PollerUltralightModeNTAG;
  275. instance->total_blocks = 64 * 2;
  276. break;
  277. case MfUltralightTypeUL11:
  278. case MfUltralightTypeUL21:
  279. // UL-C?
  280. // UL?
  281. default:
  282. instance->config[27] = Gen4PollerUltralightModeUL_EV1;
  283. break;
  284. }
  285. if(iso3_data->uid_len == 4) {
  286. instance->config[1] = Gen4PollerUIDLengthSingle;
  287. } else if(iso3_data->uid_len == 7) {
  288. instance->config[1] = Gen4PollerUIDLengthDouble;
  289. } else {
  290. FURI_LOG_E(TAG, "Unsupported UID len: %d", iso3_data->uid_len);
  291. instance->state = Gen4PollerStateFail;
  292. break;
  293. }
  294. instance->config[6] = Gen4PollerShadowModeHighSpeedIgnore;
  295. instance->config[24] = iso3_data->atqa[0];
  296. instance->config[25] = iso3_data->atqa[1];
  297. instance->config[26] = iso3_data->sak;
  298. instance->config[27] = 0x00;
  299. instance->config[28] = instance->total_blocks;
  300. instance->config[29] = 0x01;
  301. Gen4PollerError error = gen4_poller_set_config(
  302. instance, instance->password, instance->config, sizeof(instance->config), false);
  303. if(error != Gen4PollerErrorNone) {
  304. FURI_LOG_D(TAG, "Failed to write config: %d", error);
  305. instance->state = Gen4PollerStateFail;
  306. break;
  307. }
  308. }
  309. if(instance->current_block < mfu_data->pages_read) {
  310. FURI_LOG_D(
  311. TAG, "Writing page %zu / %zu", instance->current_block, mfu_data->pages_read);
  312. Gen4PollerError error = gen4_poller_write_block(
  313. instance,
  314. instance->password,
  315. instance->current_block,
  316. mfu_data->page[instance->current_block].data);
  317. if(error != Gen4PollerErrorNone) {
  318. FURI_LOG_D(TAG, "Failed to write %d page: %d", instance->current_block, error);
  319. instance->state = Gen4PollerStateFail;
  320. break;
  321. }
  322. instance->current_block++;
  323. } else {
  324. uint8_t block[GEN4_POLLER_BLOCK_SIZE] = {};
  325. bool write_success = true;
  326. for(size_t i = 0; i < 8; i++) {
  327. memcpy(block, &mfu_data->signature.data[i * 4], 4); //-V1086
  328. Gen4PollerError error =
  329. gen4_poller_write_block(instance, instance->password, 0xF2 + i, block);
  330. if(error != Gen4PollerErrorNone) {
  331. write_success = false;
  332. break;
  333. }
  334. }
  335. if(!write_success) {
  336. FURI_LOG_E(TAG, "Failed to write Signature");
  337. instance->state = Gen4PollerStateFail;
  338. break;
  339. }
  340. block[0] = mfu_data->version.header;
  341. block[1] = mfu_data->version.vendor_id;
  342. block[2] = mfu_data->version.prod_type;
  343. block[3] = mfu_data->version.prod_subtype;
  344. Gen4PollerError error =
  345. gen4_poller_write_block(instance, instance->password, 0xFA, block);
  346. if(error != Gen4PollerErrorNone) {
  347. FURI_LOG_E(TAG, "Failed to write 1st part Version");
  348. instance->state = Gen4PollerStateFail;
  349. break;
  350. }
  351. block[0] = mfu_data->version.prod_ver_major;
  352. block[1] = mfu_data->version.prod_ver_minor;
  353. block[2] = mfu_data->version.storage_size;
  354. block[3] = mfu_data->version.protocol_type;
  355. error = gen4_poller_write_block(instance, instance->password, 0xFB, block);
  356. if(error != Gen4PollerErrorNone) {
  357. FURI_LOG_E(TAG, "Failed to write 2nd part Version");
  358. instance->state = Gen4PollerStateFail;
  359. break;
  360. }
  361. instance->state = Gen4PollerStateSuccess;
  362. }
  363. } while(false);
  364. return command;
  365. }
  366. NfcCommand gen4_poller_write_handler(Gen4Poller* instance) {
  367. NfcCommand command = NfcCommandContinue;
  368. memcpy(instance->config, gen4_poller_default_config, sizeof(gen4_poller_default_config));
  369. uint8_t password_arr[4] = {};
  370. nfc_util_num2bytes(instance->password, sizeof(password_arr), password_arr);
  371. memcpy(&instance->config[2], password_arr, sizeof(password_arr));
  372. memset(&instance->config[7], 0, 17);
  373. if(instance->protocol == NfcProtocolMfClassic) {
  374. command = gen4_poller_write_mf_classic(instance);
  375. } else if(instance->protocol == NfcProtocolMfUltralight) {
  376. command = gen4_poller_write_mf_ultralight(instance);
  377. } else {
  378. furi_crash("Unsupported protocol to write");
  379. }
  380. return command;
  381. }
  382. NfcCommand gen4_poller_change_password_handler(Gen4Poller* instance) {
  383. NfcCommand command = NfcCommandContinue;
  384. do {
  385. instance->gen4_event.type = Gen4PollerEventTypeRequestNewPassword;
  386. command = instance->callback(instance->gen4_event, instance->context);
  387. if(command != NfcCommandContinue) break;
  388. uint32_t new_password = instance->gen4_event_data.request_password.password;
  389. Gen4PollerError error =
  390. gen4_poller_change_password(instance, instance->password, new_password);
  391. if(error != Gen4PollerErrorNone) {
  392. FURI_LOG_E(TAG, "Failed to change password: %d", error);
  393. instance->state = Gen4PollerStateFail;
  394. break;
  395. }
  396. instance->password = new_password;
  397. instance->state = Gen4PollerStateSuccess;
  398. } while(false);
  399. return command;
  400. }
  401. NfcCommand gen4_poller_set_default_cfg_handler(Gen4Poller* instance) {
  402. NfcCommand command = NfcCommandContinue;
  403. do {
  404. Gen4PollerError error = gen4_poller_set_config(
  405. instance,
  406. instance->password,
  407. gen4_poller_default_config,
  408. sizeof(gen4_poller_default_config),
  409. false);
  410. if(error != Gen4PollerErrorNone) {
  411. FURI_LOG_E(TAG, "Failed to set default config: %d", error);
  412. instance->state = Gen4PollerStateFail;
  413. break;
  414. }
  415. instance->state = Gen4PollerStateSuccess;
  416. } while(false);
  417. return command;
  418. }
  419. NfcCommand gen4_poller_get_current_cfg_handler(Gen4Poller* instance) {
  420. NfcCommand command = NfcCommandContinue;
  421. do {
  422. uint8_t the_config[32] = {};
  423. Gen4PollerError error = gen4_poller_get_config(instance, instance->password, the_config);
  424. if(error != Gen4PollerErrorNone) {
  425. FURI_LOG_E(TAG, "Failed to get current config: %d", error);
  426. instance->state = Gen4PollerStateFail;
  427. break;
  428. }
  429. // Copy config data to event data buffer
  430. memcpy(instance->gen4_event_data.display_config, the_config, sizeof(the_config));
  431. instance->state = Gen4PollerStateSuccess;
  432. } while(false);
  433. return command;
  434. }
  435. NfcCommand gen4_poller_get_revision_handler(Gen4Poller* instance) {
  436. NfcCommand command = NfcCommandContinue;
  437. do {
  438. uint8_t the_revision[5] = {0};
  439. Gen4PollerError error =
  440. gen4_poller_get_revision(instance, instance->password, the_revision);
  441. if(error != Gen4PollerErrorNone) {
  442. FURI_LOG_E(TAG, "Failed to get revision: %d", error);
  443. instance->state = Gen4PollerStateFail;
  444. break;
  445. }
  446. // Copy revision data to event data buffer
  447. memcpy(instance->gen4_event_data.revision_data, the_revision, sizeof(the_revision));
  448. instance->state = Gen4PollerStateSuccess;
  449. } while(false);
  450. return command;
  451. }
  452. NfcCommand gen4_poller_success_handler(Gen4Poller* instance) {
  453. NfcCommand command = NfcCommandContinue;
  454. instance->gen4_event.type = Gen4PollerEventTypeSuccess;
  455. command = instance->callback(instance->gen4_event, instance->context);
  456. if(command != NfcCommandStop) {
  457. furi_delay_ms(100);
  458. }
  459. return command;
  460. }
  461. NfcCommand gen4_poller_fail_handler(Gen4Poller* instance) {
  462. NfcCommand command = NfcCommandContinue;
  463. instance->gen4_event.type = Gen4PollerEventTypeFail;
  464. command = instance->callback(instance->gen4_event, instance->context);
  465. if(command != NfcCommandStop) {
  466. furi_delay_ms(100);
  467. }
  468. return command;
  469. }
  470. static const Gen4PollerStateHandler gen4_poller_state_handlers[Gen4PollerStateNum] = {
  471. [Gen4PollerStateIdle] = gen4_poller_idle_handler,
  472. [Gen4PollerStateRequestMode] = gen4_poller_request_mode_handler,
  473. [Gen4PollerStateRequestWriteData] = gen4_poller_request_write_data_handler,
  474. [Gen4PollerStateWrite] = gen4_poller_write_handler,
  475. [Gen4PollerStateWipe] = gen4_poller_wipe_handler,
  476. [Gen4PollerStateChangePassword] = gen4_poller_change_password_handler,
  477. [Gen4PollerStateSetDefaultConfig] = gen4_poller_set_default_cfg_handler,
  478. [Gen4PollerStateGetCurrentConfig] = gen4_poller_get_current_cfg_handler,
  479. [Gen4PollerStateGetRevision] = gen4_poller_get_revision_handler,
  480. [Gen4PollerStateSuccess] = gen4_poller_success_handler,
  481. [Gen4PollerStateFail] = gen4_poller_fail_handler,
  482. };
  483. static NfcCommand gen4_poller_callback(NfcGenericEvent event, void* context) {
  484. furi_assert(context);
  485. furi_assert(event.protocol == NfcProtocolIso14443_3a);
  486. furi_assert(event.event_data);
  487. furi_assert(event.instance);
  488. NfcCommand command = NfcCommandContinue;
  489. Gen4Poller* instance = context;
  490. instance->iso3_poller = event.instance;
  491. Iso14443_3aPollerEvent* iso3_event = event.event_data;
  492. if(iso3_event->type == Iso14443_3aPollerEventTypeReady) {
  493. command = gen4_poller_state_handlers[instance->state](instance);
  494. }
  495. return command;
  496. }
  497. void gen4_poller_start(Gen4Poller* instance, Gen4PollerCallback callback, void* context) {
  498. furi_assert(instance);
  499. furi_assert(callback);
  500. instance->callback = callback;
  501. instance->context = context;
  502. nfc_poller_start(instance->poller, gen4_poller_callback, instance);
  503. }
  504. void gen4_poller_stop(Gen4Poller* instance) {
  505. furi_assert(instance);
  506. nfc_poller_stop(instance->poller);
  507. }