 bf15d3ce74
			
		
	
	
		bf15d3ce74
		
			
		
	
	
	
	
		
			
			* NFC: Improved MFC emulation on some readers * NFC: Improved emulation on some readers (part 2): Some Android devices don't like this * NFC: Improved emulation on some readers (part 3): I knew that during the emulation timings are critical, but one log breaks all... * NFC: Improved emulation on some readers (part 4): Add fixes to Detect reader and refactor code * NFC: Improved emulation on some readers (part 5) * NFC: Improved emulation on some readers (part 6): GUI doesn't update without delay * NFC: Improved emulation on some readers (part 7): Reworked emulation flow, some bug fixes and improvements Co-authored-by: あく <alleteam@gmail.com> Co-authored-by: gornekich <n.gorbadey@gmail.com>
		
			
				
	
	
		
			141 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			141 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include "nfca.h"
 | |
| #include <string.h>
 | |
| #include <stdio.h>
 | |
| #include <furi.h>
 | |
| 
 | |
| #define NFCA_CRC_INIT (0x6363)
 | |
| 
 | |
| #define NFCA_F_SIG (13560000.0)
 | |
| #define T_SIG 7374 //73.746ns*100
 | |
| #define T_SIG_x8 58992 //T_SIG*8
 | |
| #define T_SIG_x8_x8 471936 //T_SIG*8*8
 | |
| #define T_SIG_x8_x9 530928 //T_SIG*8*9
 | |
| 
 | |
| #define NFCA_SIGNAL_MAX_EDGES (1350)
 | |
| 
 | |
| typedef struct {
 | |
|     uint8_t cmd;
 | |
|     uint8_t param;
 | |
| } nfca_cmd_rats;
 | |
| 
 | |
| static uint8_t nfca_default_ats[] = {0x05, 0x78, 0x80, 0x80, 0x00};
 | |
| 
 | |
| static uint8_t nfca_halt_req[] = {NFCA_CMD_HALT, 0x00};
 | |
| 
 | |
| uint16_t nfca_get_crc16(uint8_t* buff, uint16_t len) {
 | |
|     uint16_t crc = NFCA_CRC_INIT;
 | |
|     uint8_t byte = 0;
 | |
| 
 | |
|     for(uint8_t i = 0; i < len; i++) {
 | |
|         byte = buff[i];
 | |
|         byte ^= (uint8_t)(crc & 0xff);
 | |
|         byte ^= byte << 4;
 | |
|         crc = (crc >> 8) ^ (((uint16_t)byte) << 8) ^ (((uint16_t)byte) << 3) ^
 | |
|               (((uint16_t)byte) >> 4);
 | |
|     }
 | |
| 
 | |
|     return crc;
 | |
| }
 | |
| 
 | |
| void nfca_append_crc16(uint8_t* buff, uint16_t len) {
 | |
|     uint16_t crc = nfca_get_crc16(buff, len);
 | |
|     buff[len] = (uint8_t)crc;
 | |
|     buff[len + 1] = (uint8_t)(crc >> 8);
 | |
| }
 | |
| 
 | |
| bool nfca_emulation_handler(
 | |
|     uint8_t* buff_rx,
 | |
|     uint16_t buff_rx_len,
 | |
|     uint8_t* buff_tx,
 | |
|     uint16_t* buff_tx_len) {
 | |
|     bool halt = false;
 | |
|     uint8_t rx_bytes = buff_rx_len / 8;
 | |
| 
 | |
|     if(rx_bytes == sizeof(nfca_halt_req) && !memcmp(buff_rx, nfca_halt_req, rx_bytes)) {
 | |
|         halt = true;
 | |
|     } else if(rx_bytes == sizeof(nfca_cmd_rats) && buff_rx[0] == NFCA_CMD_RATS) {
 | |
|         memcpy(buff_tx, nfca_default_ats, sizeof(nfca_default_ats));
 | |
|         *buff_tx_len = sizeof(nfca_default_ats) * 8;
 | |
|     }
 | |
| 
 | |
|     return halt;
 | |
| }
 | |
