| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517 |
- #include <gui/gui.h>
- #include <notification/notification.h>
- #include <notification/notification_messages.h>
- #include <totp_icons.h>
- #include "totp_scene_generate_token.h"
- #include "../../../types/token_info.h"
- #include "../../../types/common.h"
- #include "../../constants.h"
- #include "../../../services/totp/totp.h"
- #include "../../../services/config/config.h"
- #include "../../../services/crypto/crypto.h"
- #include "../../../services/convert/convert.h"
- #include "../../../lib/polyfills/memset_s.h"
- #include "../../../lib/roll_value/roll_value.h"
- #include "../../scene_director.h"
- #include "../token_menu/totp_scene_token_menu.h"
- #include "../../../features_config.h"
- #include "../../../workers/usb_type_code/usb_type_code.h"
- #ifdef TOTP_BADBT_TYPE_ENABLED
- #include "../../../workers/bt_type_code/bt_type_code.h"
- #endif
- #include "../../fonts/mode-nine/mode-nine.h"
- #define PROGRESS_BAR_MARGIN (3)
- #define PROGRESS_BAR_HEIGHT (4)
- static const char* STEAM_ALGO_ALPHABET = "23456789BCDFGHJKMNPQRTVWXY";
- typedef struct {
- uint16_t current_token_index;
- char last_code[TOTP_TOKEN_DIGITS_MAX_COUNT + 1];
- bool need_token_update;
- TokenInfo* current_token;
- uint32_t last_token_gen_time;
- TotpUsbTypeCodeWorkerContext* usb_type_code_worker_context;
- NotificationMessage const** notification_sequence_new_token;
- NotificationMessage const** notification_sequence_badusb;
- FuriMutex* last_code_update_sync;
- } SceneState;
- static const NotificationSequence*
- get_notification_sequence_new_token(const PluginState* plugin_state, SceneState* scene_state) {
- if(scene_state->notification_sequence_new_token == NULL) {
- uint8_t i = 0;
- uint8_t length = 4;
- if(plugin_state->notification_method & NotificationMethodVibro) {
- length += 2;
- }
- if(plugin_state->notification_method & NotificationMethodSound) {
- length += 2;
- }
- scene_state->notification_sequence_new_token = malloc(sizeof(void*) * length);
- furi_check(scene_state->notification_sequence_new_token != NULL);
- scene_state->notification_sequence_new_token[i++] = &message_display_backlight_on;
- scene_state->notification_sequence_new_token[i++] = &message_green_255;
- if(plugin_state->notification_method & NotificationMethodVibro) {
- scene_state->notification_sequence_new_token[i++] = &message_vibro_on;
- }
- if(plugin_state->notification_method & NotificationMethodSound) {
- scene_state->notification_sequence_new_token[i++] = &message_note_c5;
- }
- scene_state->notification_sequence_new_token[i++] = &message_delay_50;
- if(plugin_state->notification_method & NotificationMethodVibro) {
- scene_state->notification_sequence_new_token[i++] = &message_vibro_off;
- }
- if(plugin_state->notification_method & NotificationMethodSound) {
- scene_state->notification_sequence_new_token[i++] = &message_sound_off;
- }
- scene_state->notification_sequence_new_token[i++] = NULL;
- }
- return (NotificationSequence*)scene_state->notification_sequence_new_token;
- }
- static const NotificationSequence*
- get_notification_sequence_automation(const PluginState* plugin_state, SceneState* scene_state) {
- if(scene_state->notification_sequence_badusb == NULL) {
- uint8_t i = 0;
- uint8_t length = 3;
- if(plugin_state->notification_method & NotificationMethodVibro) {
- length += 2;
- }
- if(plugin_state->notification_method & NotificationMethodSound) {
- length += 6;
- }
- scene_state->notification_sequence_badusb = malloc(sizeof(void*) * length);
- furi_check(scene_state->notification_sequence_badusb != NULL);
- scene_state->notification_sequence_badusb[i++] = &message_blue_255;
- if(plugin_state->notification_method & NotificationMethodVibro) {
- scene_state->notification_sequence_badusb[i++] = &message_vibro_on;
- }
- if(plugin_state->notification_method & NotificationMethodSound) {
- scene_state->notification_sequence_badusb[i++] = &message_note_d5; //-V525
- scene_state->notification_sequence_badusb[i++] = &message_delay_50;
- scene_state->notification_sequence_badusb[i++] = &message_note_e4;
- scene_state->notification_sequence_badusb[i++] = &message_delay_50;
- scene_state->notification_sequence_badusb[i++] = &message_note_f3;
- }
- scene_state->notification_sequence_badusb[i++] = &message_delay_50;
- if(plugin_state->notification_method & NotificationMethodVibro) {
- scene_state->notification_sequence_badusb[i++] = &message_vibro_off;
- }
- if(plugin_state->notification_method & NotificationMethodSound) {
- scene_state->notification_sequence_badusb[i++] = &message_sound_off;
- }
- scene_state->notification_sequence_badusb[i++] = NULL;
- }
- return (NotificationSequence*)scene_state->notification_sequence_badusb;
- }
- static void
- int_token_to_str(uint64_t i_token_code, char* str, TokenDigitsCount len, TokenHashAlgo algo) {
- if(i_token_code == OTP_ERROR) {
- memset(&str[0], '-', len);
- } else {
- if(algo == STEAM) {
- for(uint8_t i = 0; i < len; i++) {
- str[i] = STEAM_ALGO_ALPHABET[i_token_code % 26];
- i_token_code = i_token_code / 26;
- }
- } else {
- for(int8_t i = len - 1; i >= 0; i--) {
- str[i] = CONVERT_DIGIT_TO_CHAR(i_token_code % 10);
- i_token_code = i_token_code / 10;
- }
- }
- }
- str[len] = '\0';
- }
- static TOTP_ALGO get_totp_algo_impl(TokenHashAlgo algo) {
- switch(algo) {
- case SHA1:
- case STEAM:
- return TOTP_ALGO_SHA1;
- case SHA256:
- return TOTP_ALGO_SHA256;
- case SHA512:
- return TOTP_ALGO_SHA512;
- default:
- break;
- }
- return NULL;
- }
- static void update_totp_params(PluginState* const plugin_state) {
- SceneState* scene_state = (SceneState*)plugin_state->current_scene_state;
- if(scene_state->current_token_index < plugin_state->tokens_count) {
- TokenInfo* tokenInfo =
- list_element_at(plugin_state->tokens_list, scene_state->current_token_index)->data;
- scene_state->need_token_update = true;
- scene_state->current_token = tokenInfo;
- }
- }
- static void draw_totp_code(Canvas* const canvas, const SceneState* const scene_state) {
- uint8_t code_length = scene_state->current_token->digits;
- uint8_t char_width = modeNine_15ptFontInfo.charInfo[0].width;
- uint8_t total_length = code_length * (char_width + modeNine_15ptFontInfo.spacePixels);
- uint8_t offset_x = (SCREEN_WIDTH - total_length) >> 1;
- uint8_t offset_x_inc = char_width + modeNine_15ptFontInfo.spacePixels;
- uint8_t offset_y = SCREEN_HEIGHT_CENTER - (modeNine_15ptFontInfo.height >> 1);
- for(uint8_t i = 0; i < code_length; i++) {
- char ch = scene_state->last_code[i];
- uint8_t char_index = ch - modeNine_15ptFontInfo.startChar;
- canvas_draw_xbm(
- canvas,
- offset_x,
- offset_y,
- char_width,
- modeNine_15ptFontInfo.height,
- &modeNine_15ptFontInfo.data[modeNine_15ptFontInfo.charInfo[char_index].offset]);
- offset_x += offset_x_inc;
- }
- }
- void totp_scene_generate_token_activate(
- PluginState* plugin_state,
- const GenerateTokenSceneContext* context) {
- if(!plugin_state->token_list_loaded) {
- TokenLoadingResult token_load_result = totp_config_file_load_tokens(plugin_state);
- if(token_load_result != TokenLoadingResultSuccess) {
- DialogMessage* message = dialog_message_alloc();
- dialog_message_set_buttons(message, NULL, "Okay", NULL);
- if(token_load_result == TokenLoadingResultWarning) {
- dialog_message_set_text(
- message,
- "Unable to load some tokens\nPlease review conf file",
- SCREEN_WIDTH_CENTER,
- SCREEN_HEIGHT_CENTER,
- AlignCenter,
- AlignCenter);
- } else if(token_load_result == TokenLoadingResultError) {
- dialog_message_set_text(
- message,
- "Unable to load tokens\nPlease review conf file",
- SCREEN_WIDTH_CENTER,
- SCREEN_HEIGHT_CENTER,
- AlignCenter,
- AlignCenter);
- }
- dialog_message_show(plugin_state->dialogs_app, message);
- dialog_message_free(message);
- }
- }
- SceneState* scene_state = malloc(sizeof(SceneState));
- furi_check(scene_state != NULL);
- if(context == NULL || context->current_token_index > plugin_state->tokens_count) {
- scene_state->current_token_index = 0;
- } else {
- scene_state->current_token_index = context->current_token_index;
- }
- scene_state->need_token_update = true;
- plugin_state->current_scene_state = scene_state;
- FURI_LOG_D(LOGGING_TAG, "Timezone set to: %f", (double)plugin_state->timezone_offset);
- update_totp_params(plugin_state);
- scene_state->last_code_update_sync = furi_mutex_alloc(FuriMutexTypeNormal);
- if(plugin_state->automation_method & AutomationMethodBadUsb) {
- scene_state->usb_type_code_worker_context = totp_usb_type_code_worker_start(
- &scene_state->last_code[0],
- TOTP_TOKEN_DIGITS_MAX_COUNT + 1,
- scene_state->last_code_update_sync);
- }
- #ifdef TOTP_BADBT_TYPE_ENABLED
- if(plugin_state->automation_method & AutomationMethodBadBt) {
- if(plugin_state->bt_type_code_worker_context == NULL) {
- plugin_state->bt_type_code_worker_context = totp_bt_type_code_worker_init();
- }
- totp_bt_type_code_worker_start(
- plugin_state->bt_type_code_worker_context,
- &scene_state->last_code[0],
- TOTP_TOKEN_DIGITS_MAX_COUNT + 1,
- scene_state->last_code_update_sync);
- }
- #endif
- }
- void totp_scene_generate_token_render(Canvas* const canvas, PluginState* plugin_state) {
- if(plugin_state->tokens_count == 0) {
- canvas_draw_str_aligned(
- canvas,
- SCREEN_WIDTH_CENTER,
- SCREEN_HEIGHT_CENTER - 10,
- AlignCenter,
- AlignCenter,
- "Token list is empty");
- canvas_draw_str_aligned(
- canvas,
- SCREEN_WIDTH_CENTER,
- SCREEN_HEIGHT_CENTER + 10,
- AlignCenter,
- AlignCenter,
- "Press OK button to add");
- return;
- }
- SceneState* scene_state = (SceneState*)plugin_state->current_scene_state;
- FuriHalRtcDateTime curr_dt;
- furi_hal_rtc_get_datetime(&curr_dt);
- uint32_t curr_ts = furi_hal_rtc_datetime_to_timestamp(&curr_dt);
- bool is_new_token_time = curr_ts % scene_state->current_token->duration == 0;
- if(is_new_token_time && scene_state->last_token_gen_time != curr_ts) {
- scene_state->need_token_update = true;
- }
- if(scene_state->need_token_update) {
- scene_state->need_token_update = false;
- scene_state->last_token_gen_time = curr_ts;
- const TokenInfo* tokenInfo = scene_state->current_token;
- if(tokenInfo->token != NULL && tokenInfo->token_length > 0) {
- furi_mutex_acquire(scene_state->last_code_update_sync, FuriWaitForever);
- size_t key_length;
- uint8_t* key = totp_crypto_decrypt(
- tokenInfo->token, tokenInfo->token_length, &plugin_state->iv[0], &key_length);
- int_token_to_str(
- totp_at(
- get_totp_algo_impl(tokenInfo->algo),
- key,
- key_length,
- curr_ts,
- plugin_state->timezone_offset,
- tokenInfo->duration),
- scene_state->last_code,
- tokenInfo->digits,
- tokenInfo->algo);
- memset_s(key, key_length, 0, key_length);
- free(key);
- } else {
- furi_mutex_acquire(scene_state->last_code_update_sync, FuriWaitForever);
- int_token_to_str(0, scene_state->last_code, tokenInfo->digits, tokenInfo->algo);
- }
- furi_mutex_release(scene_state->last_code_update_sync);
- if(is_new_token_time) {
- notification_message(
- plugin_state->notification_app,
- get_notification_sequence_new_token(plugin_state, scene_state));
- }
- }
- canvas_set_font(canvas, FontPrimary);
- uint16_t token_name_width = canvas_string_width(canvas, scene_state->current_token->name);
- if(SCREEN_WIDTH - token_name_width > 18) {
- canvas_draw_str_aligned(
- canvas,
- SCREEN_WIDTH_CENTER,
- SCREEN_HEIGHT_CENTER - 20,
- AlignCenter,
- AlignCenter,
- scene_state->current_token->name);
- } else {
- canvas_draw_str_aligned(
- canvas,
- 9,
- SCREEN_HEIGHT_CENTER - 20,
- AlignLeft,
- AlignCenter,
- scene_state->current_token->name);
- canvas_set_color(canvas, ColorWhite);
- canvas_draw_box(canvas, 0, SCREEN_HEIGHT_CENTER - 24, 9, 9);
- canvas_draw_box(canvas, SCREEN_WIDTH - 10, SCREEN_HEIGHT_CENTER - 24, 9, 9);
- canvas_set_color(canvas, ColorBlack);
- }
- draw_totp_code(canvas, scene_state);
- const uint8_t TOKEN_LIFETIME = scene_state->current_token->duration;
- float percentDone = (float)(TOKEN_LIFETIME - curr_ts % TOKEN_LIFETIME) / (float)TOKEN_LIFETIME;
- uint8_t barWidth = (uint8_t)((float)(SCREEN_WIDTH - (PROGRESS_BAR_MARGIN << 1)) * percentDone);
- uint8_t barX =
- ((SCREEN_WIDTH - (PROGRESS_BAR_MARGIN << 1) - barWidth) >> 1) + PROGRESS_BAR_MARGIN;
- canvas_draw_box(
- canvas,
- barX,
- SCREEN_HEIGHT - PROGRESS_BAR_MARGIN - PROGRESS_BAR_HEIGHT,
- barWidth,
- PROGRESS_BAR_HEIGHT);
- if(plugin_state->tokens_count > 1) {
- canvas_draw_icon(canvas, 0, SCREEN_HEIGHT_CENTER - 24, &I_totp_arrow_left_8x9);
- canvas_draw_icon(
- canvas, SCREEN_WIDTH - 9, SCREEN_HEIGHT_CENTER - 24, &I_totp_arrow_right_8x9);
- }
- #ifdef TOTP_AUTOMATION_ICONS_ENABLED
- if(plugin_state->automation_method & AutomationMethodBadUsb) {
- canvas_draw_icon(
- canvas,
- #ifdef TOTP_BADBT_TYPE_ENABLED
- SCREEN_WIDTH_CENTER -
- (plugin_state->automation_method & AutomationMethodBadBt ? 33 : 15),
- #else
- SCREEN_WIDTH_CENTER - 15,
- #endif
- SCREEN_HEIGHT_CENTER + 12,
- &I_hid_usb_31x9);
- }
- #ifdef TOTP_BADBT_TYPE_ENABLED
- if(plugin_state->automation_method & AutomationMethodBadBt &&
- plugin_state->bt_type_code_worker_context != NULL &&
- plugin_state->bt_type_code_worker_context->is_advertising) {
- canvas_draw_icon(
- canvas,
- SCREEN_WIDTH_CENTER +
- (plugin_state->automation_method & AutomationMethodBadUsb ? 2 : -15),
- SCREEN_HEIGHT_CENTER + 12,
- &I_hid_ble_31x9);
- }
- #endif
- #endif
- }
- bool totp_scene_generate_token_handle_event(
- const PluginEvent* const event,
- PluginState* plugin_state) {
- if(event->type != EventTypeKey) {
- return true;
- }
- if(event->input.type == InputTypeLong && event->input.key == InputKeyBack) {
- return false;
- }
- SceneState* scene_state;
- if(event->input.type == InputTypeLong) {
- if(event->input.key == InputKeyDown &&
- plugin_state->automation_method & AutomationMethodBadUsb) {
- scene_state = (SceneState*)plugin_state->current_scene_state;
- totp_usb_type_code_worker_notify(
- scene_state->usb_type_code_worker_context,
- TotpUsbTypeCodeWorkerEventType,
- scene_state->current_token->automation_features);
- notification_message(
- plugin_state->notification_app,
- get_notification_sequence_automation(plugin_state, scene_state));
- return true;
- }
- #ifdef TOTP_BADBT_TYPE_ENABLED
- else if(
- event->input.key == InputKeyUp &&
- plugin_state->automation_method & AutomationMethodBadBt) {
- scene_state = (SceneState*)plugin_state->current_scene_state;
- totp_bt_type_code_worker_notify(
- plugin_state->bt_type_code_worker_context,
- TotpBtTypeCodeWorkerEventType,
- scene_state->current_token->automation_features);
- notification_message(
- plugin_state->notification_app,
- get_notification_sequence_automation(plugin_state, scene_state));
- return true;
- }
- #endif
- }
- if(event->input.type != InputTypePress && event->input.type != InputTypeRepeat) {
- return true;
- }
- scene_state = (SceneState*)plugin_state->current_scene_state;
- switch(event->input.key) {
- case InputKeyUp:
- break;
- case InputKeyDown:
- break;
- case InputKeyRight:
- totp_roll_value_uint16_t(
- &scene_state->current_token_index,
- 1,
- 0,
- plugin_state->tokens_count - 1,
- RollOverflowBehaviorRoll);
- update_totp_params(plugin_state);
- break;
- case InputKeyLeft:
- totp_roll_value_uint16_t(
- &scene_state->current_token_index,
- -1,
- 0,
- plugin_state->tokens_count - 1,
- RollOverflowBehaviorRoll);
- update_totp_params(plugin_state);
- break;
- case InputKeyOk:
- if(plugin_state->tokens_count == 0) {
- totp_scene_director_activate_scene(plugin_state, TotpSceneTokenMenu, NULL);
- } else {
- TokenMenuSceneContext ctx = {.current_token_index = scene_state->current_token_index};
- totp_scene_director_activate_scene(plugin_state, TotpSceneTokenMenu, &ctx);
- }
- break;
- case InputKeyBack:
- break;
- default:
- break;
- }
- return true;
- }
- void totp_scene_generate_token_deactivate(PluginState* plugin_state) {
- if(plugin_state->current_scene_state == NULL) return;
- SceneState* scene_state = (SceneState*)plugin_state->current_scene_state;
- if(plugin_state->automation_method & AutomationMethodBadUsb) {
- totp_usb_type_code_worker_stop(scene_state->usb_type_code_worker_context);
- }
- #ifdef TOTP_BADBT_TYPE_ENABLED
- if(plugin_state->automation_method & AutomationMethodBadBt) {
- totp_bt_type_code_worker_stop(plugin_state->bt_type_code_worker_context);
- }
- #endif
- if(scene_state->notification_sequence_new_token != NULL) {
- free(scene_state->notification_sequence_new_token);
- }
- if(scene_state->notification_sequence_badusb != NULL) {
- free(scene_state->notification_sequence_badusb);
- }
- furi_mutex_free(scene_state->last_code_update_sync);
- free(scene_state);
- plugin_state->current_scene_state = NULL;
- }
|