picopass.c 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230
  1. #include "picopass_i.h"
  2. #define TAG "PicoPass"
  3. bool picopass_custom_event_callback(void* context, uint32_t event) {
  4. furi_assert(context);
  5. Picopass* picopass = context;
  6. return scene_manager_handle_custom_event(picopass->scene_manager, event);
  7. }
  8. bool picopass_back_event_callback(void* context) {
  9. furi_assert(context);
  10. Picopass* picopass = context;
  11. return scene_manager_handle_back_event(picopass->scene_manager);
  12. }
  13. void picopass_tick_event_callback(void* context) {
  14. furi_assert(context);
  15. Picopass* picopass = context;
  16. scene_manager_handle_tick_event(picopass->scene_manager);
  17. }
  18. Picopass* picopass_alloc() {
  19. Picopass* picopass = malloc(sizeof(Picopass));
  20. picopass->worker = picopass_worker_alloc();
  21. picopass->view_dispatcher = view_dispatcher_alloc();
  22. picopass->scene_manager = scene_manager_alloc(&picopass_scene_handlers, picopass);
  23. view_dispatcher_enable_queue(picopass->view_dispatcher);
  24. view_dispatcher_set_event_callback_context(picopass->view_dispatcher, picopass);
  25. view_dispatcher_set_custom_event_callback(
  26. picopass->view_dispatcher, picopass_custom_event_callback);
  27. view_dispatcher_set_navigation_event_callback(
  28. picopass->view_dispatcher, picopass_back_event_callback);
  29. view_dispatcher_set_tick_event_callback(
  30. picopass->view_dispatcher, picopass_tick_event_callback, 100);
  31. // Picopass device
  32. picopass->dev = picopass_device_alloc();
  33. // Open GUI record
  34. picopass->gui = furi_record_open(RECORD_GUI);
  35. view_dispatcher_attach_to_gui(
  36. picopass->view_dispatcher, picopass->gui, ViewDispatcherTypeFullscreen);
  37. // Open Notification record
  38. picopass->notifications = furi_record_open(RECORD_NOTIFICATION);
  39. // Submenu
  40. picopass->submenu = submenu_alloc();
  41. view_dispatcher_add_view(
  42. picopass->view_dispatcher, PicopassViewMenu, submenu_get_view(picopass->submenu));
  43. // Popup
  44. picopass->popup = popup_alloc();
  45. view_dispatcher_add_view(
  46. picopass->view_dispatcher, PicopassViewPopup, popup_get_view(picopass->popup));
  47. // Loading
  48. picopass->loading = loading_alloc();
  49. view_dispatcher_add_view(
  50. picopass->view_dispatcher, PicopassViewLoading, loading_get_view(picopass->loading));
  51. // Text Input
  52. picopass->text_input = text_input_alloc();
  53. view_dispatcher_add_view(
  54. picopass->view_dispatcher,
  55. PicopassViewTextInput,
  56. text_input_get_view(picopass->text_input));
  57. // Custom Widget
  58. picopass->widget = widget_alloc();
  59. view_dispatcher_add_view(
  60. picopass->view_dispatcher, PicopassViewWidget, widget_get_view(picopass->widget));
  61. picopass->dict_attack = dict_attack_alloc();
  62. view_dispatcher_add_view(
  63. picopass->view_dispatcher,
  64. PicopassViewDictAttack,
  65. dict_attack_get_view(picopass->dict_attack));
  66. picopass->loclass = loclass_alloc();
  67. view_dispatcher_add_view(
  68. picopass->view_dispatcher, PicopassViewLoclass, loclass_get_view(picopass->loclass));
  69. return picopass;
  70. }
  71. void picopass_free(Picopass* picopass) {
  72. furi_assert(picopass);
  73. // Picopass device
  74. picopass_device_free(picopass->dev);
  75. picopass->dev = NULL;
  76. // Submenu
  77. view_dispatcher_remove_view(picopass->view_dispatcher, PicopassViewMenu);
  78. submenu_free(picopass->submenu);
  79. // Popup
  80. view_dispatcher_remove_view(picopass->view_dispatcher, PicopassViewPopup);
  81. popup_free(picopass->popup);
  82. // Loading
  83. view_dispatcher_remove_view(picopass->view_dispatcher, PicopassViewLoading);
  84. loading_free(picopass->loading);
  85. // TextInput
  86. view_dispatcher_remove_view(picopass->view_dispatcher, PicopassViewTextInput);
  87. text_input_free(picopass->text_input);
  88. // Custom Widget
  89. view_dispatcher_remove_view(picopass->view_dispatcher, PicopassViewWidget);
  90. widget_free(picopass->widget);
  91. view_dispatcher_remove_view(picopass->view_dispatcher, PicopassViewDictAttack);
  92. dict_attack_free(picopass->dict_attack);
  93. view_dispatcher_remove_view(picopass->view_dispatcher, PicopassViewLoclass);
  94. loclass_free(picopass->loclass);
  95. // Worker
  96. picopass_worker_stop(picopass->worker);
  97. picopass_worker_free(picopass->worker);
  98. // View Dispatcher
  99. view_dispatcher_free(picopass->view_dispatcher);
  100. // Scene Manager
  101. scene_manager_free(picopass->scene_manager);
  102. // GUI
  103. furi_record_close(RECORD_GUI);
  104. picopass->gui = NULL;
  105. // Notifications
  106. furi_record_close(RECORD_NOTIFICATION);
  107. picopass->notifications = NULL;
  108. free(picopass);
  109. }
  110. void picopass_text_store_set(Picopass* picopass, const char* text, ...) {
  111. va_list args;
  112. va_start(args, text);
  113. vsnprintf(picopass->text_store, sizeof(picopass->text_store), text, args);
  114. va_end(args);
  115. }
  116. void picopass_text_store_clear(Picopass* picopass) {
  117. memset(picopass->text_store, 0, sizeof(picopass->text_store));
  118. }
  119. static const NotificationSequence picopass_sequence_blink_start_cyan = {
  120. &message_blink_start_10,
  121. &message_blink_set_color_cyan,
  122. &message_do_not_reset,
  123. NULL,
  124. };
  125. static const NotificationSequence picopass_sequence_blink_start_magenta = {
  126. &message_blink_start_10,
  127. &message_blink_set_color_magenta,
  128. &message_do_not_reset,
  129. NULL,
  130. };
  131. static const NotificationSequence picopass_sequence_blink_stop = {
  132. &message_blink_stop,
  133. NULL,
  134. };
  135. void picopass_blink_start(Picopass* picopass) {
  136. notification_message(picopass->notifications, &picopass_sequence_blink_start_cyan);
  137. }
  138. void picopass_blink_emulate_start(Picopass* picopass) {
  139. notification_message(picopass->notifications, &picopass_sequence_blink_start_magenta);
  140. }
  141. void picopass_blink_stop(Picopass* picopass) {
  142. notification_message(picopass->notifications, &picopass_sequence_blink_stop);
  143. }
  144. void picopass_show_loading_popup(void* context, bool show) {
  145. Picopass* picopass = context;
  146. TaskHandle_t timer_task = xTaskGetHandle(configTIMER_SERVICE_TASK_NAME);
  147. if(show) {
  148. // Raise timer priority so that animations can play
  149. vTaskPrioritySet(timer_task, configMAX_PRIORITIES - 1);
  150. view_dispatcher_switch_to_view(picopass->view_dispatcher, PicopassViewLoading);
  151. } else {
  152. // Restore default timer priority
  153. vTaskPrioritySet(timer_task, configTIMER_TASK_PRIORITY);
  154. }
  155. }
  156. static void picopass_migrate_from_old_folder() {
  157. Storage* storage = furi_record_open(RECORD_STORAGE);
  158. storage_common_migrate(storage, "/ext/picopass", STORAGE_APP_DATA_PATH_PREFIX);
  159. furi_record_close(RECORD_STORAGE);
  160. }
  161. bool picopass_is_memset(const uint8_t* data, const uint8_t pattern, size_t size) {
  162. bool result = size > 0;
  163. while(size > 0) {
  164. result &= (*data == pattern);
  165. data++;
  166. size--;
  167. }
  168. return result;
  169. }
  170. int32_t picopass_app(void* p) {
  171. UNUSED(p);
  172. picopass_migrate_from_old_folder();
  173. Picopass* picopass = picopass_alloc();
  174. scene_manager_next_scene(picopass->scene_manager, PicopassSceneStart);
  175. view_dispatcher_run(picopass->view_dispatcher);
  176. picopass_free(picopass);
  177. return 0;
  178. }