 490447bbd4
			
		
	
	
		490447bbd4
		
			
		
	
	
	
	
		
			
			* NFC: gen4 gtu detect in magic app * NFC: more support for GTU card * NFC: Fix Gen1 in Magic * Allow double UIDs for MFClassic on GTU cards * NFC: Small magic app tweaks * nfc magic: notify card event on wiping * nfc magic: fix power consumption * nfc magic: disable i2c writing and fix wipe loop * NfcMagic: correct formatting in printf * NfcMagic: correct formatting in printf, proper version * nfc_magic: rework card found notification and gen4 wiping Co-authored-by: あく <alleteam@gmail.com>
		
			
				
	
	
		
			175 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			175 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #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;
 | |
| } |