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 "felica_poller_i.h"
|
|
|
|
|
|
|
|
|
|
#include <nfc/helpers/felica_crc.h>
|
|
|
|
|
|
|
|
|
|
#define TAG "FelicaPoller"
|
|
|
|
|
|
|
|
|
|
static FelicaError felica_poller_process_error(NfcError error) {
|
|
|
|
|
switch(error) {
|
|
|
|
|
case NfcErrorNone:
|
|
|
|
|
return FelicaErrorNone;
|
|
|
|
|
case NfcErrorTimeout:
|
|
|
|
|
return FelicaErrorTimeout;
|
|
|
|
|
default:
|
|
|
|
|
return FelicaErrorNotPresent;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-10 12:51:36 +03:00
|
|
|
FelicaError felica_poller_frame_exchange(
|
|
|
|
|
const FelicaPoller* instance,
|
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
|
|
|
const BitBuffer* tx_buffer,
|
|
|
|
|
BitBuffer* rx_buffer,
|
|
|
|
|
uint32_t fwt) {
|
|
|
|
|
furi_assert(instance);
|
|
|
|
|
|
|
|
|
|
const size_t tx_bytes = bit_buffer_get_size_bytes(tx_buffer);
|
|
|
|
|
furi_assert(tx_bytes <= bit_buffer_get_capacity_bytes(instance->tx_buffer) - FELICA_CRC_SIZE);
|
|
|
|
|
|
|
|
|
|
felica_crc_append(instance->tx_buffer);
|
|
|
|
|
|
|
|
|
|
FelicaError ret = FelicaErrorNone;
|
|
|
|
|
|
|
|
|
|
do {
|
|
|
|
|
NfcError error =
|
|
|
|
|
nfc_poller_trx(instance->nfc, instance->tx_buffer, instance->rx_buffer, fwt);
|
|
|
|
|
if(error != NfcErrorNone) {
|
|
|
|
|
ret = felica_poller_process_error(error);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bit_buffer_copy(rx_buffer, instance->rx_buffer);
|
|
|
|
|
if(!felica_crc_check(instance->rx_buffer)) {
|
|
|
|
|
ret = FelicaErrorWrongCrc;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
felica_crc_trim(rx_buffer);
|
|
|
|
|
} while(false);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
2023-11-15 12:32:45 +04:00
|
|
|
FelicaError felica_poller_polling(
|
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
|
|
|
FelicaPoller* instance,
|
|
|
|
|
const FelicaPollerPollingCommand* cmd,
|
|
|
|
|
FelicaPollerPollingResponse* resp) {
|
|
|
|
|
furi_assert(instance);
|
|
|
|
|
furi_assert(cmd);
|
|
|
|
|
furi_assert(resp);
|
|
|
|
|
|
|
|
|
|
FelicaError error = FelicaErrorNone;
|
|
|
|
|
|
|
|
|
|
do {
|
|
|
|
|
bit_buffer_set_size_bytes(instance->tx_buffer, 2);
|
|
|
|
|
// Set frame len
|
|
|
|
|
bit_buffer_set_byte(
|
|
|
|
|
instance->tx_buffer, 0, sizeof(FelicaPollerPollingCommand) + FELICA_CRC_SIZE);
|
|
|
|
|
// Set command code
|
|
|
|
|
bit_buffer_set_byte(instance->tx_buffer, 1, FELICA_POLLER_CMD_POLLING_REQ_CODE);
|
|
|
|
|
// Set other data
|
|
|
|
|
bit_buffer_append_bytes(
|
|
|
|
|
instance->tx_buffer, (uint8_t*)cmd, sizeof(FelicaPollerPollingCommand));
|
|
|
|
|
|
|
|
|
|
error = felica_poller_frame_exchange(
|
|
|
|
|
instance, instance->tx_buffer, instance->rx_buffer, FELICA_POLLER_POLLING_FWT);
|
|
|
|
|
|
|
|
|
|
if(error != FelicaErrorNone) break;
|
|
|
|
|
if(bit_buffer_get_byte(instance->rx_buffer, 1) != FELICA_POLLER_CMD_POLLING_RESP_CODE) {
|
|
|
|
|
error = FelicaErrorProtocol;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if(bit_buffer_get_size_bytes(instance->rx_buffer) <
|
|
|
|
|
sizeof(FelicaIDm) + sizeof(FelicaPMm) + 1) {
|
|
|
|
|
error = FelicaErrorProtocol;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bit_buffer_write_bytes_mid(instance->rx_buffer, resp->idm.data, 2, sizeof(FelicaIDm));
|
|
|
|
|
bit_buffer_write_bytes_mid(
|
|
|
|
|
instance->rx_buffer, resp->pmm.data, sizeof(FelicaIDm) + 2, sizeof(FelicaPMm));
|
|
|
|
|
|
|
|
|
|
} while(false);
|
|
|
|
|
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
|
2025-10-01 18:54:08 +04:00
|
|
|
// This is in fact a buffer preparer for a specified service. It should be have the _ex suffix. The prepare_tx_buffer_raw should have this name.
|
2024-04-10 12:51:36 +03:00
|
|
|
static void felica_poller_prepare_tx_buffer(
|
|
|
|
|
const FelicaPoller* instance,
|
|
|
|
|
const uint8_t command,
|
|
|
|
|
const uint16_t service_code,
|
|
|
|
|
const uint8_t block_count,
|
|
|
|
|
const uint8_t* const blocks,
|
|
|
|
|
const uint8_t data_block_count,
|
|
|
|
|
const uint8_t* data) {
|
|
|
|
|
FelicaCommandHeader cmd = {
|
|
|
|
|
.code = command,
|
|
|
|
|
.idm = instance->data->idm,
|
|
|
|
|
.service_num = 1,
|
|
|
|
|
.service_code = service_code,
|
|
|
|
|
.block_count = block_count,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
FelicaBlockListElement block_list[4] = {{0}, {0}, {0}, {0}};
|
|
|
|
|
for(uint8_t i = 0; i < block_count; i++) {
|
|
|
|
|
block_list[i].length = 1;
|
|
|
|
|
block_list[i].block_number = blocks[i];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t block_list_count = block_count;
|
|
|
|
|
uint8_t block_list_size = block_list_count * sizeof(FelicaBlockListElement);
|
|
|
|
|
uint8_t total_size = sizeof(FelicaCommandHeader) + 1 + block_list_size +
|
|
|
|
|
data_block_count * FELICA_DATA_BLOCK_SIZE;
|
|
|
|
|
bit_buffer_reset(instance->tx_buffer);
|
|
|
|
|
bit_buffer_append_byte(instance->tx_buffer, total_size);
|
|
|
|
|
bit_buffer_append_bytes(instance->tx_buffer, (uint8_t*)&cmd, sizeof(FelicaCommandHeader));
|
|
|
|
|
bit_buffer_append_bytes(instance->tx_buffer, (uint8_t*)&block_list, block_list_size);
|
|
|
|
|
|
|
|
|
|
if(data_block_count != 0) {
|
|
|
|
|
bit_buffer_append_bytes(
|
|
|
|
|
instance->tx_buffer, data, data_block_count * FELICA_DATA_BLOCK_SIZE);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
FelicaError felica_poller_read_blocks(
|
|
|
|
|
FelicaPoller* instance,
|
|
|
|
|
const uint8_t block_count,
|
|
|
|
|
const uint8_t* const block_numbers,
|
2025-02-24 16:07:45 -05:00
|
|
|
uint16_t service_code,
|
2024-04-10 12:51:36 +03:00
|
|
|
FelicaPollerReadCommandResponse** const response_ptr) {
|
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
|
|
|
furi_assert(instance);
|
2024-04-10 12:51:36 +03:00
|
|
|
furi_assert(block_count <= 4);
|
|
|
|
|
furi_assert(block_numbers);
|
|
|
|
|
furi_assert(response_ptr);
|
|
|
|
|
|
|
|
|
|
felica_poller_prepare_tx_buffer(
|
|
|
|
|
instance,
|
|
|
|
|
FELICA_CMD_READ_WITHOUT_ENCRYPTION,
|
2025-02-24 16:07:45 -05:00
|
|
|
service_code,
|
2024-04-10 12:51:36 +03:00
|
|
|
block_count,
|
|
|
|
|
block_numbers,
|
|
|
|
|
0,
|
|
|
|
|
NULL);
|
|
|
|
|
bit_buffer_reset(instance->rx_buffer);
|
|
|
|
|
|
|
|
|
|
FelicaError error = felica_poller_frame_exchange(
|
|
|
|
|
instance, instance->tx_buffer, instance->rx_buffer, FELICA_POLLER_POLLING_FWT);
|
|
|
|
|
if(error == FelicaErrorNone) {
|
|
|
|
|
*response_ptr = (FelicaPollerReadCommandResponse*)bit_buffer_get_data(instance->rx_buffer);
|
|
|
|
|
}
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
FelicaError felica_poller_write_blocks(
|
|
|
|
|
const FelicaPoller* instance,
|
|
|
|
|
const uint8_t block_count,
|
|
|
|
|
const uint8_t* const block_numbers,
|
|
|
|
|
const uint8_t* data,
|
|
|
|
|
FelicaPollerWriteCommandResponse** const response_ptr) {
|
|
|
|
|
furi_assert(instance);
|
|
|
|
|
furi_assert(block_count <= 2);
|
|
|
|
|
furi_assert(block_numbers);
|
|
|
|
|
furi_assert(data);
|
|
|
|
|
furi_assert(response_ptr);
|
|
|
|
|
|
|
|
|
|
felica_poller_prepare_tx_buffer(
|
|
|
|
|
instance,
|
|
|
|
|
FELICA_CMD_WRITE_WITHOUT_ENCRYPTION,
|
|
|
|
|
FELICA_SERVICE_RW_ACCESS,
|
|
|
|
|
block_count,
|
|
|
|
|
block_numbers,
|
|
|
|
|
block_count,
|
|
|
|
|
data);
|
|
|
|
|
bit_buffer_reset(instance->rx_buffer);
|
|
|
|
|
|
|
|
|
|
FelicaError error = felica_poller_frame_exchange(
|
|
|
|
|
instance, instance->tx_buffer, instance->rx_buffer, FELICA_POLLER_POLLING_FWT);
|
|
|
|
|
if(error == FelicaErrorNone) {
|
|
|
|
|
*response_ptr =
|
|
|
|
|
(FelicaPollerWriteCommandResponse*)bit_buffer_get_data(instance->rx_buffer);
|
|
|
|
|
}
|
|
|
|
|
return error;
|
|
|
|
|
}
|
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
|
|
|
|
2024-04-10 12:51:36 +03:00
|
|
|
FelicaError felica_poller_activate(FelicaPoller* instance, FelicaData* data) {
|
|
|
|
|
furi_assert(instance);
|
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
|
|
|
|
|
|
|
|
FelicaError ret;
|
|
|
|
|
|
|
|
|
|
do {
|
|
|
|
|
bit_buffer_reset(instance->tx_buffer);
|
|
|
|
|
bit_buffer_reset(instance->rx_buffer);
|
|
|
|
|
|
|
|
|
|
// Send Polling command
|
|
|
|
|
const FelicaPollerPollingCommand polling_cmd = {
|
|
|
|
|
.system_code = FELICA_SYSTEM_CODE_CODE,
|
|
|
|
|
.request_code = 0,
|
|
|
|
|
.time_slot = FELICA_TIME_SLOT_1,
|
|
|
|
|
};
|
|
|
|
|
FelicaPollerPollingResponse polling_resp = {};
|
|
|
|
|
|
2023-11-15 12:32:45 +04:00
|
|
|
ret = felica_poller_polling(instance, &polling_cmd, &polling_resp);
|
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
|
|
|
|
|
|
|
|
if(ret != FelicaErrorNone) {
|
|
|
|
|
FURI_LOG_T(TAG, "Activation failed error: %d", ret);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
data->idm = polling_resp.idm;
|
|
|
|
|
data->pmm = polling_resp.pmm;
|
|
|
|
|
instance->state = FelicaPollerStateActivated;
|
|
|
|
|
} while(false);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
2025-10-01 18:54:08 +04:00
|
|
|
|
|
|
|
|
static void felica_poller_prepare_tx_buffer_raw(
|
|
|
|
|
const FelicaPoller* instance,
|
|
|
|
|
const uint8_t command,
|
|
|
|
|
const uint8_t* data,
|
|
|
|
|
const uint8_t data_length) {
|
|
|
|
|
FelicaCommandHeaderRaw cmd = {.length = 0x00, .command = command, .idm = instance->data->idm};
|
|
|
|
|
|
|
|
|
|
cmd.length = sizeof(FelicaCommandHeaderRaw) + data_length;
|
|
|
|
|
bit_buffer_reset(instance->tx_buffer);
|
|
|
|
|
bit_buffer_append_bytes(instance->tx_buffer, (uint8_t*)&cmd, sizeof(FelicaCommandHeaderRaw));
|
2025-11-06 22:30:02 +04:00
|
|
|
if(data_length > 0) {
|
|
|
|
|
bit_buffer_append_bytes(instance->tx_buffer, data, data_length);
|
|
|
|
|
}
|
2025-10-01 18:54:08 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
FelicaError felica_poller_list_service_by_cursor(
|
|
|
|
|
FelicaPoller* instance,
|
|
|
|
|
uint16_t cursor,
|
|
|
|
|
FelicaListServiceCommandResponse** const response_ptr) {
|
|
|
|
|
furi_assert(instance);
|
|
|
|
|
furi_assert(response_ptr);
|
|
|
|
|
|
|
|
|
|
const uint8_t data[2] = {(uint8_t)(cursor & 0xFF), (uint8_t)((cursor >> 8) & 0xFF)};
|
|
|
|
|
|
|
|
|
|
felica_poller_prepare_tx_buffer_raw(
|
|
|
|
|
instance, FELICA_CMD_LIST_SERVICE_CODE, data, sizeof(data));
|
|
|
|
|
|
|
|
|
|
bit_buffer_reset(instance->rx_buffer);
|
|
|
|
|
|
|
|
|
|
FelicaError error = felica_poller_frame_exchange(
|
|
|
|
|
instance, instance->tx_buffer, instance->rx_buffer, FELICA_POLLER_POLLING_FWT);
|
|
|
|
|
if(error != FelicaErrorNone) {
|
|
|
|
|
FURI_LOG_E(TAG, "List service by cursor failed with error: %d", error);
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
size_t rx_len = bit_buffer_get_size_bytes(instance->rx_buffer);
|
|
|
|
|
if(rx_len < sizeof(FelicaCommandHeaderRaw) + 2) return FelicaErrorProtocol;
|
|
|
|
|
|
|
|
|
|
// error is known to be FelicaErrorNone here
|
|
|
|
|
*response_ptr = (FelicaListServiceCommandResponse*)bit_buffer_get_data(instance->rx_buffer);
|
|
|
|
|
return error;
|
|
|
|
|
}
|
2025-11-06 22:30:02 +04:00
|
|
|
|
|
|
|
|
FelicaError felica_poller_list_system_code(
|
|
|
|
|
FelicaPoller* instance,
|
|
|
|
|
FelicaListSystemCodeCommandResponse** const response_ptr) {
|
|
|
|
|
furi_assert(instance);
|
|
|
|
|
furi_assert(response_ptr);
|
|
|
|
|
|
|
|
|
|
uint8_t data[] = {0};
|
|
|
|
|
|
|
|
|
|
felica_poller_prepare_tx_buffer_raw(instance, FELICA_CMD_REQUEST_SYSTEM_CODE, data, 0);
|
|
|
|
|
|
|
|
|
|
bit_buffer_reset(instance->rx_buffer);
|
|
|
|
|
|
|
|
|
|
FelicaError error = felica_poller_frame_exchange(
|
|
|
|
|
instance, instance->tx_buffer, instance->rx_buffer, FELICA_POLLER_POLLING_FWT);
|
|
|
|
|
if(error != FelicaErrorNone) {
|
|
|
|
|
FURI_LOG_E(TAG, "Request system code failed with error: %d", error);
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
size_t rx_len = bit_buffer_get_size_bytes(instance->rx_buffer);
|
|
|
|
|
if(rx_len < sizeof(FelicaCommandHeaderRaw) + 3) return FelicaErrorProtocol;
|
|
|
|
|
// at least 1 system code + the count being 0x01
|
|
|
|
|
|
|
|
|
|
// error is known to be FelicaErrorNone here
|
|
|
|
|
*response_ptr = (FelicaListSystemCodeCommandResponse*)bit_buffer_get_data(instance->rx_buffer);
|
|
|
|
|
return error;
|
|
|
|
|
}
|