Files
seader/seader.c
T
2026-03-24 14:47:14 -07:00

1074 lines
33 KiB
C

#include "seader_i.h"
#include "trace_log.h"
#define TAG "Seader"
#define SEADER_PLUGIN_DIR APP_ASSETS_PATH("plugins")
#define SEADER_WIEGAND_PLUGIN_PATH APP_ASSETS_PATH("plugins/plugin_wiegand.fal")
#define SEADER_HF_PLUGIN_PATH APP_ASSETS_PATH("plugins/plugin_hf.fal")
typedef struct {
volatile bool done;
volatile bool detected;
} SeaderHfPicopassDetectContext;
static void seader_hf_worker_event_callback(uint32_t event, void* context);
static void seader_hf_teardown_blocking(Seader* seader);
static NfcCommand seader_hf_picopass_detect_callback(PicopassPollerEvent event, void* context) {
SeaderHfPicopassDetectContext* detect_context = context;
if(event.type == PicopassPollerEventTypeCardDetected ||
event.type == PicopassPollerEventTypeSuccess) {
detect_context->detected = true;
detect_context->done = true;
return NfcCommandStop;
} else if(event.type == PicopassPollerEventTypeFail) {
detect_context->done = true;
return NfcCommandStop;
}
return NfcCommandContinue;
}
static void seader_hf_plugin_notify_event(void* host_ctx, uint32_t event) {
Seader* seader = host_ctx;
if(!seader || !seader->view_dispatcher) {
FURI_LOG_W(TAG, "Drop HF plugin event %lu without dispatcher", event);
return;
}
view_dispatcher_send_custom_event(seader->view_dispatcher, event);
}
static void seader_hf_plugin_notify_card_detected(void* host_ctx) {
seader_hf_plugin_notify_event(host_ctx, SeaderCustomEventPollerDetect);
}
static void seader_hf_plugin_notify_worker_exit(void* host_ctx) {
seader_hf_plugin_notify_event(host_ctx, SeaderCustomEventWorkerExit);
}
static bool seader_hf_plugin_sam_can_accept_card(void* host_ctx) {
return seader_sam_can_accept_card(host_ctx);
}
static void seader_hf_plugin_send_card_detected(
void* host_ctx,
uint8_t sak,
const uint8_t* uid,
uint8_t uid_len,
const uint8_t* ats,
uint8_t ats_len) {
Seader* seader = host_ctx;
if(!seader || !seader->worker || !seader->credential || !uid || uid_len == 0U) {
FURI_LOG_E(
TAG,
"Drop HF cardDetected invalid state seader=%p worker=%p cred=%p uid=%p uid_len=%u",
(void*)seader,
seader ? (void*)seader->worker : NULL,
seader ? (void*)seader->credential : NULL,
(const void*)uid,
uid_len);
return;
}
FURI_LOG_D(
TAG,
"HF plugin cardDetected sak=%02x uid_len=%u ats_len=%u stage=%d",
sak,
uid_len,
ats_len,
seader->worker->stage);
seader_worker_card_detect(seader, sak, NULL, uid, uid_len, (uint8_t*)ats, ats_len);
}
static void seader_hf_plugin_send_nfc_rx(void* host_ctx, uint8_t* buffer, size_t len) {
Seader* seader = host_ctx;
seader_send_nfc_rx(seader, buffer, len);
}
static void seader_hf_plugin_run_conversation(void* host_ctx) {
Seader* seader = host_ctx;
if(!seader || !seader->worker) {
FURI_LOG_W(TAG, "Skip HF conversation without worker");
return;
}
FURI_LOG_D(TAG, "HF plugin run conversation stage=%d", seader->worker->stage);
seader_worker_run_hf_conversation(seader);
}
static void seader_hf_plugin_set_stage(void* host_ctx, PluginHfStage stage) {
Seader* seader = host_ctx;
if(seader->worker) {
switch(stage) {
case PluginHfStageCardDetect:
seader->worker->stage = SeaderPollerEventTypeCardDetect;
break;
case PluginHfStageConversation:
seader->worker->stage = SeaderPollerEventTypeConversation;
break;
case PluginHfStageComplete:
seader->worker->stage = SeaderPollerEventTypeComplete;
break;
case PluginHfStageSuccess:
seader->worker->stage = SeaderPollerEventTypeSuccess;
break;
case PluginHfStageFail:
default:
seader->worker->stage = SeaderPollerEventTypeFail;
break;
}
}
}
static PluginHfStage seader_hf_plugin_get_stage(void* host_ctx) {
Seader* seader = host_ctx;
if(!seader->worker) {
return PluginHfStageFail;
}
switch(seader->worker->stage) {
case SeaderPollerEventTypeCardDetect:
return PluginHfStageCardDetect;
case SeaderPollerEventTypeConversation:
return PluginHfStageConversation;
case SeaderPollerEventTypeComplete:
return PluginHfStageComplete;
case SeaderPollerEventTypeSuccess:
return PluginHfStageSuccess;
case SeaderPollerEventTypeFail:
default:
return PluginHfStageFail;
}
}
static void seader_hf_plugin_set_credential_type(void* host_ctx, SeaderCredentialType type) {
Seader* seader = host_ctx;
seader->credential->type = type;
seader->credential->sio_len = 0U;
seader->credential->sio_start_block = 0U;
seader->credential->isDesfireEV2 = false;
}
static SeaderCredentialType seader_hf_plugin_get_credential_type(void* host_ctx) {
Seader* seader = host_ctx;
return seader->credential->type;
}
static bool seader_hf_plugin_get_desfire_ev2(void* host_ctx) {
Seader* seader = host_ctx;
return seader->credential->isDesfireEV2;
}
static void seader_hf_plugin_set_desfire_ev2(void* host_ctx, bool is_desfire_ev2) {
Seader* seader = host_ctx;
seader->credential->isDesfireEV2 = is_desfire_ev2;
}
static void seader_hf_plugin_append_picopass_sio(
void* host_ctx,
uint8_t block_num,
const uint8_t* data,
size_t len) {
Seader* seader = host_ctx;
SeaderCredential* credential = seader->credential;
if(!data || len == 0U || credential->type != SeaderCredentialTypePicopass) {
return;
}
if(credential->sio_len == 0U && data[0] == 0x30U) {
credential->sio_start_block = block_num;
}
const size_t offset = (size_t)(block_num - credential->sio_start_block) * PICOPASS_BLOCK_LEN;
if(offset >= sizeof(credential->sio)) {
return;
}
const size_t copy_len = MIN(len, sizeof(credential->sio) - offset);
memcpy(credential->sio + offset, data, copy_len);
credential->sio_len = MAX(credential->sio_len, offset + copy_len);
}
static void seader_hf_plugin_set_14a_sio(void* host_ctx, const uint8_t* data, size_t len) {
Seader* seader = host_ctx;
SeaderCredential* credential = seader->credential;
if(!data || credential->type != SeaderCredentialType14A) {
return;
}
const size_t copy_len = MIN(len, sizeof(credential->sio));
memcpy(credential->sio, data, copy_len);
credential->sio_len = copy_len;
}
static Nfc* seader_hf_plugin_get_nfc(void* host_ctx) {
Seader* seader = host_ctx;
return seader ? seader->nfc : NULL;
}
static NfcDevice* seader_hf_plugin_get_nfc_device(void* host_ctx) {
Seader* seader = host_ctx;
return seader ? seader->nfc_device : NULL;
}
static bool seader_hf_plugin_picopass_detect(void* host_ctx) {
Seader* seader = host_ctx;
bool detected = false;
PicopassPoller* poller = picopass_poller_alloc(seader->nfc);
SeaderHfPicopassDetectContext detect_context = {0};
if(!poller) {
FURI_LOG_W(TAG, "Failed to allocate Picopass detect poller");
return false;
}
picopass_poller_start(poller, seader_hf_picopass_detect_callback, &detect_context);
for(uint8_t i = 0; i < 10 && !detect_context.done; i++) {
furi_delay_ms(10);
}
picopass_poller_stop(poller);
detected = detect_context.detected;
picopass_poller_free(poller);
return detected;
}
static bool seader_hf_plugin_picopass_start(
void* host_ctx,
PicopassPollerCallback callback,
void* callback_ctx) {
Seader* seader = host_ctx;
if(seader->picopass_poller) {
picopass_poller_stop(seader->picopass_poller);
picopass_poller_free(seader->picopass_poller);
seader->picopass_poller = NULL;
}
seader->picopass_poller = picopass_poller_alloc(seader->nfc);
if(!seader->picopass_poller) {
return false;
}
picopass_poller_start(seader->picopass_poller, callback, callback_ctx);
return true;
}
static void seader_hf_plugin_picopass_stop(void* host_ctx) {
Seader* seader = host_ctx;
if(seader->picopass_poller) {
picopass_poller_stop(seader->picopass_poller);
picopass_poller_free(seader->picopass_poller);
seader->picopass_poller = NULL;
}
}
static uint8_t* seader_hf_plugin_picopass_get_csn(void* host_ctx) {
Seader* seader = host_ctx;
if(!seader->picopass_poller) {
return NULL;
}
return picopass_poller_get_csn(seader->picopass_poller);
}
static bool seader_hf_plugin_picopass_transmit(
void* host_ctx,
const uint8_t* tx_data,
size_t tx_len,
uint8_t* rx_data,
size_t rx_capacity,
size_t* rx_len,
uint32_t fwt_fc) {
Seader* seader = host_ctx;
if(!seader->picopass_poller || !tx_data || !rx_data || !rx_len) {
return false;
}
BitBuffer* tx_buffer = bit_buffer_alloc(tx_len);
BitBuffer* rx_buffer = bit_buffer_alloc(rx_capacity);
bool success = false;
if(!tx_buffer || !rx_buffer) {
FURI_LOG_E(TAG, "Failed to allocate picopass host tx/rx buffers");
if(tx_buffer) bit_buffer_free(tx_buffer);
if(rx_buffer) bit_buffer_free(rx_buffer);
return false;
}
bit_buffer_append_bytes(tx_buffer, tx_data, tx_len);
PicopassError error =
picopass_poller_send_frame(seader->picopass_poller, tx_buffer, rx_buffer, fwt_fc);
if(error == PicopassErrorIncorrectCrc) {
error = PicopassErrorNone;
}
if(error == PicopassErrorNone) {
*rx_len = bit_buffer_get_size_bytes(rx_buffer);
memcpy(rx_data, bit_buffer_get_data(rx_buffer), *rx_len);
success = true;
}
bit_buffer_free(tx_buffer);
bit_buffer_free(rx_buffer);
return success;
}
static void seader_hf_plugin_set_read_error(void* host_ctx, const char* text) {
Seader* seader = host_ctx;
if(!text) {
seader->read_error[0] = '\0';
return;
}
strlcpy(seader->read_error, text, sizeof(seader->read_error));
}
static const PluginHfHostApi seader_hf_plugin_host_api = {
.notify_card_detected = seader_hf_plugin_notify_card_detected,
.notify_worker_exit = seader_hf_plugin_notify_worker_exit,
.sam_can_accept_card = seader_hf_plugin_sam_can_accept_card,
.send_card_detected = seader_hf_plugin_send_card_detected,
.send_nfc_rx = seader_hf_plugin_send_nfc_rx,
.run_conversation = seader_hf_plugin_run_conversation,
.set_stage = seader_hf_plugin_set_stage,
.get_stage = seader_hf_plugin_get_stage,
.set_credential_type = seader_hf_plugin_set_credential_type,
.get_credential_type = seader_hf_plugin_get_credential_type,
.get_desfire_ev2 = seader_hf_plugin_get_desfire_ev2,
.set_desfire_ev2 = seader_hf_plugin_set_desfire_ev2,
.append_picopass_sio = seader_hf_plugin_append_picopass_sio,
.set_14a_sio = seader_hf_plugin_set_14a_sio,
.get_nfc = seader_hf_plugin_get_nfc,
.get_nfc_device = seader_hf_plugin_get_nfc_device,
.picopass_detect = seader_hf_plugin_picopass_detect,
.picopass_start = seader_hf_plugin_picopass_start,
.picopass_stop = seader_hf_plugin_picopass_stop,
.picopass_get_csn = seader_hf_plugin_picopass_get_csn,
.picopass_transmit = seader_hf_plugin_picopass_transmit,
.set_read_error = seader_hf_plugin_set_read_error,
};
static void seader_hf_worker_event_callback(uint32_t event, void* context) {
Seader* seader = context;
if(!seader || !seader->view_dispatcher) {
return;
}
view_dispatcher_send_custom_event(seader->view_dispatcher, event);
}
static void seader_hf_session_force_unloaded(Seader* seader) {
if(!seader) {
return;
}
seader->hf_plugin_ctx = NULL;
seader->plugin_hf = NULL;
seader->hf_plugin_manager = NULL;
seader->poller = NULL;
seader->picopass_poller = NULL;
seader->hf_session_state = SeaderHfSessionStateUnloaded;
if(seader->mode_runtime == SeaderModeRuntimeHF) {
seader->mode_runtime = SeaderModeRuntimeNone;
}
}
bool seader_custom_event_callback(void* context, uint32_t event) {
furi_assert(context);
Seader* seader = context;
return scene_manager_handle_custom_event(seader->scene_manager, event);
}
bool seader_back_event_callback(void* context) {
furi_assert(context);
Seader* seader = context;
return scene_manager_handle_back_event(seader->scene_manager);
}
void seader_tick_event_callback(void* context) {
furi_assert(context);
Seader* seader = context;
scene_manager_handle_tick_event(seader->scene_manager);
}
static bool seader_align_is_valid(size_t align) {
return align != 0U && ((align & (align - 1U)) == 0U);
}
Seader* seader_alloc() {
Seader* seader = malloc(sizeof(Seader));
seader_trace_reset();
seader->revert_power = !furi_hal_power_is_otg_enabled();
if(seader->revert_power) {
furi_hal_power_enable_otg();
}
seader->is_debug_enabled = furi_hal_rtc_is_flag_set(FuriHalRtcFlagDebug);
seader->samCommand = SamCommand_PR_NOTHING;
seader->sam_state = SeaderSamStateIdle;
seader->sam_intent = SeaderSamIntentNone;
seader->sam_present = false;
memset(seader->sam_version, 0, sizeof(seader->sam_version));
seader_sam_key_label_format(
false, NULL, 0U, seader->sam_key_label, sizeof(seader->sam_key_label));
seader_uhf_status_label_format(
false, false, false, false, seader->uhf_status_label, sizeof(seader->uhf_status_label));
seader_uhf_snmp_probe_init(&seader->snmp_probe);
seader->nfc = nfc_alloc();
seader->nfc_device = seader->nfc ? nfc_device_alloc() : NULL;
seader->scratch.offset = 0U;
seader->scratch.high_water = 0U;
seader->hf_mode = NULL;
seader->worker = seader_worker_alloc();
seader->view_dispatcher = view_dispatcher_alloc();
seader->scene_manager = scene_manager_alloc(&seader_scene_handlers, seader);
view_dispatcher_set_event_callback_context(seader->view_dispatcher, seader);
view_dispatcher_set_custom_event_callback(
seader->view_dispatcher, seader_custom_event_callback);
view_dispatcher_set_navigation_event_callback(
seader->view_dispatcher, seader_back_event_callback);
view_dispatcher_set_tick_event_callback(
seader->view_dispatcher, seader_tick_event_callback, 100);
seader->uart = seader_uart_alloc(seader);
seader->credential = seader_credential_alloc();
if(!seader->nfc || !seader->nfc_device) {
FURI_LOG_W(
TAG,
"HF host NFC objects unavailable at startup nfc=%p device=%p",
seader->nfc,
seader->nfc_device);
}
// Open GUI record
seader->gui = furi_record_open(RECORD_GUI);
view_dispatcher_attach_to_gui(
seader->view_dispatcher, seader->gui, ViewDispatcherTypeFullscreen);
// Open Notification record
seader->notifications = furi_record_open(RECORD_NOTIFICATION);
// Submenu
seader->submenu = submenu_alloc();
view_dispatcher_add_view(
seader->view_dispatcher, SeaderViewMenu, submenu_get_view(seader->submenu));
// Popup
seader->popup = popup_alloc();
view_dispatcher_add_view(
seader->view_dispatcher, SeaderViewPopup, popup_get_view(seader->popup));
// Loading
seader->loading = loading_alloc();
view_dispatcher_add_view(
seader->view_dispatcher, SeaderViewLoading, loading_get_view(seader->loading));
// Text Input
seader->text_input = text_input_alloc();
view_dispatcher_add_view(
seader->view_dispatcher, SeaderViewTextInput, text_input_get_view(seader->text_input));
// TextBox
seader->text_box = text_box_alloc();
view_dispatcher_add_view(
seader->view_dispatcher, SeaderViewTextBox, text_box_get_view(seader->text_box));
seader->text_box_store = furi_string_alloc();
// Custom Widget
seader->widget = widget_alloc();
view_dispatcher_add_view(
seader->view_dispatcher, SeaderViewWidget, widget_get_view(seader->widget));
// Allocate reusable strings for scene optimization
seader->temp_string1 = furi_string_alloc();
seader->temp_string2 = furi_string_alloc();
seader->temp_string3 = furi_string_alloc();
seader->temp_string4 = furi_string_alloc();
seader->plugin_manager = NULL;
seader->plugin_wiegand = NULL;
seader->hf_plugin_manager = NULL;
seader->plugin_hf = NULL;
seader->hf_plugin_ctx = NULL;
seader->mode_runtime = SeaderModeRuntimeNone;
seader->hf_session_state = SeaderHfSessionStateUnloaded;
seader->hf_teardown_action = SeaderHfTeardownActionNone;
seader->loading_popup_enabled = true;
if(seader->nfc_device) {
nfc_device_set_loading_callback(seader->nfc_device, seader_nfc_loading_callback, seader);
}
return seader;
}
void seader_free(Seader* seader) {
furi_assert(seader);
if(seader->revert_power) {
furi_hal_power_disable_otg();
}
seader->loading_popup_enabled = false;
seader_hf_teardown_blocking(seader);
seader_hf_mode_deactivate(seader);
seader_worker_release(seader);
if(seader->worker) {
seader_worker_free(seader->worker);
seader->worker = NULL;
}
seader_wiegand_plugin_release(seader);
if(seader->nfc_device) {
nfc_device_free(seader->nfc_device);
seader->nfc_device = NULL;
}
if(seader->nfc) {
nfc_free(seader->nfc);
seader->nfc = NULL;
}
seader_uart_free(seader->uart);
seader->uart = NULL;
seader_credential_free(seader->credential);
seader->credential = NULL;
// Submenu
view_dispatcher_remove_view(seader->view_dispatcher, SeaderViewMenu);
submenu_free(seader->submenu);
// Popup
view_dispatcher_remove_view(seader->view_dispatcher, SeaderViewPopup);
popup_free(seader->popup);
// Loading
view_dispatcher_remove_view(seader->view_dispatcher, SeaderViewLoading);
loading_free(seader->loading);
// TextInput
view_dispatcher_remove_view(seader->view_dispatcher, SeaderViewTextInput);
text_input_free(seader->text_input);
// TextBox
view_dispatcher_remove_view(seader->view_dispatcher, SeaderViewTextBox);
text_box_free(seader->text_box);
furi_string_free(seader->text_box_store);
// Custom Widget
view_dispatcher_remove_view(seader->view_dispatcher, SeaderViewWidget);
widget_free(seader->widget);
// Free reusable strings
furi_string_free(seader->temp_string1);
furi_string_free(seader->temp_string2);
furi_string_free(seader->temp_string3);
furi_string_free(seader->temp_string4);
// View Dispatcher
view_dispatcher_free(seader->view_dispatcher);
// Scene Manager
scene_manager_free(seader->scene_manager);
// GUI
furi_record_close(RECORD_GUI);
seader->gui = NULL;
// Notifications
furi_record_close(RECORD_NOTIFICATION);
seader->notifications = NULL;
free(seader);
}
void seader_text_store_set(Seader* seader, const char* text, ...) {
va_list args;
va_start(args, text);
vsnprintf(seader->text_store, sizeof(seader->text_store), text, args);
va_end(args);
}
void seader_text_store_clear(Seader* seader) {
memset(seader->text_store, 0, sizeof(seader->text_store));
}
static const NotificationSequence seader_sequence_blink_start_blue = {
&message_blink_start_10,
&message_blink_set_color_blue,
&message_do_not_reset,
NULL,
};
static const NotificationSequence seader_sequence_blink_stop = {
&message_blink_stop,
NULL,
};
void seader_blink_start(Seader* seader) {
notification_message(seader->notifications, &seader_sequence_blink_start_blue);
}
void seader_blink_stop(Seader* seader) {
notification_message(seader->notifications, &seader_sequence_blink_stop);
}
void seader_nfc_loading_callback(void* context, bool show) {
Seader* seader = context;
if(!seader || !seader->loading_popup_enabled || !seader->view_dispatcher) {
return;
}
seader_show_loading_popup(seader, show);
}
void seader_show_loading_popup(void* context, bool show) {
Seader* seader = context;
if(!seader || !seader->loading_popup_enabled || !seader->view_dispatcher) {
return;
}
if(show) {
// Raise timer priority so that animations can play
furi_timer_set_thread_priority(FuriTimerThreadPriorityElevated);
view_dispatcher_switch_to_view(seader->view_dispatcher, SeaderViewLoading);
} else {
// Restore default timer priority
furi_timer_set_thread_priority(FuriTimerThreadPriorityNormal);
}
}
bool seader_wiegand_plugin_acquire(Seader* seader) {
furi_assert(seader);
if(seader->plugin_wiegand) {
return true;
}
if(!seader->plugin_manager) {
seader->plugin_manager =
plugin_manager_alloc(PLUGIN_APP_ID, PLUGIN_API_VERSION, firmware_api_interface);
if(!seader->plugin_manager) {
FURI_LOG_E(TAG, "Failed to allocate plugin manager");
return false;
}
}
FURI_LOG_I(TAG, "Loading Wiegand plugin from %s", SEADER_WIEGAND_PLUGIN_PATH);
if(plugin_manager_load_single(seader->plugin_manager, SEADER_WIEGAND_PLUGIN_PATH) !=
PluginManagerErrorNone) {
FURI_LOG_E(TAG, "Failed to load Wiegand plugin");
plugin_manager_free(seader->plugin_manager);
seader->plugin_manager = NULL;
return false;
}
if(plugin_manager_get_count(seader->plugin_manager) == 0) {
FURI_LOG_E(TAG, "Wiegand plugin manager is empty after load");
plugin_manager_free(seader->plugin_manager);
seader->plugin_manager = NULL;
return false;
}
seader->plugin_wiegand = (PluginWiegand*)plugin_manager_get_ep(seader->plugin_manager, 0);
if(!seader->plugin_wiegand) {
FURI_LOG_E(TAG, "Failed to resolve Wiegand plugin entry point");
plugin_manager_free(seader->plugin_manager);
seader->plugin_manager = NULL;
return false;
}
FURI_LOG_I(TAG, "Wiegand plugin loaded: %s", seader->plugin_wiegand->name);
return true;
}
void seader_wiegand_plugin_release(Seader* seader) {
furi_assert(seader);
if(!seader->plugin_manager) {
seader->plugin_wiegand = NULL;
return;
}
FURI_LOG_I(TAG, "Unloading Wiegand plugin");
seader->plugin_wiegand = NULL;
plugin_manager_free(seader->plugin_manager);
seader->plugin_manager = NULL;
}
bool seader_hf_plugin_acquire(Seader* seader) {
furi_assert(seader);
if(seader->mode_runtime == SeaderModeRuntimeUHF) {
FURI_LOG_W(TAG, "Reject HF plugin acquire while UHF runtime is active");
return false;
}
if(seader->hf_session_state == SeaderHfSessionStateTearingDown) {
FURI_LOG_W(TAG, "Reject HF plugin acquire during teardown");
return false;
}
if(seader->plugin_hf && seader->hf_plugin_ctx) {
if(seader->hf_session_state == SeaderHfSessionStateUnloaded) {
seader->hf_session_state = SeaderHfSessionStateLoaded;
}
seader->mode_runtime = SeaderModeRuntimeHF;
return true;
}
if(seader->hf_plugin_manager || seader->plugin_hf || seader->hf_plugin_ctx) {
FURI_LOG_W(
TAG,
"Normalize partial HF session manager=%p plugin=%p ctx=%p state=%d",
(void*)seader->hf_plugin_manager,
(void*)seader->plugin_hf,
seader->hf_plugin_ctx,
seader->hf_session_state);
seader_hf_plugin_release(seader);
}
if(!seader->nfc || !seader->nfc_device) {
FURI_LOG_E(
TAG,
"Host NFC objects unavailable nfc=%p device=%p",
seader->nfc,
seader->nfc_device);
return false;
}
if(!seader->hf_plugin_manager) {
seader->hf_plugin_manager =
plugin_manager_alloc(HF_PLUGIN_APP_ID, HF_PLUGIN_API_VERSION, firmware_api_interface);
if(!seader->hf_plugin_manager) {
FURI_LOG_E(TAG, "Failed to allocate HF plugin manager");
return false;
}
}
FURI_LOG_I(TAG, "Loading HF plugin from %s", SEADER_HF_PLUGIN_PATH);
if(plugin_manager_load_single(seader->hf_plugin_manager, SEADER_HF_PLUGIN_PATH) !=
PluginManagerErrorNone) {
FURI_LOG_E(TAG, "Failed to load HF plugin");
plugin_manager_free(seader->hf_plugin_manager);
seader_hf_session_force_unloaded(seader);
return false;
}
seader->plugin_hf = (PluginHf*)plugin_manager_get_ep(seader->hf_plugin_manager, 0);
if(!seader->plugin_hf) {
FURI_LOG_E(TAG, "Failed to resolve HF plugin entry point");
plugin_manager_free(seader->hf_plugin_manager);
seader_hf_session_force_unloaded(seader);
return false;
}
seader->hf_plugin_ctx = seader->plugin_hf->alloc(&seader_hf_plugin_host_api, seader);
if(!seader->hf_plugin_ctx) {
FURI_LOG_E(TAG, "Failed to allocate HF plugin context");
plugin_manager_free(seader->hf_plugin_manager);
seader_hf_session_force_unloaded(seader);
return false;
}
seader->hf_session_state = SeaderHfSessionStateLoaded;
seader->mode_runtime = SeaderModeRuntimeHF;
FURI_LOG_I(TAG, "HF plugin loaded: %s", seader->plugin_hf->name);
return true;
}
static bool seader_hf_has_runtime(const Seader* seader) {
return seader &&
(seader->hf_plugin_manager || seader->plugin_hf || seader->hf_plugin_ctx ||
seader->poller || seader->picopass_poller);
}
static void seader_hf_teardown_blocking(Seader* seader) {
if(!seader || !seader_hf_has_runtime(seader)) {
return;
}
seader->hf_session_state = SeaderHfSessionStateTearingDown;
if(!seader_worker_acquire(seader) || !seader->worker || !seader->uart) {
FURI_LOG_W(TAG, "HF blocking teardown fallback");
seader_hf_plugin_release(seader);
return;
}
seader_worker_stop(seader->worker);
FURI_LOG_I(TAG, "HF teardown blocking");
seader_worker_start(
seader->worker,
SeaderWorkerStateHfTeardown,
seader->uart,
NULL,
seader);
seader_worker_join(seader->worker);
}
void seader_hf_plugin_release(Seader* seader) {
furi_assert(seader);
seader->hf_session_state = SeaderHfSessionStateTearingDown;
if(seader->plugin_hf && seader->hf_plugin_ctx) {
seader->plugin_hf->stop(seader->hf_plugin_ctx);
}
if(seader->poller) {
FURI_LOG_I(TAG, "Stopping host NFC poller");
nfc_poller_stop(seader->poller);
nfc_poller_free(seader->poller);
seader->poller = NULL;
}
if(seader->picopass_poller) {
FURI_LOG_I(TAG, "Stopping host Picopass poller");
picopass_poller_stop(seader->picopass_poller);
picopass_poller_free(seader->picopass_poller);
seader->picopass_poller = NULL;
}
if(seader->plugin_hf && seader->hf_plugin_ctx) {
seader->plugin_hf->free(seader->hf_plugin_ctx);
}
seader->hf_plugin_ctx = NULL;
seader->plugin_hf = NULL;
if(seader->hf_plugin_manager) {
FURI_LOG_I(TAG, "Unloading HF plugin");
plugin_manager_free(seader->hf_plugin_manager);
seader->hf_plugin_manager = NULL;
}
if(seader->worker) {
seader_worker_reset_poller_session(seader->worker);
}
if(seader->mode_runtime == SeaderModeRuntimeHF) {
seader->mode_runtime = SeaderModeRuntimeNone;
}
seader->hf_session_state = SeaderHfSessionStateUnloaded;
}
bool seader_hf_finish_teardown_action(Seader* seader) {
if(!seader) {
return false;
}
FURI_LOG_I(TAG, "HF teardown complete action=%d", seader->hf_teardown_action);
seader_show_loading_popup(seader, false);
seader_hf_mode_set_selected_read_type(seader, SeaderCredentialTypeNone);
seader_hf_mode_clear_detected_types(seader);
seader_hf_mode_deactivate(seader);
const SeaderHfTeardownAction action = seader->hf_teardown_action;
seader->hf_teardown_action = SeaderHfTeardownActionNone;
switch(action) {
case SeaderHfTeardownActionSamPresent:
return scene_manager_search_and_switch_to_another_scene(
seader->scene_manager, SeaderSceneSamPresent);
case SeaderHfTeardownActionRestartRead:
scene_manager_next_scene(seader->scene_manager, SeaderSceneRead);
return true;
case SeaderHfTeardownActionStopApp:
scene_manager_stop(seader->scene_manager);
view_dispatcher_stop(seader->view_dispatcher);
return true;
case SeaderHfTeardownActionNone:
default:
return false;
}
}
bool seader_hf_request_teardown(Seader* seader, SeaderHfTeardownAction action) {
furi_assert(seader);
FURI_LOG_I(
TAG,
"HF teardown requested action=%d state=%d worker_state=%d",
action,
seader->hf_session_state,
seader->worker ? seader_worker_get_state(seader->worker) : -1);
seader->hf_teardown_action = action;
if(!seader_hf_has_runtime(seader)) {
seader->hf_session_state = SeaderHfSessionStateUnloaded;
return seader_hf_finish_teardown_action(seader);
}
if(!seader_worker_acquire(seader)) {
return seader_hf_finish_teardown_action(seader);
}
if(seader->hf_session_state == SeaderHfSessionStateTearingDown ||
(seader->worker &&
seader_worker_get_state(seader->worker) == SeaderWorkerStateHfTeardown)) {
return true;
}
seader->hf_session_state = SeaderHfSessionStateTearingDown;
seader_worker_stop(seader->worker);
seader_worker_start(
seader->worker,
SeaderWorkerStateHfTeardown,
seader->uart,
seader_hf_worker_event_callback,
seader);
return true;
}
bool seader_worker_acquire(Seader* seader) {
furi_assert(seader);
if(seader->worker) {
return true;
}
seader->worker = seader_worker_alloc();
return seader->worker != NULL;
}
void seader_worker_release(Seader* seader) {
furi_assert(seader);
if(!seader->worker) {
return;
}
seader_worker_stop(seader->worker);
seader->worker->callback = NULL;
seader->worker->context = NULL;
seader_worker_change_state(seader->worker, SeaderWorkerStateReady);
}
void seader_scratch_reset(Seader* seader) {
furi_assert(seader);
seader->scratch.offset = 0U;
}
void* seader_scratch_alloc(Seader* seader, size_t size, size_t align) {
furi_assert(seader);
furi_assert(seader_align_is_valid(align));
const size_t mask = align - 1U;
const size_t aligned_offset = (seader->scratch.offset + mask) & ~mask;
if(aligned_offset + size > sizeof(seader->scratch.arena)) {
FURI_LOG_E(TAG, "Scratch overflow: need=%zu offset=%zu", size, aligned_offset);
return NULL;
}
void* ptr = &seader->scratch.arena[aligned_offset];
memset(ptr, 0, size);
seader->scratch.offset = aligned_offset + size;
if(seader->scratch.offset > seader->scratch.high_water) {
seader->scratch.high_water = seader->scratch.offset;
}
return ptr;
}
bool seader_hf_mode_activate(Seader* seader) {
furi_assert(seader);
if(seader->hf_mode) {
return true;
}
seader_scratch_reset(seader);
seader->hf_mode =
seader_scratch_alloc(seader, sizeof(SeaderHfModeContext), _Alignof(SeaderHfModeContext));
if(!seader->hf_mode) {
return false;
}
seader->hf_mode->selected_read_type = SeaderCredentialTypeNone;
return true;
}
void seader_hf_mode_deactivate(Seader* seader) {
furi_assert(seader);
seader->hf_mode = NULL;
seader_scratch_reset(seader);
}
SeaderCredentialType seader_hf_mode_get_selected_read_type(const Seader* seader) {
return seader && seader->hf_mode ? seader->hf_mode->selected_read_type : SeaderCredentialTypeNone;
}
void seader_hf_mode_set_selected_read_type(Seader* seader, SeaderCredentialType type) {
if(!seader || !seader->hf_mode) {
FURI_LOG_W(
TAG,
"Ignoring HF selected read type update without mode context seader=%p hf_mode=%p type=%d",
seader,
seader ? seader->hf_mode : NULL,
type);
return;
}
seader->hf_mode->selected_read_type = type;
}
void seader_hf_mode_set_detected_types(
Seader* seader,
const SeaderCredentialType* types,
size_t count) {
if(!seader || !seader->hf_mode) {
FURI_LOG_W(
TAG,
"Ignoring HF detected types update without mode context seader=%p hf_mode=%p count=%zu",
seader,
seader ? seader->hf_mode : NULL,
count);
return;
}
if(count > SEADER_MAX_DETECTED_CARD_TYPES) {
count = SEADER_MAX_DETECTED_CARD_TYPES;
}
memset(seader->hf_mode->detected_card_types, 0, sizeof(seader->hf_mode->detected_card_types));
if(types && count > 0) {
memcpy(seader->hf_mode->detected_card_types, types, count * sizeof(types[0]));
}
seader->hf_mode->detected_card_type_count = count;
}
size_t seader_hf_mode_get_detected_type_count(const Seader* seader) {
return seader && seader->hf_mode ? seader->hf_mode->detected_card_type_count : 0U;
}
const SeaderCredentialType* seader_hf_mode_get_detected_types(const Seader* seader) {
return seader && seader->hf_mode ? seader->hf_mode->detected_card_types : NULL;
}
void seader_hf_mode_clear_detected_types(Seader* seader) {
seader_hf_mode_set_detected_types(seader, NULL, 0U);
}
int32_t seader_app(void* p) {
UNUSED(p);
Seader* seader = seader_alloc();
scene_manager_next_scene(seader->scene_manager, SeaderSceneStart);
view_dispatcher_run(seader->view_dispatcher);
seader_free(seader);
return 0;
}