Skip to content

Commit 1c3cbec

Browse files
[FL-3640] NFC: Felica UID emulation (#3190)
* Added basic template of Felica listener * Raw nfc felica listener functions * Added functions to setup chip for felica listener * Cleanup function templates from unnecessary parts * Removed todo comment * Updated api versions * Adjusted chip config for felica * Set proper chip passive target mode for felica * Added felica function to unit tests * Update furi_hal_nfc_felica.c * Removed duplication Co-authored-by: gornekich <[email protected]>
1 parent 4b3e8ab commit 1c3cbec

File tree

13 files changed

+346
-4
lines changed

13 files changed

+346
-4
lines changed

applications/debug/unit_tests/nfc/nfc_transport.c

+15
Original file line numberDiff line numberDiff line change
@@ -455,4 +455,19 @@ NfcError nfc_iso15693_listener_tx_sof(Nfc* instance) {
455455
return NfcErrorNone;
456456
}
457457

458+
NfcError nfc_felica_listener_set_sensf_res_data(
459+
Nfc* instance,
460+
const uint8_t* idm,
461+
const uint8_t idm_len,
462+
const uint8_t* pmm,
463+
const uint8_t pmm_len) {
464+
furi_assert(instance);
465+
furi_assert(idm);
466+
furi_assert(pmm);
467+
furi_assert(idm_len == 8);
468+
furi_assert(pmm_len == 8);
469+
470+
return NfcErrorNone;
471+
}
472+
458473
#endif

applications/main/nfc/helpers/protocol_support/felica/felica.c

+8-2
Original file line numberDiff line numberDiff line change
@@ -67,8 +67,14 @@ static bool nfc_scene_saved_menu_on_event_felica(NfcApp* instance, uint32_t even
6767
return false;
6868
}
6969

70+
static void nfc_scene_emulate_on_enter_felica(NfcApp* instance) {
71+
const FelicaData* data = nfc_device_get_data(instance->nfc_device, NfcProtocolFelica);
72+
instance->listener = nfc_listener_alloc(instance->nfc, NfcProtocolFelica, data);
73+
nfc_listener_start(instance->listener, NULL, NULL);
74+
}
75+
7076
const NfcProtocolSupportBase nfc_protocol_support_felica = {
71-
.features = NfcProtocolFeatureNone,
77+
.features = NfcProtocolFeatureEmulateUid,
7278

7379
.scene_info =
7480
{
@@ -102,7 +108,7 @@ const NfcProtocolSupportBase nfc_protocol_support_felica = {
102108
},
103109
.scene_emulate =
104110
{
105-
.on_enter = nfc_protocol_support_common_on_enter_empty,
111+
.on_enter = nfc_scene_emulate_on_enter_felica,
106112
.on_event = nfc_protocol_support_common_on_event_empty,
107113
},
108114
};

lib/nfc/nfc.c

+17-1
Original file line numberDiff line numberDiff line change
@@ -76,7 +76,7 @@ static const FuriHalNfcTech nfc_tech_table[NfcModeNum][NfcTechNum] = {
7676
[NfcTechIso14443a] = FuriHalNfcTechIso14443a,
7777
[NfcTechIso14443b] = FuriHalNfcTechInvalid,
7878
[NfcTechIso15693] = FuriHalNfcTechIso15693,
79-
[NfcTechFelica] = FuriHalNfcTechInvalid,
79+
[NfcTechFelica] = FuriHalNfcTechFelica,
8080
},
8181
};
8282

@@ -646,4 +646,20 @@ NfcError nfc_iso15693_listener_tx_sof(Nfc* instance) {
646646
return ret;
647647
}
648648

649+
NfcError nfc_felica_listener_set_sensf_res_data(
650+
Nfc* instance,
651+
const uint8_t* idm,
652+
const uint8_t idm_len,
653+
const uint8_t* pmm,
654+
const uint8_t pmm_len) {
655+
furi_assert(instance);
656+
furi_assert(idm);
657+
furi_assert(pmm);
658+
659+
FuriHalNfcError error =
660+
furi_hal_nfc_felica_listener_set_sensf_res_data(idm, idm_len, pmm, pmm_len);
661+
instance->comm_state = NfcCommStateIdle;
662+
return nfc_process_hal_error(error);
663+
}
664+
649665
#endif // APP_UNIT_TESTS

lib/nfc/nfc.h

+19
Original file line numberDiff line numberDiff line change
@@ -351,6 +351,25 @@ NfcError nfc_iso14443a_listener_set_col_res_data(
351351
uint8_t* atqa,
352352
uint8_t sak);
353353

354+
/**
355+
* @brief Set FeliCa collision resolution parameters in listener mode.
356+
*
357+
* Configures the NFC hardware for automatic collision resolution.
358+
*
359+
* @param[in,out] instance pointer to the instance to be configured.
360+
* @param[in] idm pointer to a byte array containing the IDm.
361+
* @param[in] idm_len IDm length in bytes.
362+
* @param[in] pmm pointer to a byte array containing the PMm.
363+
* @param[in] pmm_len PMm length in bytes.
364+
* @returns NfcErrorNone on success, any other error code on failure.
365+
*/
366+
NfcError nfc_felica_listener_set_sensf_res_data(
367+
Nfc* instance,
368+
const uint8_t* idm,
369+
const uint8_t idm_len,
370+
const uint8_t* pmm,
371+
const uint8_t pmm_len);
372+
354373
/**
355374
* @brief Send ISO15693 Start of Frame pattern in listener mode
356375
*

lib/nfc/protocols/felica/felica.h

+2
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,8 @@ extern "C" {
1414
#define FELICA_FDT_POLL_FC (10000U)
1515
#define FELICA_POLL_POLL_MIN_US (1280U)
1616

17+
#define FELICA_FDT_LISTEN_FC (1172)
18+
1719
#define FELICA_SYSTEM_CODE_CODE (0xFFFFU)
1820
#define FELICA_TIME_SLOT_1 (0x00U)
1921
#define FELICA_TIME_SLOT_2 (0x01U)
+79
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,79 @@
1+
#include "felica_listener_i.h"
2+
3+
#include "nfc/protocols/nfc_listener_base.h"
4+
5+
#define FELICA_LISTENER_MAX_BUFFER_SIZE (64)
6+
#define TAG "Felica"
7+
8+
FelicaListener* felica_listener_alloc(Nfc* nfc, FelicaData* data) {
9+
furi_assert(nfc);
10+
furi_assert(data);
11+
12+
FelicaListener* instance = malloc(sizeof(FelicaListener));
13+
instance->nfc = nfc;
14+
instance->data = data;
15+
instance->tx_buffer = bit_buffer_alloc(FELICA_LISTENER_MAX_BUFFER_SIZE);
16+
instance->rx_buffer = bit_buffer_alloc(FELICA_LISTENER_MAX_BUFFER_SIZE);
17+
18+
nfc_set_fdt_listen_fc(instance->nfc, FELICA_FDT_LISTEN_FC);
19+
20+
nfc_config(instance->nfc, NfcModeListener, NfcTechFelica);
21+
nfc_felica_listener_set_sensf_res_data(
22+
nfc, data->idm.data, sizeof(data->idm), data->pmm.data, sizeof(data->pmm));
23+
24+
return instance;
25+
}
26+
27+
void felica_listener_free(FelicaListener* instance) {
28+
furi_assert(instance);
29+
furi_assert(instance->tx_buffer);
30+
31+
bit_buffer_free(instance->tx_buffer);
32+
bit_buffer_free(instance->rx_buffer);
33+
free(instance);
34+
}
35+
36+
void felica_listener_set_callback(
37+
FelicaListener* listener,
38+
NfcGenericCallback callback,
39+
void* context) {
40+
UNUSED(listener);
41+
UNUSED(callback);
42+
UNUSED(context);
43+
}
44+
45+
const FelicaData* felica_listener_get_data(const FelicaListener* instance) {
46+
furi_assert(instance);
47+
furi_assert(instance->data);
48+
49+
return instance->data;
50+
}
51+
52+
NfcCommand felica_listener_run(NfcGenericEvent event, void* context) {
53+
furi_assert(context);
54+
furi_assert(event.protocol == NfcProtocolInvalid);
55+
furi_assert(event.event_data);
56+
57+
FelicaListener* instance = context;
58+
NfcEvent* nfc_event = event.event_data;
59+
NfcCommand command = NfcCommandContinue;
60+
61+
if(nfc_event->type == NfcEventTypeListenerActivated) {
62+
instance->state = Felica_ListenerStateActivated;
63+
FURI_LOG_D(TAG, "Activated");
64+
} else if(nfc_event->type == NfcEventTypeFieldOff) {
65+
instance->state = Felica_ListenerStateIdle;
66+
FURI_LOG_D(TAG, "Field Off");
67+
} else if(nfc_event->type == NfcEventTypeRxEnd) {
68+
FURI_LOG_D(TAG, "Rx Done");
69+
}
70+
return command;
71+
}
72+
73+
const NfcListenerBase nfc_listener_felica = {
74+
.alloc = (NfcListenerAlloc)felica_listener_alloc,
75+
.free = (NfcListenerFree)felica_listener_free,
76+
.set_callback = (NfcListenerSetCallback)felica_listener_set_callback,
77+
.get_data = (NfcListenerGetData)felica_listener_get_data,
78+
.run = (NfcListenerRun)felica_listener_run,
79+
};
+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
#pragma once
2+
3+
#include "felica.h"
4+
#include <lib/nfc/nfc.h>
5+
6+
#ifdef __cplusplus
7+
extern "C" {
8+
#endif
9+
10+
typedef struct FelicaListener FelicaListener;
11+
12+
#ifdef __cplusplus
13+
}
14+
#endif
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
#pragma once
2+
3+
#include <nfc/protocols/nfc_listener_base.h>
4+
5+
#ifdef __cplusplus
6+
extern "C" {
7+
#endif
8+
9+
extern const NfcListenerBase nfc_listener_felica;
10+
11+
#ifdef __cplusplus
12+
}
13+
#endif
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
#include "felica_listener.h"
2+
3+
#include <nfc/protocols/nfc_generic_event.h>
4+
5+
typedef enum {
6+
Felica_ListenerStateIdle,
7+
Felica_ListenerStateActivated,
8+
} FelicaListenerState;
9+
10+
struct FelicaListener {
11+
Nfc* nfc;
12+
FelicaData* data;
13+
FelicaListenerState state;
14+
15+
BitBuffer* tx_buffer;
16+
BitBuffer* rx_buffer;
17+
18+
NfcGenericEvent generic_event;
19+
NfcGenericCallback callback;
20+
void* context;
21+
};

lib/nfc/protocols/nfc_listener_defs.c

+2
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@
66
#include <nfc/protocols/mf_ultralight/mf_ultralight_listener_defs.h>
77
#include <nfc/protocols/mf_classic/mf_classic_listener_defs.h>
88
#include <nfc/protocols/slix/slix_listener_defs.h>
9+
#include <nfc/protocols/felica/felica_listener_defs.h>
910

1011
const NfcListenerBase* nfc_listeners_api[NfcProtocolNum] = {
1112
[NfcProtocolIso14443_3a] = &nfc_listener_iso14443_3a,
@@ -18,4 +19,5 @@ const NfcListenerBase* nfc_listeners_api[NfcProtocolNum] = {
1819
[NfcProtocolMfDesfire] = NULL,
1920
[NfcProtocolSlix] = &nfc_listener_slix,
2021
[NfcProtocolSt25tb] = NULL,
22+
[NfcProtocolFelica] = &nfc_listener_felica,
2123
};

targets/f7/api_symbols.csv

+2
Original file line numberDiff line numberDiff line change
@@ -1287,6 +1287,7 @@ Function,+,furi_hal_nfc_abort,FuriHalNfcError,
12871287
Function,+,furi_hal_nfc_acquire,FuriHalNfcError,
12881288
Function,+,furi_hal_nfc_event_start,FuriHalNfcError,
12891289
Function,+,furi_hal_nfc_event_stop,FuriHalNfcError,
1290+
Function,+,furi_hal_nfc_felica_listener_set_sensf_res_data,FuriHalNfcError,"const uint8_t*, const uint8_t, const uint8_t*, const uint8_t"
12901291
Function,+,furi_hal_nfc_field_detect_start,FuriHalNfcError,
12911292
Function,+,furi_hal_nfc_field_detect_stop,FuriHalNfcError,
12921293
Function,+,furi_hal_nfc_field_is_present,_Bool,
@@ -2316,6 +2317,7 @@ Function,+,nfc_dict_get_next_key,_Bool,"NfcDict*, uint8_t*, size_t"
23162317
Function,+,nfc_dict_get_total_keys,uint32_t,NfcDict*
23172318
Function,+,nfc_dict_is_key_present,_Bool,"NfcDict*, const uint8_t*, size_t"
23182319
Function,+,nfc_dict_rewind,_Bool,NfcDict*
2320+
Function,+,nfc_felica_listener_set_sensf_res_data,NfcError,"Nfc*, const uint8_t*, const uint8_t, const uint8_t*, const uint8_t"
23192321
Function,+,nfc_free,void,Nfc*
23202322
Function,+,nfc_iso14443a_listener_set_col_res_data,NfcError,"Nfc*, uint8_t*, uint8_t, uint8_t*, uint8_t"
23212323
Function,+,nfc_iso14443a_listener_tx_custom_parity,NfcError,"Nfc*, const BitBuffer*"

0 commit comments

Comments
 (0)