nfc_magic_worker.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488
  1. #include "nfc_magic_worker_i.h"
  2. #include "nfc_magic_i.h"
  3. #include "lib/magic/common.h"
  4. #include "lib/magic/classic_gen1.h"
  5. #include "lib/magic/gen4.h"
  6. #define TAG "NfcMagicWorker"
  7. static void
  8. nfc_magic_worker_change_state(NfcMagicWorker* nfc_magic_worker, NfcMagicWorkerState state) {
  9. furi_assert(nfc_magic_worker);
  10. nfc_magic_worker->state = state;
  11. }
  12. NfcMagicWorker* nfc_magic_worker_alloc() {
  13. NfcMagicWorker* nfc_magic_worker = malloc(sizeof(NfcMagicWorker));
  14. // Worker thread attributes
  15. nfc_magic_worker->thread =
  16. furi_thread_alloc_ex("NfcMagicWorker", 8192, nfc_magic_worker_task, nfc_magic_worker);
  17. nfc_magic_worker->callback = NULL;
  18. nfc_magic_worker->context = NULL;
  19. nfc_magic_worker_change_state(nfc_magic_worker, NfcMagicWorkerStateReady);
  20. return nfc_magic_worker;
  21. }
  22. void nfc_magic_worker_free(NfcMagicWorker* nfc_magic_worker) {
  23. furi_assert(nfc_magic_worker);
  24. furi_thread_free(nfc_magic_worker->thread);
  25. free(nfc_magic_worker);
  26. }
  27. void nfc_magic_worker_stop(NfcMagicWorker* nfc_magic_worker) {
  28. furi_assert(nfc_magic_worker);
  29. nfc_magic_worker_change_state(nfc_magic_worker, NfcMagicWorkerStateStop);
  30. furi_thread_join(nfc_magic_worker->thread);
  31. }
  32. void nfc_magic_worker_start(
  33. NfcMagicWorker* nfc_magic_worker,
  34. NfcMagicWorkerState state,
  35. NfcMagicDevice* magic_dev,
  36. NfcDeviceData* dev_data,
  37. uint32_t new_password,
  38. NfcMagicWorkerCallback callback,
  39. void* context) {
  40. furi_assert(nfc_magic_worker);
  41. furi_assert(magic_dev);
  42. furi_assert(dev_data);
  43. furi_hal_nfc_deinit();
  44. furi_hal_nfc_init();
  45. nfc_magic_worker->callback = callback;
  46. nfc_magic_worker->context = context;
  47. nfc_magic_worker->magic_dev = magic_dev;
  48. nfc_magic_worker->dev_data = dev_data;
  49. nfc_magic_worker->new_password = new_password;
  50. nfc_magic_worker_change_state(nfc_magic_worker, state);
  51. furi_thread_start(nfc_magic_worker->thread);
  52. }
  53. int32_t nfc_magic_worker_task(void* context) {
  54. NfcMagicWorker* nfc_magic_worker = context;
  55. if(nfc_magic_worker->state == NfcMagicWorkerStateCheck) {
  56. nfc_magic_worker_check(nfc_magic_worker);
  57. } else if(nfc_magic_worker->state == NfcMagicWorkerStateWrite) {
  58. nfc_magic_worker_write(nfc_magic_worker);
  59. } else if(nfc_magic_worker->state == NfcMagicWorkerStateRekey) {
  60. nfc_magic_worker_rekey(nfc_magic_worker);
  61. } else if(nfc_magic_worker->state == NfcMagicWorkerStateWipe) {
  62. nfc_magic_worker_wipe(nfc_magic_worker);
  63. }
  64. nfc_magic_worker_change_state(nfc_magic_worker, NfcMagicWorkerStateReady);
  65. return 0;
  66. }
  67. void nfc_magic_worker_write(NfcMagicWorker* nfc_magic_worker) {
  68. bool card_found_notified = false;
  69. bool done = false;
  70. FuriHalNfcDevData nfc_data = {};
  71. NfcMagicDevice* magic_dev = nfc_magic_worker->magic_dev;
  72. NfcDeviceData* dev_data = nfc_magic_worker->dev_data;
  73. NfcProtocol dev_protocol = dev_data->protocol;
  74. while(nfc_magic_worker->state == NfcMagicWorkerStateWrite) {
  75. do {
  76. if(magic_dev->type == MagicTypeClassicGen1) {
  77. if(furi_hal_nfc_detect(&nfc_data, 200)) {
  78. magic_deactivate();
  79. magic_activate();
  80. if(!magic_gen1_wupa()) {
  81. FURI_LOG_E(TAG, "No card response to WUPA (not a magic card)");
  82. nfc_magic_worker->callback(
  83. NfcMagicWorkerEventWrongCard, nfc_magic_worker->context);
  84. done = true;
  85. break;
  86. }
  87. magic_deactivate();
  88. }
  89. magic_activate();
  90. if(magic_gen1_wupa()) {
  91. magic_gen1_data_access_cmd();
  92. MfClassicData* mfc_data = &dev_data->mf_classic_data;
  93. for(size_t i = 0; i < 64; i++) {
  94. FURI_LOG_D(TAG, "Writing block %d", i);
  95. if(!magic_gen1_write_blk(i, &mfc_data->block[i])) {
  96. FURI_LOG_E(TAG, "Failed to write %d block", i);
  97. done = true;
  98. nfc_magic_worker->callback(
  99. NfcMagicWorkerEventFail, nfc_magic_worker->context);
  100. break;
  101. }
  102. }
  103. done = true;
  104. nfc_magic_worker->callback(
  105. NfcMagicWorkerEventSuccess, nfc_magic_worker->context);
  106. break;
  107. }
  108. } else if(magic_dev->type == MagicTypeGen4) {
  109. if(furi_hal_nfc_detect(&nfc_data, 200)) {
  110. uint8_t gen4_config[MAGIC_GEN4_CONFIG_LEN];
  111. memcpy(gen4_config, MAGIC_DEFAULT_CONFIG, MAGIC_GEN4_CONFIG_LEN);
  112. uint32_t password = magic_dev->password;
  113. uint32_t cuid;
  114. size_t block_count = 64;
  115. MfClassicData* mfc_data;
  116. if(dev_protocol == NfcDeviceProtocolMifareClassic) {
  117. gen4_config[0] = 0x00;
  118. gen4_config[27] = 0x00;
  119. mfc_data = &dev_data->mf_classic_data;
  120. if(mfc_data->type == MfClassicType4k) block_count = 256;
  121. } else if(dev_protocol == NfcDeviceProtocolMifareUl) {
  122. MfUltralightData* mf_ul_data = &dev_data->mf_ul_data;
  123. gen4_config[0] = 0x01;
  124. switch(mf_ul_data->type) {
  125. case MfUltralightTypeUL11:
  126. case MfUltralightTypeUL21:
  127. // UL-C?
  128. // UL?
  129. default:
  130. gen4_config[27] = MagicGen4UltralightModeUL_EV1;
  131. break;
  132. case MfUltralightTypeNTAG203:
  133. case MfUltralightTypeNTAG213:
  134. case MfUltralightTypeNTAG215:
  135. case MfUltralightTypeNTAG216:
  136. case MfUltralightTypeNTAGI2C1K:
  137. case MfUltralightTypeNTAGI2C2K:
  138. case MfUltralightTypeNTAGI2CPlus1K:
  139. case MfUltralightTypeNTAGI2CPlus2K:
  140. gen4_config[27] = MagicGen4UltralightModeNTAG;
  141. block_count = 64 * 2;
  142. break;
  143. }
  144. }
  145. if(dev_data->nfc_data.uid_len == 4) {
  146. gen4_config[1] = MagicGen4UIDLengthSingle;
  147. } else if(dev_data->nfc_data.uid_len == 7) {
  148. gen4_config[1] = MagicGen4UIDLengthDouble;
  149. } else {
  150. FURI_LOG_E(TAG, "Unexpected UID length %d", dev_data->nfc_data.uid_len);
  151. nfc_magic_worker->callback(
  152. NfcMagicWorkerEventFail, nfc_magic_worker->context);
  153. done = true;
  154. break;
  155. }
  156. gen4_config[2] = (uint8_t)(password >> 24);
  157. gen4_config[3] = (uint8_t)(password >> 16);
  158. gen4_config[4] = (uint8_t)(password >> 8);
  159. gen4_config[5] = (uint8_t)password;
  160. if(dev_protocol == NfcDeviceProtocolMifareUl) {
  161. gen4_config[6] = MagicGen4ShadowModeHighSpeedIgnore;
  162. } else {
  163. gen4_config[6] = MagicGen4ShadowModeIgnore;
  164. }
  165. gen4_config[7] = 0x00;
  166. memset(gen4_config + 8, 0, 16);
  167. gen4_config[24] = dev_data->nfc_data.atqa[0];
  168. gen4_config[25] = dev_data->nfc_data.atqa[1];
  169. gen4_config[26] = dev_data->nfc_data.sak;
  170. gen4_config[28] = block_count;
  171. gen4_config[29] = 0x01;
  172. furi_hal_nfc_sleep();
  173. furi_hal_nfc_activate_nfca(200, &cuid);
  174. if(!magic_gen4_set_cfg(password, gen4_config, sizeof(gen4_config), false)) {
  175. nfc_magic_worker->callback(
  176. NfcMagicWorkerEventFail, nfc_magic_worker->context);
  177. done = true;
  178. break;
  179. }
  180. if(dev_protocol == NfcDeviceProtocolMifareClassic) {
  181. for(size_t i = 0; i < block_count; i++) {
  182. FURI_LOG_D(TAG, "Writing block %d", i);
  183. if(!magic_gen4_write_blk(password, i, mfc_data->block[i].value)) {
  184. FURI_LOG_E(TAG, "Failed to write %d block", i);
  185. nfc_magic_worker->callback(
  186. NfcMagicWorkerEventFail, nfc_magic_worker->context);
  187. done = true;
  188. break;
  189. }
  190. }
  191. } else if(dev_protocol == NfcDeviceProtocolMifareUl) {
  192. MfUltralightData* mf_ul_data = &dev_data->mf_ul_data;
  193. for(size_t i = 0; (i * 4) < mf_ul_data->data_read; i++) {
  194. size_t data_offset = i * 4;
  195. FURI_LOG_D(
  196. TAG,
  197. "Writing page %zu (%zu/%u)",
  198. i,
  199. data_offset,
  200. mf_ul_data->data_read);
  201. uint8_t* block = mf_ul_data->data + data_offset;
  202. if(!magic_gen4_write_blk(password, i, block)) {
  203. FURI_LOG_E(TAG, "Failed to write %zu page", i);
  204. nfc_magic_worker->callback(
  205. NfcMagicWorkerEventFail, nfc_magic_worker->context);
  206. done = true;
  207. break;
  208. }
  209. }
  210. uint8_t buffer[16] = {0};
  211. for(size_t i = 0; i < 8; i++) {
  212. memcpy(buffer, &mf_ul_data->signature[i * 4], 4); //-V1086
  213. if(!magic_gen4_write_blk(password, 0xF2 + i, buffer)) {
  214. FURI_LOG_E(TAG, "Failed to write signature block %d", i);
  215. nfc_magic_worker->callback(
  216. NfcMagicWorkerEventFail, nfc_magic_worker->context);
  217. done = true;
  218. break;
  219. }
  220. }
  221. buffer[0] = mf_ul_data->version.header;
  222. buffer[1] = mf_ul_data->version.vendor_id;
  223. buffer[2] = mf_ul_data->version.prod_type;
  224. buffer[3] = mf_ul_data->version.prod_subtype;
  225. if(!magic_gen4_write_blk(password, 0xFA, buffer)) {
  226. FURI_LOG_E(TAG, "Failed to write version block 0");
  227. nfc_magic_worker->callback(
  228. NfcMagicWorkerEventFail, nfc_magic_worker->context);
  229. done = true;
  230. break;
  231. }
  232. buffer[0] = mf_ul_data->version.prod_ver_major;
  233. buffer[1] = mf_ul_data->version.prod_ver_minor;
  234. buffer[2] = mf_ul_data->version.storage_size;
  235. buffer[3] = mf_ul_data->version.protocol_type;
  236. if(!magic_gen4_write_blk(password, 0xFB, buffer)) {
  237. FURI_LOG_E(TAG, "Failed to write version block 1");
  238. nfc_magic_worker->callback(
  239. NfcMagicWorkerEventFail, nfc_magic_worker->context);
  240. done = true;
  241. break;
  242. }
  243. }
  244. nfc_magic_worker->callback(
  245. NfcMagicWorkerEventSuccess, nfc_magic_worker->context);
  246. done = true;
  247. break;
  248. }
  249. }
  250. } while(false);
  251. if(done) break;
  252. if(card_found_notified) {
  253. nfc_magic_worker->callback(
  254. NfcMagicWorkerEventNoCardDetected, nfc_magic_worker->context);
  255. card_found_notified = false;
  256. }
  257. furi_delay_ms(300);
  258. }
  259. magic_deactivate();
  260. }
  261. void nfc_magic_worker_check(NfcMagicWorker* nfc_magic_worker) {
  262. FuriHalNfcDevData nfc_data = {};
  263. NfcMagicDevice* magic_dev = nfc_magic_worker->magic_dev;
  264. bool card_found_notified = false;
  265. uint8_t gen4_config[MAGIC_GEN4_CONFIG_LEN];
  266. while(nfc_magic_worker->state == NfcMagicWorkerStateCheck) {
  267. magic_activate();
  268. if(magic_gen1_wupa()) {
  269. magic_dev->type = MagicTypeClassicGen1;
  270. if(!card_found_notified) {
  271. nfc_magic_worker->callback(
  272. NfcMagicWorkerEventCardDetected, nfc_magic_worker->context);
  273. card_found_notified = true;
  274. }
  275. if(furi_hal_nfc_detect(&nfc_data, 200)) {
  276. magic_dev->cuid = nfc_data.cuid;
  277. magic_dev->uid_len = nfc_data.uid_len;
  278. } else {
  279. // wrong BCC
  280. magic_dev->uid_len = 4;
  281. }
  282. nfc_magic_worker->callback(NfcMagicWorkerEventSuccess, nfc_magic_worker->context);
  283. break;
  284. } else {
  285. magic_deactivate();
  286. magic_activate();
  287. if(furi_hal_nfc_detect(&nfc_data, 200)) {
  288. magic_dev->cuid = nfc_data.cuid;
  289. magic_dev->uid_len = nfc_data.uid_len;
  290. if(magic_gen4_get_cfg(magic_dev->password, gen4_config)) {
  291. magic_dev->type = MagicTypeGen4;
  292. if(!card_found_notified) {
  293. nfc_magic_worker->callback(
  294. NfcMagicWorkerEventCardDetected, nfc_magic_worker->context);
  295. card_found_notified = true;
  296. }
  297. nfc_magic_worker->callback(
  298. NfcMagicWorkerEventSuccess, nfc_magic_worker->context);
  299. } else {
  300. nfc_magic_worker->callback(
  301. NfcMagicWorkerEventWrongCard, nfc_magic_worker->context);
  302. card_found_notified = true;
  303. }
  304. break;
  305. } else {
  306. if(card_found_notified) {
  307. nfc_magic_worker->callback(
  308. NfcMagicWorkerEventNoCardDetected, nfc_magic_worker->context);
  309. card_found_notified = false;
  310. }
  311. }
  312. }
  313. magic_deactivate();
  314. furi_delay_ms(300);
  315. }
  316. magic_deactivate();
  317. }
  318. void nfc_magic_worker_rekey(NfcMagicWorker* nfc_magic_worker) {
  319. NfcMagicDevice* magic_dev = nfc_magic_worker->magic_dev;
  320. bool card_found_notified = false;
  321. if(magic_dev->type != MagicTypeGen4) {
  322. nfc_magic_worker->callback(NfcMagicWorkerEventCardDetected, nfc_magic_worker->context);
  323. return;
  324. }
  325. while(nfc_magic_worker->state == NfcMagicWorkerStateRekey) {
  326. magic_activate();
  327. uint32_t cuid;
  328. furi_hal_nfc_activate_nfca(200, &cuid);
  329. if(cuid != magic_dev->cuid) {
  330. if(card_found_notified) {
  331. nfc_magic_worker->callback(
  332. NfcMagicWorkerEventNoCardDetected, nfc_magic_worker->context);
  333. card_found_notified = false;
  334. }
  335. continue;
  336. }
  337. nfc_magic_worker->callback(NfcMagicWorkerEventCardDetected, nfc_magic_worker->context);
  338. card_found_notified = true;
  339. if(magic_gen4_set_pwd(magic_dev->password, nfc_magic_worker->new_password)) {
  340. magic_dev->password = nfc_magic_worker->new_password;
  341. nfc_magic_worker->callback(NfcMagicWorkerEventSuccess, nfc_magic_worker->context);
  342. break;
  343. }
  344. if(card_found_notified) { //-V547
  345. nfc_magic_worker->callback(
  346. NfcMagicWorkerEventNoCardDetected, nfc_magic_worker->context);
  347. card_found_notified = false;
  348. }
  349. furi_delay_ms(300);
  350. }
  351. magic_deactivate();
  352. }
  353. void nfc_magic_worker_wipe(NfcMagicWorker* nfc_magic_worker) {
  354. NfcMagicDevice* magic_dev = nfc_magic_worker->magic_dev;
  355. bool card_found_notified = false;
  356. bool card_wiped = false;
  357. MfClassicBlock block;
  358. memset(&block, 0, sizeof(MfClassicBlock));
  359. MfClassicBlock empty_block;
  360. memset(&empty_block, 0, sizeof(MfClassicBlock));
  361. MfClassicBlock trailer_block;
  362. memset(&trailer_block, 0xff, sizeof(MfClassicBlock));
  363. block.value[0] = 0x01;
  364. block.value[1] = 0x02;
  365. block.value[2] = 0x03;
  366. block.value[3] = 0x04;
  367. block.value[4] = 0x04;
  368. block.value[5] = 0x08;
  369. block.value[6] = 0x04;
  370. trailer_block.value[7] = 0x07;
  371. trailer_block.value[8] = 0x80;
  372. trailer_block.value[9] = 0x69;
  373. while(nfc_magic_worker->state == NfcMagicWorkerStateWipe) {
  374. do {
  375. magic_deactivate();
  376. furi_delay_ms(300);
  377. if(!magic_activate()) break;
  378. if(magic_dev->type == MagicTypeClassicGen1) {
  379. if(!magic_gen1_wupa()) break;
  380. if(!card_found_notified) {
  381. nfc_magic_worker->callback(
  382. NfcMagicWorkerEventCardDetected, nfc_magic_worker->context);
  383. card_found_notified = true;
  384. }
  385. if(!magic_gen1_data_access_cmd()) break;
  386. if(!magic_gen1_write_blk(0, &block)) break;
  387. for(size_t i = 1; i < 64; i++) {
  388. FURI_LOG_D(TAG, "Wiping block %d", i);
  389. bool success = false;
  390. if((i | 0x03) == i) {
  391. success = magic_gen1_write_blk(i, &trailer_block);
  392. } else {
  393. success = magic_gen1_write_blk(i, &empty_block);
  394. }
  395. if(!success) {
  396. FURI_LOG_E(TAG, "Failed to write %d block", i);
  397. nfc_magic_worker->callback(
  398. NfcMagicWorkerEventFail, nfc_magic_worker->context);
  399. break;
  400. }
  401. }
  402. card_wiped = true;
  403. nfc_magic_worker->callback(NfcMagicWorkerEventSuccess, nfc_magic_worker->context);
  404. } else if(magic_dev->type == MagicTypeGen4) {
  405. uint32_t cuid;
  406. if(!furi_hal_nfc_activate_nfca(200, &cuid)) break;
  407. if(cuid != magic_dev->cuid) break;
  408. if(!card_found_notified) {
  409. nfc_magic_worker->callback(
  410. NfcMagicWorkerEventCardDetected, nfc_magic_worker->context);
  411. card_found_notified = true;
  412. }
  413. if(!magic_gen4_wipe(magic_dev->password)) break;
  414. card_wiped = true;
  415. nfc_magic_worker->callback(NfcMagicWorkerEventSuccess, nfc_magic_worker->context);
  416. }
  417. } while(false);
  418. if(card_wiped) break;
  419. if(card_found_notified) {
  420. nfc_magic_worker->callback(
  421. NfcMagicWorkerEventNoCardDetected, nfc_magic_worker->context);
  422. card_found_notified = false;
  423. }
  424. }
  425. magic_deactivate();
  426. }