NFC refactoring (#3050)
"A long time ago in a galaxy far, far away...." we started NFC subsystem refactoring.
Starring:
- @gornekich - NFC refactoring project lead, architect, senior developer
- @gsurkov - architect, senior developer
- @RebornedBrain - senior developer
Supporting roles:
- @skotopes, @DrZlo13, @hedger - general architecture advisors, code review
- @Astrrra, @doomwastaken, @Hellitron, @ImagineVagon333 - quality assurance
Special thanks:
@bettse, @pcunning, @nxv, @noproto, @AloneLiberty and everyone else who has been helping us all this time and contributing valuable knowledges, ideas and source code.
2023-10-24 07:08:09 +04:00
|
|
|
#include "iso14443_4_layer.h"
|
|
|
|
|
|
|
|
|
|
#include <furi.h>
|
|
|
|
|
|
2024-10-05 10:09:19 +03:00
|
|
|
// EMV Specific masks
|
2024-07-15 20:02:45 +03:00
|
|
|
#define ISO14443_4_BLOCK_PCB_I_ (0U << 6)
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_R_ (2U << 6)
|
2024-01-20 05:35:37 +09:00
|
|
|
#define ISO14443_4_BLOCK_PCB_TYPE_MASK (3U << 6)
|
2024-10-05 10:09:19 +03:00
|
|
|
#define ISO14443_4_BLOCK_PCB_S_WTX (3U << 4)
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_S (3U << 6)
|
|
|
|
|
//
|
|
|
|
|
|
2024-09-28 21:48:48 +03:00
|
|
|
#define ISO14443_4_BLOCK_PCB (1U << 1)
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_MASK (0x03)
|
|
|
|
|
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_I (0U)
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_I_NAD_OFFSET (2)
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_I_CID_OFFSET (3)
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_I_CHAIN_OFFSET (4)
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_I_NAD_MASK (1U << ISO14443_4_BLOCK_PCB_I_NAD_OFFSET)
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_I_CID_MASK (1U << ISO14443_4_BLOCK_PCB_I_CID_OFFSET)
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_I_CHAIN_MASK (1U << ISO14443_4_BLOCK_PCB_I_CHAIN_OFFSET)
|
|
|
|
|
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_R_MASK (5U << 5)
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_R_NACK_OFFSET (4)
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_R_CID_OFFSET (3)
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_R_CID_MASK (1U << ISO14443_4_BLOCK_PCB_R_CID_OFFSET)
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_R_NACK_MASK (1U << ISO14443_4_BLOCK_PCB_R_NACK_OFFSET)
|
|
|
|
|
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_S_MASK (3U << 6)
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_S_CID_OFFSET (3)
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_S_WTX_DESELECT_OFFSET (4)
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_S_CID_MASK (1U << ISO14443_4_BLOCK_PCB_R_CID_OFFSET)
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_S_WTX_DESELECT_MASK (3U << ISO14443_4_BLOCK_PCB_S_WTX_DESELECT_OFFSET)
|
2024-01-23 20:47:17 +00:00
|
|
|
|
2024-09-28 21:48:48 +03:00
|
|
|
#define ISO14443_4_BLOCK_PCB_BITS_ACTIVE(pcb, mask) (((pcb) & mask) == mask)
|
2024-01-29 23:12:17 +09:00
|
|
|
|
2024-09-28 21:48:48 +03:00
|
|
|
#define ISO14443_4_BLOCK_PCB_IS_R_BLOCK(pcb) \
|
|
|
|
|
ISO14443_4_BLOCK_PCB_BITS_ACTIVE(pcb, ISO14443_4_BLOCK_PCB_R_MASK)
|
|
|
|
|
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_IS_S_BLOCK(pcb) \
|
|
|
|
|
ISO14443_4_BLOCK_PCB_BITS_ACTIVE(pcb, ISO14443_4_BLOCK_PCB_S_MASK)
|
|
|
|
|
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_IS_CHAIN_ACTIVE(pcb) \
|
|
|
|
|
ISO14443_4_BLOCK_PCB_BITS_ACTIVE(pcb, ISO14443_4_BLOCK_PCB_I_CHAIN_MASK)
|
|
|
|
|
|
|
|
|
|
#define ISO14443_4_BLOCK_PCB_R_NACK_ACTIVE(pcb) \
|
|
|
|
|
ISO14443_4_BLOCK_PCB_BITS_ACTIVE(pcb, ISO14443_4_BLOCK_PCB_R_NACK_MASK)
|
2024-01-20 05:35:37 +09:00
|
|
|
|
NFC refactoring (#3050)
"A long time ago in a galaxy far, far away...." we started NFC subsystem refactoring.
Starring:
- @gornekich - NFC refactoring project lead, architect, senior developer
- @gsurkov - architect, senior developer
- @RebornedBrain - senior developer
Supporting roles:
- @skotopes, @DrZlo13, @hedger - general architecture advisors, code review
- @Astrrra, @doomwastaken, @Hellitron, @ImagineVagon333 - quality assurance
Special thanks:
@bettse, @pcunning, @nxv, @noproto, @AloneLiberty and everyone else who has been helping us all this time and contributing valuable knowledges, ideas and source code.
2023-10-24 07:08:09 +04:00
|
|
|
struct Iso14443_4Layer {
|
|
|
|
|
uint8_t pcb;
|
|
|
|
|
uint8_t pcb_prev;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static inline void iso14443_4_layer_update_pcb(Iso14443_4Layer* instance) {
|
|
|
|
|
instance->pcb_prev = instance->pcb;
|
|
|
|
|
instance->pcb ^= (uint8_t)0x01;
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-25 13:53:32 +03:00
|
|
|
Iso14443_4Layer* iso14443_4_layer_alloc(void) {
|
NFC refactoring (#3050)
"A long time ago in a galaxy far, far away...." we started NFC subsystem refactoring.
Starring:
- @gornekich - NFC refactoring project lead, architect, senior developer
- @gsurkov - architect, senior developer
- @RebornedBrain - senior developer
Supporting roles:
- @skotopes, @DrZlo13, @hedger - general architecture advisors, code review
- @Astrrra, @doomwastaken, @Hellitron, @ImagineVagon333 - quality assurance
Special thanks:
@bettse, @pcunning, @nxv, @noproto, @AloneLiberty and everyone else who has been helping us all this time and contributing valuable knowledges, ideas and source code.
2023-10-24 07:08:09 +04:00
|
|
|
Iso14443_4Layer* instance = malloc(sizeof(Iso14443_4Layer));
|
|
|
|
|
|
|
|
|
|
iso14443_4_layer_reset(instance);
|
|
|
|
|
return instance;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void iso14443_4_layer_free(Iso14443_4Layer* instance) {
|
|
|
|
|
furi_assert(instance);
|
|
|
|
|
free(instance);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void iso14443_4_layer_reset(Iso14443_4Layer* instance) {
|
|
|
|
|
furi_assert(instance);
|
2024-09-28 21:48:48 +03:00
|
|
|
instance->pcb_prev = 0;
|
NFC refactoring (#3050)
"A long time ago in a galaxy far, far away...." we started NFC subsystem refactoring.
Starring:
- @gornekich - NFC refactoring project lead, architect, senior developer
- @gsurkov - architect, senior developer
- @RebornedBrain - senior developer
Supporting roles:
- @skotopes, @DrZlo13, @hedger - general architecture advisors, code review
- @Astrrra, @doomwastaken, @Hellitron, @ImagineVagon333 - quality assurance
Special thanks:
@bettse, @pcunning, @nxv, @noproto, @AloneLiberty and everyone else who has been helping us all this time and contributing valuable knowledges, ideas and source code.
2023-10-24 07:08:09 +04:00
|
|
|
instance->pcb = ISO14443_4_BLOCK_PCB_I | ISO14443_4_BLOCK_PCB;
|
|
|
|
|
}
|
|
|
|
|
|
2024-09-28 21:48:48 +03:00
|
|
|
void iso14443_4_layer_set_i_block(Iso14443_4Layer* instance, bool chaining, bool CID_present) {
|
|
|
|
|
uint8_t block_pcb = instance->pcb & ISO14443_4_BLOCK_PCB_MASK;
|
|
|
|
|
instance->pcb = ISO14443_4_BLOCK_PCB_I | (chaining << ISO14443_4_BLOCK_PCB_I_CHAIN_OFFSET) |
|
|
|
|
|
(CID_present << ISO14443_4_BLOCK_PCB_I_CID_OFFSET) | block_pcb;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void iso14443_4_layer_set_r_block(Iso14443_4Layer* instance, bool acknowledged, bool CID_present) {
|
|
|
|
|
furi_assert(instance);
|
|
|
|
|
uint8_t block_pcb = instance->pcb & ISO14443_4_BLOCK_PCB_MASK;
|
|
|
|
|
instance->pcb = ISO14443_4_BLOCK_PCB_R_MASK |
|
|
|
|
|
(!acknowledged << ISO14443_4_BLOCK_PCB_R_NACK_OFFSET) |
|
|
|
|
|
(CID_present << ISO14443_4_BLOCK_PCB_R_CID_OFFSET) | block_pcb;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void iso14443_4_layer_set_s_block(Iso14443_4Layer* instance, bool deselect, bool CID_present) {
|
|
|
|
|
furi_assert(instance);
|
|
|
|
|
uint8_t des_wtx = !deselect ? (ISO14443_4_BLOCK_PCB_S_WTX_DESELECT_MASK) : 0;
|
|
|
|
|
instance->pcb = ISO14443_4_BLOCK_PCB_S_MASK | des_wtx |
|
|
|
|
|
(CID_present << ISO14443_4_BLOCK_PCB_S_CID_OFFSET) | ISO14443_4_BLOCK_PCB;
|
|
|
|
|
}
|
|
|
|
|
|
NFC refactoring (#3050)
"A long time ago in a galaxy far, far away...." we started NFC subsystem refactoring.
Starring:
- @gornekich - NFC refactoring project lead, architect, senior developer
- @gsurkov - architect, senior developer
- @RebornedBrain - senior developer
Supporting roles:
- @skotopes, @DrZlo13, @hedger - general architecture advisors, code review
- @Astrrra, @doomwastaken, @Hellitron, @ImagineVagon333 - quality assurance
Special thanks:
@bettse, @pcunning, @nxv, @noproto, @AloneLiberty and everyone else who has been helping us all this time and contributing valuable knowledges, ideas and source code.
2023-10-24 07:08:09 +04:00
|
|
|
void iso14443_4_layer_encode_block(
|
|
|
|
|
Iso14443_4Layer* instance,
|
|
|
|
|
const BitBuffer* input_data,
|
|
|
|
|
BitBuffer* block_data) {
|
|
|
|
|
furi_assert(instance);
|
|
|
|
|
|
|
|
|
|
bit_buffer_append_byte(block_data, instance->pcb);
|
|
|
|
|
bit_buffer_append(block_data, input_data);
|
|
|
|
|
|
|
|
|
|
iso14443_4_layer_update_pcb(instance);
|
|
|
|
|
}
|
|
|
|
|
|
2024-09-28 21:48:48 +03:00
|
|
|
static inline uint8_t iso14443_4_layer_get_response_pcb(const BitBuffer* block_data) {
|
|
|
|
|
const uint8_t* data = bit_buffer_get_data(block_data);
|
|
|
|
|
return data[0];
|
|
|
|
|
}
|
|
|
|
|
|
NFC refactoring (#3050)
"A long time ago in a galaxy far, far away...." we started NFC subsystem refactoring.
Starring:
- @gornekich - NFC refactoring project lead, architect, senior developer
- @gsurkov - architect, senior developer
- @RebornedBrain - senior developer
Supporting roles:
- @skotopes, @DrZlo13, @hedger - general architecture advisors, code review
- @Astrrra, @doomwastaken, @Hellitron, @ImagineVagon333 - quality assurance
Special thanks:
@bettse, @pcunning, @nxv, @noproto, @AloneLiberty and everyone else who has been helping us all this time and contributing valuable knowledges, ideas and source code.
2023-10-24 07:08:09 +04:00
|
|
|
bool iso14443_4_layer_decode_block(
|
|
|
|
|
Iso14443_4Layer* instance,
|
|
|
|
|
BitBuffer* output_data,
|
|
|
|
|
const BitBuffer* block_data) {
|
|
|
|
|
furi_assert(instance);
|
|
|
|
|
|
|
|
|
|
bool ret = false;
|
|
|
|
|
|
|
|
|
|
do {
|
2024-09-28 21:48:48 +03:00
|
|
|
if(ISO14443_4_BLOCK_PCB_IS_R_BLOCK(instance->pcb_prev)) {
|
|
|
|
|
const uint8_t response_pcb = iso14443_4_layer_get_response_pcb(block_data);
|
|
|
|
|
ret = (ISO14443_4_BLOCK_PCB_IS_R_BLOCK(response_pcb)) &&
|
|
|
|
|
(!ISO14443_4_BLOCK_PCB_R_NACK_ACTIVE(response_pcb));
|
|
|
|
|
instance->pcb &= ISO14443_4_BLOCK_PCB_MASK;
|
|
|
|
|
iso14443_4_layer_update_pcb(instance);
|
|
|
|
|
} else if(ISO14443_4_BLOCK_PCB_IS_CHAIN_ACTIVE(instance->pcb_prev)) {
|
|
|
|
|
const uint8_t response_pcb = iso14443_4_layer_get_response_pcb(block_data);
|
|
|
|
|
ret = (ISO14443_4_BLOCK_PCB_IS_R_BLOCK(response_pcb)) &&
|
|
|
|
|
(!ISO14443_4_BLOCK_PCB_R_NACK_ACTIVE(response_pcb));
|
|
|
|
|
instance->pcb &= ~(ISO14443_4_BLOCK_PCB_I_CHAIN_MASK);
|
|
|
|
|
} else if(ISO14443_4_BLOCK_PCB_IS_S_BLOCK(instance->pcb_prev)) {
|
|
|
|
|
ret = bit_buffer_starts_with_byte(block_data, instance->pcb_prev);
|
|
|
|
|
if(bit_buffer_get_size_bytes(block_data) > 1)
|
|
|
|
|
bit_buffer_copy_right(output_data, block_data, 1);
|
|
|
|
|
} else {
|
|
|
|
|
if(!bit_buffer_starts_with_byte(block_data, instance->pcb_prev)) break;
|
|
|
|
|
bit_buffer_copy_right(output_data, block_data, 1);
|
|
|
|
|
ret = true;
|
|
|
|
|
}
|
NFC refactoring (#3050)
"A long time ago in a galaxy far, far away...." we started NFC subsystem refactoring.
Starring:
- @gornekich - NFC refactoring project lead, architect, senior developer
- @gsurkov - architect, senior developer
- @RebornedBrain - senior developer
Supporting roles:
- @skotopes, @DrZlo13, @hedger - general architecture advisors, code review
- @Astrrra, @doomwastaken, @Hellitron, @ImagineVagon333 - quality assurance
Special thanks:
@bettse, @pcunning, @nxv, @noproto, @AloneLiberty and everyone else who has been helping us all this time and contributing valuable knowledges, ideas and source code.
2023-10-24 07:08:09 +04:00
|
|
|
} while(false);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
2024-01-20 05:35:37 +09:00
|
|
|
|
|
|
|
|
Iso14443_4aError iso14443_4_layer_decode_block_pwt_ext(
|
|
|
|
|
Iso14443_4Layer* instance,
|
|
|
|
|
BitBuffer* output_data,
|
|
|
|
|
const BitBuffer* block_data) {
|
|
|
|
|
furi_assert(instance);
|
|
|
|
|
|
|
|
|
|
Iso14443_4aError ret = Iso14443_4aErrorProtocol;
|
2024-01-28 20:18:17 +00:00
|
|
|
bit_buffer_reset(output_data);
|
2024-01-20 05:35:37 +09:00
|
|
|
|
|
|
|
|
do {
|
|
|
|
|
const uint8_t pcb_field = bit_buffer_get_byte(block_data, 0);
|
|
|
|
|
const uint8_t block_type = pcb_field & ISO14443_4_BLOCK_PCB_TYPE_MASK;
|
|
|
|
|
switch(block_type) {
|
2024-01-29 23:12:17 +09:00
|
|
|
case ISO14443_4_BLOCK_PCB_I_:
|
2024-01-20 05:35:37 +09:00
|
|
|
if(pcb_field == instance->pcb_prev) {
|
|
|
|
|
bit_buffer_copy_right(output_data, block_data, 1);
|
|
|
|
|
ret = Iso14443_4aErrorNone;
|
|
|
|
|
} else {
|
2024-01-28 20:18:17 +00:00
|
|
|
// send original request again
|
|
|
|
|
ret = Iso14443_4aErrorSendExtra;
|
2024-01-20 05:35:37 +09:00
|
|
|
}
|
|
|
|
|
break;
|
2024-01-29 23:12:17 +09:00
|
|
|
case ISO14443_4_BLOCK_PCB_R_:
|
2024-01-20 05:35:37 +09:00
|
|
|
// TODO
|
|
|
|
|
break;
|
|
|
|
|
case ISO14443_4_BLOCK_PCB_S:
|
|
|
|
|
if((pcb_field & ISO14443_4_BLOCK_PCB_S_WTX) == ISO14443_4_BLOCK_PCB_S_WTX) {
|
|
|
|
|
const uint8_t inf_field = bit_buffer_get_byte(block_data, 1);
|
|
|
|
|
//const uint8_t power_level = inf_field >> 6;
|
|
|
|
|
const uint8_t wtxm = inf_field & 0b111111;
|
|
|
|
|
//uint32_t fwt_temp = MIN((fwt * wtxm), fwt_max);
|
|
|
|
|
|
|
|
|
|
bit_buffer_append_byte(
|
|
|
|
|
output_data,
|
|
|
|
|
ISO14443_4_BLOCK_PCB_S | ISO14443_4_BLOCK_PCB_S_WTX | ISO14443_4_BLOCK_PCB);
|
|
|
|
|
bit_buffer_append_byte(output_data, wtxm);
|
2024-01-28 20:18:17 +00:00
|
|
|
ret = Iso14443_4aErrorSendExtra;
|
2024-01-20 05:35:37 +09:00
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
} while(false);
|
|
|
|
|
|
2024-01-23 20:47:17 +00:00
|
|
|
if(ret != Iso14443_4aErrorNone) {
|
|
|
|
|
FURI_LOG_RAW_T("RAW RX:");
|
|
|
|
|
for(size_t x = 0; x < bit_buffer_get_size_bytes(block_data); x++) {
|
|
|
|
|
FURI_LOG_RAW_T("%02X ", bit_buffer_get_byte(block_data, x));
|
|
|
|
|
}
|
|
|
|
|
FURI_LOG_RAW_T("\r\n");
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-20 05:35:37 +09:00
|
|
|
return ret;
|
2024-01-23 20:47:17 +00:00
|
|
|
}
|