| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485 |
- #include "nfc_magic_worker_i.h"
- #include "nfc_magic_i.h"
- #include "lib/magic/common.h"
- #include "lib/magic/classic_gen1.h"
- #include "lib/magic/gen4.h"
- #define TAG "NfcMagicWorker"
- static void
- nfc_magic_worker_change_state(NfcMagicWorker* nfc_magic_worker, NfcMagicWorkerState state) {
- furi_assert(nfc_magic_worker);
- nfc_magic_worker->state = state;
- }
- NfcMagicWorker* nfc_magic_worker_alloc() {
- NfcMagicWorker* nfc_magic_worker = malloc(sizeof(NfcMagicWorker));
- // Worker thread attributes
- nfc_magic_worker->thread =
- furi_thread_alloc_ex("NfcMagicWorker", 8192, nfc_magic_worker_task, nfc_magic_worker);
- nfc_magic_worker->callback = NULL;
- nfc_magic_worker->context = NULL;
- nfc_magic_worker_change_state(nfc_magic_worker, NfcMagicWorkerStateReady);
- return nfc_magic_worker;
- }
- void nfc_magic_worker_free(NfcMagicWorker* nfc_magic_worker) {
- furi_assert(nfc_magic_worker);
- furi_thread_free(nfc_magic_worker->thread);
- free(nfc_magic_worker);
- }
- void nfc_magic_worker_stop(NfcMagicWorker* nfc_magic_worker) {
- furi_assert(nfc_magic_worker);
- nfc_magic_worker_change_state(nfc_magic_worker, NfcMagicWorkerStateStop);
- furi_thread_join(nfc_magic_worker->thread);
- }
- void nfc_magic_worker_start(
- NfcMagicWorker* nfc_magic_worker,
- NfcMagicWorkerState state,
- NfcMagicDevice* magic_dev,
- NfcDeviceData* dev_data,
- uint32_t new_password,
- NfcMagicWorkerCallback callback,
- void* context) {
- furi_assert(nfc_magic_worker);
- furi_assert(magic_dev);
- furi_assert(dev_data);
- nfc_magic_worker->callback = callback;
- nfc_magic_worker->context = context;
- nfc_magic_worker->magic_dev = magic_dev;
- nfc_magic_worker->dev_data = dev_data;
- nfc_magic_worker->new_password = new_password;
- nfc_magic_worker_change_state(nfc_magic_worker, state);
- furi_thread_start(nfc_magic_worker->thread);
- }
- int32_t nfc_magic_worker_task(void* context) {
- NfcMagicWorker* nfc_magic_worker = context;
- if(nfc_magic_worker->state == NfcMagicWorkerStateCheck) {
- nfc_magic_worker_check(nfc_magic_worker);
- } else if(nfc_magic_worker->state == NfcMagicWorkerStateWrite) {
- nfc_magic_worker_write(nfc_magic_worker);
- } else if(nfc_magic_worker->state == NfcMagicWorkerStateRekey) {
- nfc_magic_worker_rekey(nfc_magic_worker);
- } else if(nfc_magic_worker->state == NfcMagicWorkerStateWipe) {
- nfc_magic_worker_wipe(nfc_magic_worker);
- }
- nfc_magic_worker_change_state(nfc_magic_worker, NfcMagicWorkerStateReady);
- return 0;
- }
- void nfc_magic_worker_write(NfcMagicWorker* nfc_magic_worker) {
- bool card_found_notified = false;
- bool done = false;
- FuriHalNfcDevData nfc_data = {};
- NfcMagicDevice* magic_dev = nfc_magic_worker->magic_dev;
- NfcDeviceData* dev_data = nfc_magic_worker->dev_data;
- NfcProtocol dev_protocol = dev_data->protocol;
- while(nfc_magic_worker->state == NfcMagicWorkerStateWrite) {
- do {
- if(magic_dev->type == MagicTypeClassicGen1) {
- if(furi_hal_nfc_detect(&nfc_data, 200)) {
- magic_deactivate();
- magic_activate();
- if(!magic_gen1_wupa()) {
- FURI_LOG_E(TAG, "No card response to WUPA (not a magic card)");
- nfc_magic_worker->callback(
- NfcMagicWorkerEventWrongCard, nfc_magic_worker->context);
- done = true;
- break;
- }
- magic_deactivate();
- }
- magic_activate();
- if(magic_gen1_wupa()) {
- magic_gen1_data_access_cmd();
- MfClassicData* mfc_data = &dev_data->mf_classic_data;
- for(size_t i = 0; i < 64; i++) {
- FURI_LOG_D(TAG, "Writing block %d", i);
- if(!magic_gen1_write_blk(i, &mfc_data->block[i])) {
- FURI_LOG_E(TAG, "Failed to write %d block", i);
- done = true;
- nfc_magic_worker->callback(
- NfcMagicWorkerEventFail, nfc_magic_worker->context);
- break;
- }
- }
- done = true;
- nfc_magic_worker->callback(
- NfcMagicWorkerEventSuccess, nfc_magic_worker->context);
- break;
- }
- } else if(magic_dev->type == MagicTypeGen4) {
- if(furi_hal_nfc_detect(&nfc_data, 200)) {
- uint8_t gen4_config[MAGIC_GEN4_CONFIG_LEN];
- memcpy(gen4_config, MAGIC_DEFAULT_CONFIG, MAGIC_GEN4_CONFIG_LEN);
- uint32_t password = magic_dev->password;
- uint32_t cuid;
- size_t block_count = 64;
- MfClassicData* mfc_data;
- if(dev_protocol == NfcDeviceProtocolMifareClassic) {
- gen4_config[0] = 0x00;
- gen4_config[27] = 0x00;
- mfc_data = &dev_data->mf_classic_data;
- if(mfc_data->type == MfClassicType4k) block_count = 256;
- } else if(dev_protocol == NfcDeviceProtocolMifareUl) {
- MfUltralightData* mf_ul_data = &dev_data->mf_ul_data;
- gen4_config[0] = 0x01;
- switch(mf_ul_data->type) {
- case MfUltralightTypeUL11:
- case MfUltralightTypeUL21:
- // UL-C?
- // UL?
- default:
- gen4_config[27] = MagicGen4UltralightModeUL_EV1;
- break;
- case MfUltralightTypeNTAG203:
- case MfUltralightTypeNTAG213:
- case MfUltralightTypeNTAG215:
- case MfUltralightTypeNTAG216:
- case MfUltralightTypeNTAGI2C1K:
- case MfUltralightTypeNTAGI2C2K:
- case MfUltralightTypeNTAGI2CPlus1K:
- case MfUltralightTypeNTAGI2CPlus2K:
- gen4_config[27] = MagicGen4UltralightModeNTAG;
- block_count = 64 * 2;
- break;
- }
- }
- if(dev_data->nfc_data.uid_len == 4) {
- gen4_config[1] = MagicGen4UIDLengthSingle;
- } else if(dev_data->nfc_data.uid_len == 7) {
- gen4_config[1] = MagicGen4UIDLengthDouble;
- } else {
- FURI_LOG_E(TAG, "Unexpected UID length %d", dev_data->nfc_data.uid_len);
- nfc_magic_worker->callback(
- NfcMagicWorkerEventFail, nfc_magic_worker->context);
- done = true;
- break;
- }
- gen4_config[2] = (uint8_t)(password >> 24);
- gen4_config[3] = (uint8_t)(password >> 16);
- gen4_config[4] = (uint8_t)(password >> 8);
- gen4_config[5] = (uint8_t)password;
- if(dev_protocol == NfcDeviceProtocolMifareUl) {
- gen4_config[6] = MagicGen4ShadowModeHighSpeedIgnore;
- } else {
- gen4_config[6] = MagicGen4ShadowModeIgnore;
- }
- gen4_config[7] = 0x00;
- memset(gen4_config + 8, 0, 16);
- gen4_config[24] = dev_data->nfc_data.atqa[0];
- gen4_config[25] = dev_data->nfc_data.atqa[1];
- gen4_config[26] = dev_data->nfc_data.sak;
- gen4_config[28] = block_count;
- gen4_config[29] = 0x01;
- furi_hal_nfc_sleep();
- furi_hal_nfc_activate_nfca(200, &cuid);
- if(!magic_gen4_set_cfg(password, gen4_config, sizeof(gen4_config), false)) {
- nfc_magic_worker->callback(
- NfcMagicWorkerEventFail, nfc_magic_worker->context);
- done = true;
- break;
- }
- if(dev_protocol == NfcDeviceProtocolMifareClassic) {
- for(size_t i = 0; i < block_count; i++) {
- FURI_LOG_D(TAG, "Writing block %d", i);
- if(!magic_gen4_write_blk(password, i, mfc_data->block[i].value)) {
- FURI_LOG_E(TAG, "Failed to write %d block", i);
- nfc_magic_worker->callback(
- NfcMagicWorkerEventFail, nfc_magic_worker->context);
- done = true;
- break;
- }
- }
- } else if(dev_protocol == NfcDeviceProtocolMifareUl) {
- MfUltralightData* mf_ul_data = &dev_data->mf_ul_data;
- for(size_t i = 0; (i * 4) < mf_ul_data->data_read; i++) {
- size_t data_offset = i * 4;
- FURI_LOG_D(
- TAG,
- "Writing page %zu (%zu/%u)",
- i,
- data_offset,
- mf_ul_data->data_read);
- uint8_t* block = mf_ul_data->data + data_offset;
- if(!magic_gen4_write_blk(password, i, block)) {
- FURI_LOG_E(TAG, "Failed to write %zu page", i);
- nfc_magic_worker->callback(
- NfcMagicWorkerEventFail, nfc_magic_worker->context);
- done = true;
- break;
- }
- }
- uint8_t buffer[16] = {0};
- for(size_t i = 0; i < 8; i++) {
- memcpy(buffer, &mf_ul_data->signature[i * 4], 4); //-V1086
- if(!magic_gen4_write_blk(password, 0xF2 + i, buffer)) {
- FURI_LOG_E(TAG, "Failed to write signature block %d", i);
- nfc_magic_worker->callback(
- NfcMagicWorkerEventFail, nfc_magic_worker->context);
- done = true;
- break;
- }
- }
- buffer[0] = mf_ul_data->version.header;
- buffer[1] = mf_ul_data->version.vendor_id;
- buffer[2] = mf_ul_data->version.prod_type;
- buffer[3] = mf_ul_data->version.prod_subtype;
- if(!magic_gen4_write_blk(password, 0xFA, buffer)) {
- FURI_LOG_E(TAG, "Failed to write version block 0");
- nfc_magic_worker->callback(
- NfcMagicWorkerEventFail, nfc_magic_worker->context);
- done = true;
- break;
- }
- buffer[0] = mf_ul_data->version.prod_ver_major;
- buffer[1] = mf_ul_data->version.prod_ver_minor;
- buffer[2] = mf_ul_data->version.storage_size;
- buffer[3] = mf_ul_data->version.protocol_type;
- if(!magic_gen4_write_blk(password, 0xFB, buffer)) {
- FURI_LOG_E(TAG, "Failed to write version block 1");
- nfc_magic_worker->callback(
- NfcMagicWorkerEventFail, nfc_magic_worker->context);
- done = true;
- break;
- }
- }
- nfc_magic_worker->callback(
- NfcMagicWorkerEventSuccess, nfc_magic_worker->context);
- done = true;
- break;
- }
- }
- } while(false);
- if(done) break;
- if(card_found_notified) {
- nfc_magic_worker->callback(
- NfcMagicWorkerEventNoCardDetected, nfc_magic_worker->context);
- card_found_notified = false;
- }
- furi_delay_ms(300);
- }
- magic_deactivate();
- }
- void nfc_magic_worker_check(NfcMagicWorker* nfc_magic_worker) {
- FuriHalNfcDevData nfc_data = {};
- NfcMagicDevice* magic_dev = nfc_magic_worker->magic_dev;
- bool card_found_notified = false;
- uint8_t gen4_config[MAGIC_GEN4_CONFIG_LEN];
- while(nfc_magic_worker->state == NfcMagicWorkerStateCheck) {
- magic_activate();
- if(magic_gen1_wupa()) {
- magic_dev->type = MagicTypeClassicGen1;
- if(!card_found_notified) {
- nfc_magic_worker->callback(
- NfcMagicWorkerEventCardDetected, nfc_magic_worker->context);
- card_found_notified = true;
- }
- if(furi_hal_nfc_detect(&nfc_data, 200)) {
- magic_dev->cuid = nfc_data.cuid;
- magic_dev->uid_len = nfc_data.uid_len;
- } else {
- // wrong BCC
- magic_dev->uid_len = 4;
- }
- nfc_magic_worker->callback(NfcMagicWorkerEventSuccess, nfc_magic_worker->context);
- break;
- } else {
- magic_deactivate();
- magic_activate();
- if(furi_hal_nfc_detect(&nfc_data, 200)) {
- magic_dev->cuid = nfc_data.cuid;
- magic_dev->uid_len = nfc_data.uid_len;
- if(magic_gen4_get_cfg(magic_dev->password, gen4_config)) {
- magic_dev->type = MagicTypeGen4;
- if(!card_found_notified) {
- nfc_magic_worker->callback(
- NfcMagicWorkerEventCardDetected, nfc_magic_worker->context);
- card_found_notified = true;
- }
- nfc_magic_worker->callback(
- NfcMagicWorkerEventSuccess, nfc_magic_worker->context);
- } else {
- nfc_magic_worker->callback(
- NfcMagicWorkerEventWrongCard, nfc_magic_worker->context);
- card_found_notified = true;
- }
- break;
- } else {
- if(card_found_notified) {
- nfc_magic_worker->callback(
- NfcMagicWorkerEventNoCardDetected, nfc_magic_worker->context);
- card_found_notified = false;
- }
- }
- }
- magic_deactivate();
- furi_delay_ms(300);
- }
- magic_deactivate();
- }
- void nfc_magic_worker_rekey(NfcMagicWorker* nfc_magic_worker) {
- NfcMagicDevice* magic_dev = nfc_magic_worker->magic_dev;
- bool card_found_notified = false;
- if(magic_dev->type != MagicTypeGen4) {
- nfc_magic_worker->callback(NfcMagicWorkerEventCardDetected, nfc_magic_worker->context);
- return;
- }
- while(nfc_magic_worker->state == NfcMagicWorkerStateRekey) {
- magic_activate();
- uint32_t cuid;
- furi_hal_nfc_activate_nfca(200, &cuid);
- if(cuid != magic_dev->cuid) {
- if(card_found_notified) {
- nfc_magic_worker->callback(
- NfcMagicWorkerEventNoCardDetected, nfc_magic_worker->context);
- card_found_notified = false;
- }
- continue;
- }
- nfc_magic_worker->callback(NfcMagicWorkerEventCardDetected, nfc_magic_worker->context);
- card_found_notified = true;
- if(magic_gen4_set_pwd(magic_dev->password, nfc_magic_worker->new_password)) {
- magic_dev->password = nfc_magic_worker->new_password;
- nfc_magic_worker->callback(NfcMagicWorkerEventSuccess, nfc_magic_worker->context);
- break;
- }
- if(card_found_notified) { //-V547
- nfc_magic_worker->callback(
- NfcMagicWorkerEventNoCardDetected, nfc_magic_worker->context);
- card_found_notified = false;
- }
- furi_delay_ms(300);
- }
- magic_deactivate();
- }
- void nfc_magic_worker_wipe(NfcMagicWorker* nfc_magic_worker) {
- NfcMagicDevice* magic_dev = nfc_magic_worker->magic_dev;
- bool card_found_notified = false;
- bool card_wiped = false;
- MfClassicBlock block;
- memset(&block, 0, sizeof(MfClassicBlock));
- MfClassicBlock empty_block;
- memset(&empty_block, 0, sizeof(MfClassicBlock));
- MfClassicBlock trailer_block;
- memset(&trailer_block, 0xff, sizeof(MfClassicBlock));
- block.value[0] = 0x01;
- block.value[1] = 0x02;
- block.value[2] = 0x03;
- block.value[3] = 0x04;
- block.value[4] = 0x04;
- block.value[5] = 0x08;
- block.value[6] = 0x04;
- trailer_block.value[7] = 0x07;
- trailer_block.value[8] = 0x80;
- trailer_block.value[9] = 0x69;
- while(nfc_magic_worker->state == NfcMagicWorkerStateWipe) {
- do {
- magic_deactivate();
- furi_delay_ms(300);
- if(!magic_activate()) break;
- if(magic_dev->type == MagicTypeClassicGen1) {
- if(!magic_gen1_wupa()) break;
- if(!card_found_notified) {
- nfc_magic_worker->callback(
- NfcMagicWorkerEventCardDetected, nfc_magic_worker->context);
- card_found_notified = true;
- }
- if(!magic_gen1_data_access_cmd()) break;
- if(!magic_gen1_write_blk(0, &block)) break;
- for(size_t i = 1; i < 64; i++) {
- FURI_LOG_D(TAG, "Wiping block %d", i);
- bool success = false;
- if((i | 0x03) == i) {
- success = magic_gen1_write_blk(i, &trailer_block);
- } else {
- success = magic_gen1_write_blk(i, &empty_block);
- }
- if(!success) {
- FURI_LOG_E(TAG, "Failed to write %d block", i);
- nfc_magic_worker->callback(
- NfcMagicWorkerEventFail, nfc_magic_worker->context);
- break;
- }
- }
- card_wiped = true;
- nfc_magic_worker->callback(NfcMagicWorkerEventSuccess, nfc_magic_worker->context);
- } else if(magic_dev->type == MagicTypeGen4) {
- uint32_t cuid;
- if(!furi_hal_nfc_activate_nfca(200, &cuid)) break;
- if(cuid != magic_dev->cuid) break;
- if(!card_found_notified) {
- nfc_magic_worker->callback(
- NfcMagicWorkerEventCardDetected, nfc_magic_worker->context);
- card_found_notified = true;
- }
- if(!magic_gen4_wipe(magic_dev->password)) break;
- card_wiped = true;
- nfc_magic_worker->callback(NfcMagicWorkerEventSuccess, nfc_magic_worker->context);
- }
- } while(false);
- if(card_wiped) break;
- if(card_found_notified) {
- nfc_magic_worker->callback(
- NfcMagicWorkerEventNoCardDetected, nfc_magic_worker->context);
- card_found_notified = false;
- }
- }
- magic_deactivate();
- }
|