Co-authored-by: あく <alleteam@gmail.com>
This commit is contained in:
		
							parent
							
								
									d09c59fd2f
								
							
						
					
					
						commit
						86a64487cb
					
				| @ -5,7 +5,6 @@ | |||||||
| #define TAG "Magic" | #define TAG "Magic" | ||||||
| 
 | 
 | ||||||
| #define MAGIC_CMD_WUPA (0x40) | #define MAGIC_CMD_WUPA (0x40) | ||||||
| #define MAGIC_CMD_WIPE (0x41) |  | ||||||
| #define MAGIC_CMD_ACCESS (0x43) | #define MAGIC_CMD_ACCESS (0x43) | ||||||
| 
 | 
 | ||||||
| #define MAGIC_MIFARE_READ_CMD (0x30) | #define MAGIC_MIFARE_READ_CMD (0x30) | ||||||
| @ -144,32 +143,3 @@ bool magic_gen1_write_blk(uint8_t block_num, MfClassicBlock* data) { | |||||||
| 
 | 
 | ||||||
|     return write_success; |     return write_success; | ||||||
| } | } | ||||||
| 
 |  | ||||||
| bool magic_gen1_wipe() { |  | ||||||
|     bool wipe_success = false; |  | ||||||
|     uint8_t tx_data[MAGIC_BUFFER_SIZE] = {}; |  | ||||||
|     uint8_t rx_data[MAGIC_BUFFER_SIZE] = {}; |  | ||||||
|     uint16_t rx_len = 0; |  | ||||||
|     FuriHalNfcReturn ret = 0; |  | ||||||
| 
 |  | ||||||
|     do { |  | ||||||
|         tx_data[0] = MAGIC_CMD_WIPE; |  | ||||||
|         ret = furi_hal_nfc_ll_txrx_bits( |  | ||||||
|             tx_data, |  | ||||||
|             8, |  | ||||||
|             rx_data, |  | ||||||
|             sizeof(rx_data), |  | ||||||
|             &rx_len, |  | ||||||
|             FURI_HAL_NFC_LL_TXRX_FLAGS_CRC_TX_MANUAL | FURI_HAL_NFC_LL_TXRX_FLAGS_AGC_ON | |  | ||||||
|                 FURI_HAL_NFC_LL_TXRX_FLAGS_CRC_RX_KEEP, |  | ||||||
|             furi_hal_nfc_ll_ms2fc(2000)); |  | ||||||
| 
 |  | ||||||
|         if(ret != FuriHalNfcReturnIncompleteByte) break; |  | ||||||
|         if(rx_len != 4) break; |  | ||||||
|         if(rx_data[0] != MAGIC_ACK) break; |  | ||||||
| 
 |  | ||||||
|         wipe_success = true; |  | ||||||
|     } while(false); |  | ||||||
| 
 |  | ||||||
|     return wipe_success; |  | ||||||
| } |  | ||||||
| @ -9,5 +9,3 @@ bool magic_gen1_read_block(uint8_t block_num, MfClassicBlock* data); | |||||||
| bool magic_gen1_data_access_cmd(); | bool magic_gen1_data_access_cmd(); | ||||||
| 
 | 
 | ||||||
| bool magic_gen1_write_blk(uint8_t block_num, MfClassicBlock* data); | bool magic_gen1_write_blk(uint8_t block_num, MfClassicBlock* data); | ||||||
| 
 |  | ||||||
| bool magic_gen1_wipe(); |  | ||||||
							
								
								
									
										175
									
								
								applications/external/nfc_magic/lib/magic/magic.c
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										175
									
								
								applications/external/nfc_magic/lib/magic/magic.c
									
									
									
									
										vendored
									
									
								
							| @ -1,175 +0,0 @@ | |||||||
| #include "classic_gen1.h" |  | ||||||
| 
 |  | ||||||
| #include <furi_hal_nfc.h> |  | ||||||
| 
 |  | ||||||
| #define TAG "Magic" |  | ||||||
| 
 |  | ||||||
| #define MAGIC_CMD_WUPA (0x40) |  | ||||||
| #define MAGIC_CMD_WIPE (0x41) |  | ||||||
| #define MAGIC_CMD_ACCESS (0x43) |  | ||||||
| 
 |  | ||||||
| #define MAGIC_MIFARE_READ_CMD (0x30) |  | ||||||
| #define MAGIC_MIFARE_WRITE_CMD (0xA0) |  | ||||||
| 
 |  | ||||||
