mirror of
https://github.com/DarkFlippers/unleashed-firmware
synced 2025-01-10 03:38:53 +00:00
9c59bcd776
* nfc: add new read scene * lib: refactore nfc library * mifare desfire: add read card fuction * lib nfc: add auto read worker * nfc: add supported cards * nfc: add mifare classic read success scene * nfc: add troyka support * submodule: update protobuf * nfc: mifare classic keys cache * nfc: rework mifare classic key cache * Correct spelling * nfc: add user dictionary * nfc: introduce block read map in fff * nfc: rework dict attack * nfc: improve dict attack * nfc: rework mifare classic format * nfc: rework MFC read with Reader * nfc: add gui for MFC read success scene * nfc: fix dict attack view gui * nfc: add retry and exit confirm scenes * nfc: add retry and exit scenes navigation * nfc: check user dictionary * nfc: remove unused scenes * nfc: rename functions in nfc worker * nfc: rename mf_classic_dict_attack -> dict_attack * nfc: change scenes names * nfc: remove scene tick events * nfc: rework dict calls with buffer streams * nfc: fix notifications * nfc: fix mf desfire navigation * nfc: remove notification from mf classic read success * nfc: fix read sectors calculation * nfc: add fallback for unknown card * nfc: show file name while emulating * nfc: fix build * nfc: fix memory leak * nfc: fix desfire read * nfc: add no dict found navigation * nfc: add read views * nfc: update card fix * nfc: fix access bytes save * nfc: add exit and retry confirm to mf ultralight read success * nfc: introduce detect reader * nfc: change record open arg to macros * nfc: fix start from archive Co-authored-by: Astra <astra@astrra.space> Co-authored-by: あく <alleteam@gmail.com>
142 lines
4 KiB
C
Executable file
142 lines
4 KiB
C
Executable file
#include "nfca.h"
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <furi.h>
|
|
|
|
#define NFCA_CMD_RATS (0xE0U)
|
|
|
|
#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_sleep_req[] = {0x50, 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 sleep = false;
|
|
uint8_t rx_bytes = buff_rx_len / 8;
|
|
|
|
if(rx_bytes == sizeof(nfca_sleep_req) && !memcmp(buff_rx, nfca_sleep_req, rx_bytes)) {
|
|
sleep = 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 sleep;
|
|
}
|
|
|
|
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))));
|
|
}
|
|
}
|
|
}
|