| 
 | |
| static void nfca_add_bit(DigitalSignal* signal, bool bit) {
 | |
|     if(bit) {
 | |
|         signal->start_level = true;
 | |
|         for(size_t i = 0; i < 7; i++) {
 | |
|             signal->edge_timings[i] = T_SIG_x8;
 | |
|         }
 | |
|         signal->edge_timings[7] = T_SIG_x8_x9;
 | |
|         signal->edge_cnt = 8;
 | |
|     } else {
 | |
|         signal->start_level = false;
 | |
|         signal->edge_timings[0] = T_SIG_x8_x8;
 | |
|         for(size_t i = 1; i < 9; i++) {
 | |
|             signal->edge_timings[i] = T_SIG_x8;
 | |
|         }
 | |
|         signal->edge_cnt = 9;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void nfca_add_byte(NfcaSignal* nfca_signal, uint8_t byte, bool parity) {
 | |
|     for(uint8_t i = 0; i < 8; i++) {
 | |
|         if(byte & (1 << i)) {
 | |
|             digital_signal_append(nfca_signal->tx_signal, nfca_signal->one);
 | |
|         } else {
 | |
|             digital_signal_append(nfca_signal->tx_signal, nfca_signal->zero);
 | |
|         }
 | |
|     }
 | |
|     if(parity) {
 | |
|         digital_signal_append(nfca_signal->tx_signal, nfca_signal->one);
 | |
|     } else {
 | |
|         digital_signal_append(nfca_signal->tx_signal, nfca_signal->zero);
 | |
|     }
 | |
| }
 | |
| 
 | |
| NfcaSignal* nfca_signal_alloc() {
 | |
|     NfcaSignal* nfca_signal = malloc(sizeof(NfcaSignal));
 | |
|     nfca_signal->one = digital_signal_alloc(10);
 | |
|     nfca_signal->zero = digital_signal_alloc(10);
 | |
|     nfca_add_bit(nfca_signal->one, true);
 | |
|     nfca_add_bit(nfca_signal->zero, false);
 | |
|     nfca_signal->tx_signal = digital_signal_alloc(NFCA_SIGNAL_MAX_EDGES);
 | |
| 
 | |
|     return nfca_signal;
 | |
| }
 | |
| 
 | |
| void nfca_signal_free(NfcaSignal* nfca_signal) {
 | |
|     furi_assert(nfca_signal);
 | |
| 
 | |
|     digital_signal_free(nfca_signal->one);
 | |
|     digital_signal_free(nfca_signal->zero);
 | |
|     digital_signal_free(nfca_signal->tx_signal);
 | |
|     free(nfca_signal);
 | |
| }
 | |
| 
 | |
| void nfca_signal_encode(NfcaSignal* nfca_signal, uint8_t* data, uint16_t bits, uint8_t* parity) {
 | |
|     furi_assert(nfca_signal);
 | |
|     furi_assert(data);
 | |
|     furi_assert(parity);
 | |
| 
 | |
|     nfca_signal->tx_signal->edge_cnt = 0;
 | |
|     nfca_signal->tx_signal->start_level = true;
 | |
|     // Start of frame
 | |
|     digital_signal_append(nfca_signal->tx_signal, nfca_signal->one);
 | |
| 
 | |
|     if(bits < 8) {
 | |
|         for(size_t i = 0; i < bits; i++) {
 | |
|             if(FURI_BIT(data[0], i)) {
 | |
|                 digital_signal_append(nfca_signal->tx_signal, nfca_signal->one);
 | |
|             } else {
 | |
|                 digital_signal_append(nfca_signal->tx_signal, nfca_signal->zero);
 | |
|             }
 | |
|         }
 | |
|     } else {
 | |
|         for(size_t i = 0; i < bits / 8; i++) {
 | |
|             nfca_add_byte(nfca_signal, data[i], parity[i / 8] & (1 << (7 - (i & 0x07))));
 | |
|         }
 | |
|     }
 | |
| }
 |