| #define MAGIC_ACK (0x0A) |  | ||||||
| 
 |  | ||||||
| #define MAGIC_BUFFER_SIZE (32) |  | ||||||
| 
 |  | ||||||
| bool magic_gen1_wupa() { |  | ||||||
|     bool magic_activated = false; |  | ||||||
|     uint8_t tx_data[MAGIC_BUFFER_SIZE] = {}; |  | ||||||
|     uint8_t rx_data[MAGIC_BUFFER_SIZE] = {}; |  | ||||||
|     uint16_t rx_len = 0; |  | ||||||
|     FuriHalNfcReturn ret = 0; |  | ||||||
| 
 |  | ||||||
|     do { |  | ||||||
|         // Start communication
 |  | ||||||
|         tx_data[0] = MAGIC_CMD_WUPA; |  | ||||||
|         ret = furi_hal_nfc_ll_txrx_bits( |  | ||||||
|             tx_data, |  | ||||||
|             7, |  | ||||||
|             rx_data, |  | ||||||
|             sizeof(rx_data), |  | ||||||
|             &rx_len, |  | ||||||
|             FURI_HAL_NFC_LL_TXRX_FLAGS_CRC_TX_MANUAL | FURI_HAL_NFC_LL_TXRX_FLAGS_AGC_ON | |  | ||||||
|                 FURI_HAL_NFC_LL_TXRX_FLAGS_CRC_RX_KEEP, |  | ||||||
|             furi_hal_nfc_ll_ms2fc(20)); |  | ||||||
|         if(ret != FuriHalNfcReturnIncompleteByte) break; |  | ||||||
|         if(rx_len != 4) break; |  | ||||||
|         if(rx_data[0] != MAGIC_ACK) break; |  | ||||||
|         magic_activated = true; |  | ||||||
|     } while(false); |  | ||||||
| 
 |  | ||||||
|     return magic_activated; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| bool magic_gen1_data_access_cmd() { |  | ||||||
|     bool write_cmd_success = false; |  | ||||||
|     uint8_t tx_data[MAGIC_BUFFER_SIZE] = {}; |  | ||||||
|     uint8_t rx_data[MAGIC_BUFFER_SIZE] = {}; |  | ||||||
|     uint16_t rx_len = 0; |  | ||||||
|     FuriHalNfcReturn ret = 0; |  | ||||||
| 
 |  | ||||||
|     do { |  | ||||||
|         tx_data[0] = MAGIC_CMD_ACCESS; |  | ||||||
|         ret = furi_hal_nfc_ll_txrx_bits( |  | ||||||
|             tx_data, |  | ||||||
|             8, |  | ||||||
|             rx_data, |  | ||||||
|             sizeof(rx_data), |  | ||||||
|             &rx_len, |  | ||||||
|             FURI_HAL_NFC_LL_TXRX_FLAGS_CRC_TX_MANUAL | FURI_HAL_NFC_LL_TXRX_FLAGS_AGC_ON | |  | ||||||
|                 FURI_HAL_NFC_LL_TXRX_FLAGS_CRC_RX_KEEP, |  | ||||||
|             furi_hal_nfc_ll_ms2fc(20)); |  | ||||||
|         if(ret != FuriHalNfcReturnIncompleteByte) break; |  | ||||||
|         if(rx_len != 4) break; |  | ||||||
|         if(rx_data[0] != MAGIC_ACK) break; |  | ||||||
| 
 |  | ||||||
|         write_cmd_success = true; |  | ||||||
|     } while(false); |  | ||||||
| 
 |  | ||||||
|     return write_cmd_success; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| bool magic_gen1_read_block(uint8_t block_num, MfClassicBlock* data) { |  | ||||||
|     furi_assert(data); |  | ||||||
| 
 |  | ||||||
|     bool read_success = false; |  | ||||||
| 
 |  | ||||||
|     uint8_t tx_data[MAGIC_BUFFER_SIZE] = {}; |  | ||||||
|     uint8_t rx_data[MAGIC_BUFFER_SIZE] = {}; |  | ||||||
|     uint16_t rx_len = 0; |  | ||||||
|     FuriHalNfcReturn ret = 0; |  | ||||||
| 
 |  | ||||||
|     do { |  | ||||||
|         tx_data[0] = MAGIC_MIFARE_READ_CMD; |  | ||||||
|         tx_data[1] = block_num; |  | ||||||
|         ret = furi_hal_nfc_ll_txrx_bits( |  | ||||||
|             tx_data, |  | ||||||
|             2 * 8, |  | ||||||
|             rx_data, |  | ||||||
|             sizeof(rx_data), |  | ||||||
|             &rx_len, |  | ||||||
|             FURI_HAL_NFC_LL_TXRX_FLAGS_AGC_ON, |  | ||||||
|             furi_hal_nfc_ll_ms2fc(20)); |  | ||||||
| 
 |  | ||||||
|         if(ret != FuriHalNfcReturnOk) break; |  | ||||||
|         if(rx_len != 16 * 8) break; |  | ||||||
|         memcpy(data->value, rx_data, sizeof(data->value)); |  | ||||||
|         read_success = true; |  | ||||||
|     } while(false); |  | ||||||
| 
 |  | ||||||
|     return read_success; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| bool magic_gen1_write_blk(uint8_t block_num, MfClassicBlock* data) { |  | ||||||
|     furi_assert(data); |  | ||||||
| 
 |  | ||||||
|     bool write_success = false; |  | ||||||
|     uint8_t tx_data[MAGIC_BUFFER_SIZE] = {}; |  | ||||||
|     uint8_t rx_data[MAGIC_BUFFER_SIZE] = {}; |  | ||||||
|     uint16_t rx_len = 0; |  | ||||||
|     FuriHalNfcReturn ret = 0; |  | ||||||
| 
 |  | ||||||
|     do { |  | ||||||
|         tx_data[0] = MAGIC_MIFARE_WRITE_CMD; |  | ||||||
|         tx_data[1] = block_num; |  | ||||||
|         ret = furi_hal_nfc_ll_txrx_bits( |  | ||||||
|             tx_data, |  | ||||||
|             2 * 8, |  | ||||||
|             rx_data, |  | ||||||
|             sizeof(rx_data), |  | ||||||
|             &rx_len, |  | ||||||
|             FURI_HAL_NFC_LL_TXRX_FLAGS_AGC_ON | FURI_HAL_NFC_LL_TXRX_FLAGS_CRC_RX_KEEP, |  | ||||||
|             furi_hal_nfc_ll_ms2fc(20)); |  | ||||||
|         if(ret != FuriHalNfcReturnIncompleteByte) break; |  | ||||||
|         if(rx_len != 4) break; |  | ||||||
|         if(rx_data[0] != MAGIC_ACK) break; |  | ||||||
| 
 |  | ||||||
|         memcpy(tx_data, data->value, sizeof(data->value)); |  | ||||||
|         ret = furi_hal_nfc_ll_txrx_bits( |  | ||||||
|             tx_data, |  | ||||||
|             16 * 8, |  | ||||||
|             rx_data, |  | ||||||
|             sizeof(rx_data), |  | ||||||
|             &rx_len, |  | ||||||
|             FURI_HAL_NFC_LL_TXRX_FLAGS_AGC_ON | FURI_HAL_NFC_LL_TXRX_FLAGS_CRC_RX_KEEP, |  | ||||||
|             furi_hal_nfc_ll_ms2fc(20)); |  | ||||||
|         if(ret != FuriHalNfcReturnIncompleteByte) break; |  | ||||||
|         if(rx_len != 4) break; |  | ||||||
|         if(rx_data[0] != MAGIC_ACK) break; |  | ||||||
| 
 |  | ||||||
|         write_success = true; |  | ||||||
|     } while(false); |  | ||||||
| 
 |  | ||||||
|     return write_success; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| bool magic_gen1_wipe() { |  | ||||||
|     bool wipe_success = false; |  | ||||||
|     uint8_t tx_data[MAGIC_BUFFER_SIZE] = {}; |  | ||||||
|     uint8_t rx_data[MAGIC_BUFFER_SIZE] = {}; |  | ||||||
|     uint16_t rx_len = 0; |  | ||||||
|     FuriHalNfcReturn ret = 0; |  | ||||||
| 
 |  | ||||||
|     do { |  | ||||||
|         tx_data[0] = MAGIC_CMD_WIPE; |  | ||||||
|         ret = furi_hal_nfc_ll_txrx_bits( |  | ||||||
|             tx_data, |  | ||||||
|             8, |  | ||||||
|             rx_data, |  | ||||||
|             sizeof(rx_data), |  | ||||||
|             &rx_len, |  | ||||||
|             FURI_HAL_NFC_LL_TXRX_FLAGS_CRC_TX_MANUAL | FURI_HAL_NFC_LL_TXRX_FLAGS_AGC_ON | |  | ||||||
|                 FURI_HAL_NFC_LL_TXRX_FLAGS_CRC_RX_KEEP, |  | ||||||
|             furi_hal_nfc_ll_ms2fc(2000)); |  | ||||||
| 
 |  | ||||||
|         if(ret != FuriHalNfcReturnIncompleteByte) break; |  | ||||||
|         if(rx_len != 4) break; |  | ||||||
|         if(rx_data[0] != MAGIC_ACK) break; |  | ||||||
| 
 |  | ||||||
|         wipe_success = true; |  | ||||||
|     } while(false); |  | ||||||
| 
 |  | ||||||
|     return wipe_success; |  | ||||||
| } |  | ||||||
| @ -92,51 +92,49 @@ void nfc_magic_worker_write(NfcMagicWorker* nfc_magic_worker) { | |||||||
| 
 | 
 | ||||||
|     while(nfc_magic_worker->state == NfcMagicWorkerStateWrite) { |     while(nfc_magic_worker->state == NfcMagicWorkerStateWrite) { | ||||||
|         do { |         do { | ||||||
|             if(furi_hal_nfc_detect(&nfc_data, 200)) { |             if(magic_dev->type == MagicTypeClassicGen1) { | ||||||
|                 if(nfc_data.cuid != magic_dev->cuid) break; |                 if(furi_hal_nfc_detect(&nfc_data, 200)) { | ||||||
|                 if(!card_found_notified) { |                     magic_deactivate(); | ||||||
|                     nfc_magic_worker->callback( |                     magic_activate(); | ||||||
|                         NfcMagicWorkerEventCardDetected, nfc_magic_worker->context); |  | ||||||
|                     card_found_notified = true; |  | ||||||
|                 } |  | ||||||
|                 furi_hal_nfc_sleep(); |  | ||||||
| 
 |  | ||||||
|                 magic_activate(); |  | ||||||
|                 if(magic_dev->type == MagicTypeClassicGen1) { |  | ||||||
|                     if(dev_protocol != NfcDeviceProtocolMifareClassic) break; |  | ||||||
|                     MfClassicData* mfc_data = &dev_data->mf_classic_data; |  | ||||||
| 
 |  | ||||||
|                     if(mfc_data->type != MfClassicType1k) break; |  | ||||||
|                     if(!magic_gen1_wupa()) { |                     if(!magic_gen1_wupa()) { | ||||||
|                         FURI_LOG_E(TAG, "Not Magic card"); |                         FURI_LOG_E(TAG, "No card response to WUPA (not a magic card)"); | ||||||
|                         nfc_magic_worker->callback( |                         nfc_magic_worker->callback( | ||||||
|                             NfcMagicWorkerEventWrongCard, nfc_magic_worker->context); |                             NfcMagicWorkerEventWrongCard, nfc_magic_worker->context); | ||||||
|                         done = true; |                         done = true; | ||||||
|                         break; |                         break; | ||||||
|                     } |                     } | ||||||
|  |                     magic_deactivate(); | ||||||
|  |                 } | ||||||
|  |                 magic_activate(); | ||||||
|  |                 if(magic_gen1_wupa()) { | ||||||
|                     if(!magic_gen1_data_access_cmd()) { |                     if(!magic_gen1_data_access_cmd()) { | ||||||
|                         FURI_LOG_E(TAG, "Not Magic card"); |                         FURI_LOG_E( | ||||||
|  |                             TAG, "No card response to data access command (not a magic card)"); | ||||||
|                         nfc_magic_worker->callback( |                         nfc_magic_worker->callback( | ||||||
|                             NfcMagicWorkerEventWrongCard, nfc_magic_worker->context); |                             NfcMagicWorkerEventWrongCard, nfc_magic_worker->context); | ||||||
|                         done = true; |                         done = true; | ||||||
|                         break; |                         break; | ||||||
|                     } |                     } | ||||||
|  | 
 | ||||||
|  |                     MfClassicData* mfc_data = &dev_data->mf_classic_data; | ||||||
|                     for(size_t i = 0; i < 64; i++) { |                     for(size_t i = 0; i < 64; i++) { | ||||||
|                         FURI_LOG_D(TAG, "Writing block %d", i); |                         FURI_LOG_D(TAG, "Writing block %d", i); | ||||||
|                         if(!magic_gen1_write_blk(i, &mfc_data->block[i])) { |                         if(!magic_gen1_write_blk(i, &mfc_data->block[i])) { | ||||||
|                             FURI_LOG_E(TAG, "Failed to write %d block", i); |                             FURI_LOG_E(TAG, "Failed to write %d block", i); | ||||||
|  |                             done = true; | ||||||
|                             nfc_magic_worker->callback( |                             nfc_magic_worker->callback( | ||||||
|                                 NfcMagicWorkerEventFail, nfc_magic_worker->context); |                                 NfcMagicWorkerEventFail, nfc_magic_worker->context); | ||||||
|                             done = true; |  | ||||||
|                             break; |                             break; | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
| 
 | 
 | ||||||
|  |                     done = true; | ||||||
|                     nfc_magic_worker->callback( |                     nfc_magic_worker->callback( | ||||||
|                         NfcMagicWorkerEventSuccess, nfc_magic_worker->context); |                         NfcMagicWorkerEventSuccess, nfc_magic_worker->context); | ||||||
|                     done = true; |  | ||||||
|                     break; |                     break; | ||||||
|                 } else if(magic_dev->type == MagicTypeGen4) { |                 } | ||||||
|  |             } else if(magic_dev->type == MagicTypeGen4) { | ||||||
|  |                 if(furi_hal_nfc_detect(&nfc_data, 200)) { | ||||||
|                     uint8_t gen4_config[28]; |                     uint8_t gen4_config[28]; | ||||||
|                     uint32_t password = magic_dev->password; |                     uint32_t password = magic_dev->password; | ||||||
| 
 | 
 | ||||||
| @ -196,6 +194,7 @@ void nfc_magic_worker_write(NfcMagicWorker* nfc_magic_worker) { | |||||||
|                     gen4_config[25] = dev_data->nfc_data.atqa[1]; |                     gen4_config[25] = dev_data->nfc_data.atqa[1]; | ||||||
|                     gen4_config[26] = dev_data->nfc_data.sak; |                     gen4_config[26] = dev_data->nfc_data.sak; | ||||||
| 
 | 
 | ||||||
|  |                     furi_hal_nfc_sleep(); | ||||||
|                     furi_hal_nfc_activate_nfca(200, &cuid); |                     furi_hal_nfc_activate_nfca(200, &cuid); | ||||||
|                     if(!magic_gen4_set_cfg(password, gen4_config, sizeof(gen4_config), false)) { |                     if(!magic_gen4_set_cfg(password, gen4_config, sizeof(gen4_config), false)) { | ||||||
|                         nfc_magic_worker->callback( |                         nfc_magic_worker->callback( | ||||||
| @ -394,6 +393,11 @@ void nfc_magic_worker_wipe(NfcMagicWorker* nfc_magic_worker) { | |||||||
| 
 | 
 | ||||||
|     MfClassicBlock block; |     MfClassicBlock block; | ||||||
|     memset(&block, 0, sizeof(MfClassicBlock)); |     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[0] = 0x01; | ||||||
|     block.value[1] = 0x02; |     block.value[1] = 0x02; | ||||||
|     block.value[2] = 0x03; |     block.value[2] = 0x03; | ||||||
| @ -402,6 +406,10 @@ void nfc_magic_worker_wipe(NfcMagicWorker* nfc_magic_worker) { | |||||||
|     block.value[5] = 0x08; |     block.value[5] = 0x08; | ||||||
|     block.value[6] = 0x04; |     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) { |     while(nfc_magic_worker->state == NfcMagicWorkerStateWipe) { | ||||||
|         do { |         do { | ||||||
|             magic_deactivate(); |             magic_deactivate(); | ||||||
| @ -415,10 +423,26 @@ void nfc_magic_worker_wipe(NfcMagicWorker* nfc_magic_worker) { | |||||||
|                     card_found_notified = true; |                     card_found_notified = true; | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 if(!magic_gen1_wipe()) break; |  | ||||||
|                 if(!magic_gen1_data_access_cmd()) break; |                 if(!magic_gen1_data_access_cmd()) break; | ||||||
|                 if(!magic_gen1_write_blk(0, &block)) 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; |                 card_wiped = true; | ||||||
|                 nfc_magic_worker->callback(NfcMagicWorkerEventSuccess, nfc_magic_worker->context); |                 nfc_magic_worker->callback(NfcMagicWorkerEventSuccess, nfc_magic_worker->context); | ||||||
|             } else if(magic_dev->type == MagicTypeGen4) { |             } else if(magic_dev->type == MagicTypeGen4) { | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 AloneLiberty
						AloneLiberty