mirror of
https://github.com/DarkFlippers/unleashed-firmware
synced 2024-11-28 07:20:29 +00:00
c186d2b0cc
* added support for ISO15693 (NfcV) emulation, added support for reading SLIX tags * SLIX: fixed crash situation when an invalid password was requested * ISO15693: show emulate menu when opening file * rename NfcV emulate scene to match other NfcV names * optimize allocation size for signals * ISO15693: further optimizations of allocation and free code * ISO15693: reduce latency on state machine reset * respond with block security status when option flag is set * increased maximum memory size to match standard added security status handling/load/save added SELECT/QUIET handling more fine grained allocation routines and checks fix memset sizes * added "Listen NfcV Reader" to sniff traffic from reader to card * added correct description to delete menu * also added DSFID/AFI handling and locking * increase sniff log size * scale NfcV frequency a bit, add echo mode, fix signal level at the end * use symbolic modulated/unmodulated GPIO levels * honor AFI field, decrease verbosity and removed debug code * refactor defines for less namespace pollution by using NFCV_ prefixes * correct an oversight that original cards return an generic error when addressing outside block range * use inverse modulation, increasing readable range significantly * rework and better document nfc chip initialization * nfcv code review fixes * Disable accidentally left on signal debug gpio output * Improve NFCV Read/Info GUIs. Authored by @xMasterX, committed by @nvx * Fix crash that occurs when you exit from NFCV emulation and start it again. Authored by @xMasterX, committed by @nvx * Remove delay from emulation loop. This improves compatibility when the reader is Android. * Lib: digital signal debug output pin info Co-authored-by: Tiernan Messmer <tiernan.messmer@gmail.com> Co-authored-by: MX <10697207+xMasterX@users.noreply.github.com> Co-authored-by: gornekich <n.gorbadey@gmail.com> Co-authored-by: あく <alleteam@gmail.com>
169 lines
6.3 KiB
C
169 lines
6.3 KiB
C
#include "../nfc_i.h"
|
|
|
|
#define NFC_SCENE_EMULATE_NFCV_LOG_SIZE_MAX (200)
|
|
|
|
enum {
|
|
NfcSceneNfcVEmulateStateWidget,
|
|
NfcSceneNfcVEmulateStateTextBox,
|
|
};
|
|
|
|
bool nfc_scene_nfcv_emulate_worker_callback(NfcWorkerEvent event, void* context) {
|
|
furi_assert(context);
|
|
Nfc* nfc = context;
|
|
|
|
switch(event) {
|
|
case NfcWorkerEventNfcVCommandExecuted:
|
|
if(furi_hal_rtc_is_flag_set(FuriHalRtcFlagDebug)) {
|
|
view_dispatcher_send_custom_event(nfc->view_dispatcher, NfcCustomEventUpdateLog);
|
|
}
|
|
break;
|
|
case NfcWorkerEventNfcVContentChanged:
|
|
view_dispatcher_send_custom_event(nfc->view_dispatcher, NfcCustomEventSaveShadow);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void nfc_scene_nfcv_emulate_widget_callback(GuiButtonType result, InputType type, void* context) {
|
|
furi_assert(context);
|
|
Nfc* nfc = context;
|
|
if(type == InputTypeShort) {
|
|
view_dispatcher_send_custom_event(nfc->view_dispatcher, result);
|
|
}
|
|
}
|
|
|
|
void nfc_scene_nfcv_emulate_textbox_callback(void* context) {
|
|
furi_assert(context);
|
|
Nfc* nfc = context;
|
|
view_dispatcher_send_custom_event(nfc->view_dispatcher, NfcCustomEventViewExit);
|
|
}
|
|
|
|
static void nfc_scene_nfcv_emulate_widget_config(Nfc* nfc, bool data_received) {
|
|
FuriHalNfcDevData* data = &nfc->dev->dev_data.nfc_data;
|
|
Widget* widget = nfc->widget;
|
|
widget_reset(widget);
|
|
FuriString* info_str;
|
|
info_str = furi_string_alloc();
|
|
|
|
widget_add_icon_element(widget, 0, 3, &I_NFC_dolphin_emulation_47x61);
|
|
widget_add_string_multiline_element(
|
|
widget, 87, 13, AlignCenter, AlignTop, FontPrimary, "Emulating\nNFC V");
|
|
if(strcmp(nfc->dev->dev_name, "")) {
|
|
furi_string_printf(info_str, "%s", nfc->dev->dev_name);
|
|
} else {
|
|
for(uint8_t i = 0; i < data->uid_len; i++) {
|
|
furi_string_cat_printf(info_str, "%02X ", data->uid[i]);
|
|
}
|
|
}
|
|
furi_string_trim(info_str);
|
|
widget_add_text_box_element(
|
|
widget, 52, 40, 70, 21, AlignCenter, AlignTop, furi_string_get_cstr(info_str), true);
|
|
furi_string_free(info_str);
|
|
if(data_received) {
|
|
if(furi_hal_rtc_is_flag_set(FuriHalRtcFlagDebug)) {
|
|
widget_add_button_element(
|
|
widget, GuiButtonTypeCenter, "Log", nfc_scene_nfcv_emulate_widget_callback, nfc);
|
|
}
|
|
}
|
|
}
|
|
|
|
void nfc_scene_nfcv_emulate_on_enter(void* context) {
|
|
Nfc* nfc = context;
|
|
|
|
// Setup Widget
|
|
nfc_scene_nfcv_emulate_widget_config(nfc, false);
|
|
// Setup TextBox
|
|
TextBox* text_box = nfc->text_box;
|
|
text_box_set_font(text_box, TextBoxFontHex);
|
|
text_box_set_focus(text_box, TextBoxFocusEnd);
|
|
text_box_set_text(text_box, "");
|
|
furi_string_reset(nfc->text_box_store);
|
|
|
|
// Set Widget state and view
|
|
scene_manager_set_scene_state(
|
|
nfc->scene_manager, NfcSceneNfcVEmulate, NfcSceneNfcVEmulateStateWidget);
|
|
view_dispatcher_switch_to_view(nfc->view_dispatcher, NfcViewWidget);
|
|
// Start worker
|
|
memset(&nfc->dev->dev_data.reader_data, 0, sizeof(NfcReaderRequestData));
|
|
nfc_worker_start(
|
|
nfc->worker,
|
|
NfcWorkerStateNfcVEmulate,
|
|
&nfc->dev->dev_data,
|
|
nfc_scene_nfcv_emulate_worker_callback,
|
|
nfc);
|
|
|
|
nfc_blink_emulate_start(nfc);
|
|
}
|
|
|
|
bool nfc_scene_nfcv_emulate_on_event(void* context, SceneManagerEvent event) {
|
|
Nfc* nfc = context;
|
|
NfcVData* nfcv_data = &nfc->dev->dev_data.nfcv_data;
|
|
uint32_t state = scene_manager_get_scene_state(nfc->scene_manager, NfcSceneNfcVEmulate);
|
|
bool consumed = false;
|
|
|
|
if(event.type == SceneManagerEventTypeCustom) {
|
|
if(event.event == NfcCustomEventUpdateLog) {
|
|
// Add data button to widget if data is received for the first time
|
|
if(strlen(nfcv_data->last_command) > 0) {
|
|
if(!furi_string_size(nfc->text_box_store)) {
|
|
nfc_scene_nfcv_emulate_widget_config(nfc, true);
|
|
}
|
|
/* use the last n bytes from the log so there's enough space for the new log entry */
|
|
size_t maxSize =
|
|
NFC_SCENE_EMULATE_NFCV_LOG_SIZE_MAX - (strlen(nfcv_data->last_command) + 1);
|
|
if(furi_string_size(nfc->text_box_store) >= maxSize) {
|
|
furi_string_right(nfc->text_box_store, (strlen(nfcv_data->last_command) + 1));
|
|
}
|
|
furi_string_cat_printf(nfc->text_box_store, "%s", nfcv_data->last_command);
|
|
furi_string_push_back(nfc->text_box_store, '\n');
|
|
text_box_set_text(nfc->text_box, furi_string_get_cstr(nfc->text_box_store));
|
|
|
|
/* clear previously logged command */
|
|
strcpy(nfcv_data->last_command, "");
|
|
}
|
|
consumed = true;
|
|
} else if(event.event == NfcCustomEventSaveShadow) {
|
|
if(furi_string_size(nfc->dev->load_path)) {
|
|
nfc_device_save_shadow(nfc->dev, furi_string_get_cstr(nfc->dev->load_path));
|
|
}
|
|
consumed = true;
|
|
} else if(event.event == GuiButtonTypeCenter && state == NfcSceneNfcVEmulateStateWidget) {
|
|
if(furi_hal_rtc_is_flag_set(FuriHalRtcFlagDebug)) {
|
|
view_dispatcher_switch_to_view(nfc->view_dispatcher, NfcViewTextBox);
|
|
scene_manager_set_scene_state(
|
|
nfc->scene_manager, NfcSceneNfcVEmulate, NfcSceneNfcVEmulateStateTextBox);
|
|
}
|
|
consumed = true;
|
|
} else if(event.event == NfcCustomEventViewExit && state == NfcSceneNfcVEmulateStateTextBox) {
|
|
view_dispatcher_switch_to_view(nfc->view_dispatcher, NfcViewWidget);
|
|
scene_manager_set_scene_state(
|
|
nfc->scene_manager, NfcSceneNfcVEmulate, NfcSceneNfcVEmulateStateWidget);
|
|
consumed = true;
|
|
}
|
|
} else if(event.type == SceneManagerEventTypeBack) {
|
|
if(state == NfcSceneNfcVEmulateStateTextBox) {
|
|
view_dispatcher_switch_to_view(nfc->view_dispatcher, NfcViewWidget);
|
|
scene_manager_set_scene_state(
|
|
nfc->scene_manager, NfcSceneNfcVEmulate, NfcSceneNfcVEmulateStateWidget);
|
|
consumed = true;
|
|
}
|
|
}
|
|
|
|
return consumed;
|
|
}
|
|
|
|
void nfc_scene_nfcv_emulate_on_exit(void* context) {
|
|
Nfc* nfc = context;
|
|
|
|
// Stop worker
|
|
nfc_worker_stop(nfc->worker);
|
|
|
|
// Clear view
|
|
widget_reset(nfc->widget);
|
|
text_box_reset(nfc->text_box);
|
|
furi_string_reset(nfc->text_box_store);
|
|
|
|
nfc_blink_stop(nfc);
|
|
}
|