weebo.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366
  1. #include "weebo_i.h"
  2. #define TAG "weebo"
  3. #define WEEBO_KEY_RETAIL_FILENAME "key_retail"
  4. bool weebo_load_key_retail(Weebo* weebo) {
  5. FuriString* path = furi_string_alloc();
  6. bool parsed = false;
  7. uint8_t buffer[160];
  8. memset(buffer, 0, sizeof(buffer));
  9. Storage* storage = furi_record_open(RECORD_STORAGE);
  10. Stream* stream = file_stream_alloc(storage);
  11. do {
  12. furi_string_printf(
  13. path, "%s/%s%s", STORAGE_APP_DATA_PATH_PREFIX, WEEBO_KEY_RETAIL_FILENAME, ".bin");
  14. bool opened =
  15. file_stream_open(stream, furi_string_get_cstr(path), FSAM_READ, FSOM_OPEN_EXISTING);
  16. if(!opened) {
  17. FURI_LOG_E(TAG, "Failed to open file");
  18. break;
  19. }
  20. size_t bytes_read = stream_read(stream, buffer, sizeof(buffer));
  21. if(bytes_read != sizeof(buffer)) {
  22. FURI_LOG_E(TAG, "Insufficient data");
  23. break;
  24. }
  25. memcpy(&weebo->amiiboKeys, buffer, bytes_read);
  26. // TODO: compare SHA1
  27. parsed = true;
  28. } while(false);
  29. file_stream_close(stream);
  30. furi_record_close(RECORD_STORAGE);
  31. furi_string_free(path);
  32. return parsed;
  33. }
  34. bool weebo_load_figure(Weebo* weebo, FuriString* path, bool show_dialog) {
  35. bool parsed = false;
  36. FuriString* reason = furi_string_alloc_set("Couldn't load file");
  37. uint8_t buffer[NTAG215_SIZE];
  38. memset(buffer, 0, sizeof(buffer));
  39. if(weebo->loading_cb) {
  40. weebo->loading_cb(weebo->loading_cb_ctx, true);
  41. }
  42. do {
  43. NfcDevice* nfc_device = weebo->nfc_device;
  44. if(!nfc_device_load(nfc_device, furi_string_get_cstr(path))) break;
  45. NfcProtocol protocol = nfc_device_get_protocol(nfc_device);
  46. if(protocol != NfcProtocolMfUltralight) {
  47. furi_string_printf(reason, "Not Ultralight protocol");
  48. break;
  49. }
  50. const MfUltralightData* data = nfc_device_get_data(nfc_device, NfcProtocolMfUltralight);
  51. if(data->type != MfUltralightTypeNTAG215) {
  52. furi_string_printf(reason, "Not NTAG215");
  53. break;
  54. }
  55. if(!mf_ultralight_is_all_data_read(data)) {
  56. furi_string_printf(reason, "Incomplete data");
  57. break;
  58. }
  59. uint8_t* uid = data->iso14443_3a_data->uid;
  60. uint8_t pwd[4];
  61. pwd[0] = uid[1] ^ uid[3] ^ 0xAA;
  62. pwd[1] = uid[2] ^ uid[4] ^ 0x55;
  63. pwd[2] = uid[3] ^ uid[5] ^ 0xAA;
  64. pwd[3] = uid[4] ^ uid[6] ^ 0x55;
  65. if(memcmp(data->page[133].data, pwd, sizeof(pwd)) != 0) {
  66. furi_string_printf(reason, "Wrong password");
  67. break;
  68. }
  69. for(size_t i = 0; i < 135; i++) {
  70. memcpy(
  71. buffer + i * MF_ULTRALIGHT_PAGE_SIZE, data->page[i].data, MF_ULTRALIGHT_PAGE_SIZE);
  72. }
  73. if(!nfc3d_amiibo_unpack(&weebo->amiiboKeys, buffer, weebo->figure)) {
  74. FURI_LOG_E(TAG, "Failed to unpack");
  75. break;
  76. }
  77. parsed = true;
  78. } while(false);
  79. if(weebo->loading_cb) {
  80. weebo->loading_cb(weebo->loading_cb_ctx, false);
  81. }
  82. if((!parsed) && (show_dialog)) {
  83. dialog_message_show_storage_error(weebo->dialogs, furi_string_get_cstr(reason));
  84. }
  85. furi_string_free(reason);
  86. return parsed;
  87. }
  88. void weebo_set_loading_callback(Weebo* weebo, WeeboLoadingCallback callback, void* context) {
  89. furi_assert(weebo);
  90. weebo->loading_cb = callback;
  91. weebo->loading_cb_ctx = context;
  92. }
  93. bool weebo_file_select(Weebo* weebo) {
  94. furi_assert(weebo);
  95. bool res = false;
  96. FuriString* weebo_app_folder;
  97. if(storage_dir_exists(weebo->storage, "/ext/nfc/Amiibo")) {
  98. weebo_app_folder = furi_string_alloc_set("/ext/nfc/Amiibo");
  99. } else if(storage_dir_exists(weebo->storage, "/ext/nfc/Amiibos")) {
  100. weebo_app_folder = furi_string_alloc_set("/ext/nfc/Amiibos");
  101. } else if(storage_dir_exists(weebo->storage, "/ext/nfc/amiibo")) {
  102. weebo_app_folder = furi_string_alloc_set("/ext/nfc/amiibo");
  103. } else if(storage_dir_exists(weebo->storage, "/ext/nfc/amiibos")) {
  104. weebo_app_folder = furi_string_alloc_set("/ext/nfc/amiibos");
  105. } else {
  106. weebo_app_folder = furi_string_alloc_set("/ext/nfc");
  107. }
  108. DialogsFileBrowserOptions browser_options;
  109. dialog_file_browser_set_basic_options(&browser_options, ".nfc", &I_Nfc_10px);
  110. browser_options.base_path = STORAGE_APP_DATA_PATH_PREFIX;
  111. res = dialog_file_browser_show(
  112. weebo->dialogs, weebo->load_path, weebo_app_folder, &browser_options);
  113. furi_string_free(weebo_app_folder);
  114. if(res) {
  115. FuriString* filename;
  116. filename = furi_string_alloc();
  117. path_extract_filename(weebo->load_path, filename, true);
  118. strncpy(weebo->file_name, furi_string_get_cstr(filename), WEEBO_FILE_NAME_MAX_LENGTH);
  119. res = weebo_load_figure(weebo, weebo->load_path, true);
  120. furi_string_free(filename);
  121. }
  122. return res;
  123. }
  124. bool weebo_custom_event_callback(void* context, uint32_t event) {
  125. furi_assert(context);
  126. Weebo* weebo = context;
  127. return scene_manager_handle_custom_event(weebo->scene_manager, event);
  128. }
  129. bool weebo_back_event_callback(void* context) {
  130. furi_assert(context);
  131. Weebo* weebo = context;
  132. return scene_manager_handle_back_event(weebo->scene_manager);
  133. }
  134. void weebo_tick_event_callback(void* context) {
  135. furi_assert(context);
  136. Weebo* weebo = context;
  137. scene_manager_handle_tick_event(weebo->scene_manager);
  138. }
  139. Weebo* weebo_alloc() {
  140. Weebo* weebo = malloc(sizeof(Weebo));
  141. weebo->view_dispatcher = view_dispatcher_alloc();
  142. weebo->scene_manager = scene_manager_alloc(&weebo_scene_handlers, weebo);
  143. view_dispatcher_set_event_callback_context(weebo->view_dispatcher, weebo);
  144. view_dispatcher_set_custom_event_callback(weebo->view_dispatcher, weebo_custom_event_callback);
  145. view_dispatcher_set_navigation_event_callback(
  146. weebo->view_dispatcher, weebo_back_event_callback);
  147. view_dispatcher_set_tick_event_callback(
  148. weebo->view_dispatcher, weebo_tick_event_callback, 100);
  149. weebo->nfc = nfc_alloc();
  150. // Nfc device
  151. weebo->nfc_device = nfc_device_alloc();
  152. nfc_device_set_loading_callback(weebo->nfc_device, weebo_show_loading_popup, weebo);
  153. // Open GUI record
  154. weebo->gui = furi_record_open(RECORD_GUI);
  155. view_dispatcher_attach_to_gui(
  156. weebo->view_dispatcher, weebo->gui, ViewDispatcherTypeFullscreen);
  157. // Open Notification record
  158. weebo->notifications = furi_record_open(RECORD_NOTIFICATION);
  159. // Submenu
  160. weebo->submenu = submenu_alloc();
  161. view_dispatcher_add_view(
  162. weebo->view_dispatcher, WeeboViewMenu, submenu_get_view(weebo->submenu));
  163. // Popup
  164. weebo->popup = popup_alloc();
  165. view_dispatcher_add_view(weebo->view_dispatcher, WeeboViewPopup, popup_get_view(weebo->popup));
  166. // Loading
  167. weebo->loading = loading_alloc();
  168. view_dispatcher_add_view(
  169. weebo->view_dispatcher, WeeboViewLoading, loading_get_view(weebo->loading));
  170. // Text Input
  171. weebo->text_input = text_input_alloc();
  172. view_dispatcher_add_view(
  173. weebo->view_dispatcher, WeeboViewTextInput, text_input_get_view(weebo->text_input));
  174. // Number Input
  175. weebo->number_input = number_input_alloc();
  176. view_dispatcher_add_view(
  177. weebo->view_dispatcher, WeeboViewNumberInput, number_input_get_view(weebo->number_input));
  178. // TextBox
  179. weebo->text_box = text_box_alloc();
  180. view_dispatcher_add_view(
  181. weebo->view_dispatcher, WeeboViewTextBox, text_box_get_view(weebo->text_box));
  182. weebo->text_box_store = furi_string_alloc();
  183. // Custom Widget
  184. weebo->widget = widget_alloc();
  185. view_dispatcher_add_view(
  186. weebo->view_dispatcher, WeeboViewWidget, widget_get_view(weebo->widget));
  187. weebo->storage = furi_record_open(RECORD_STORAGE);
  188. weebo->dialogs = furi_record_open(RECORD_DIALOGS);
  189. weebo->load_path = furi_string_alloc();
  190. weebo->keys_loaded = false;
  191. return weebo;
  192. }
  193. void weebo_free(Weebo* weebo) {
  194. furi_assert(weebo);
  195. nfc_free(weebo->nfc);
  196. // Nfc device
  197. nfc_device_free(weebo->nfc_device);
  198. // Submenu
  199. view_dispatcher_remove_view(weebo->view_dispatcher, WeeboViewMenu);
  200. submenu_free(weebo->submenu);
  201. // Popup
  202. view_dispatcher_remove_view(weebo->view_dispatcher, WeeboViewPopup);
  203. popup_free(weebo->popup);
  204. // Loading
  205. view_dispatcher_remove_view(weebo->view_dispatcher, WeeboViewLoading);
  206. loading_free(weebo->loading);
  207. // TextInput
  208. view_dispatcher_remove_view(weebo->view_dispatcher, WeeboViewTextInput);
  209. text_input_free(weebo->text_input);
  210. // NumberInput
  211. view_dispatcher_remove_view(weebo->view_dispatcher, WeeboViewNumberInput);
  212. number_input_free(weebo->number_input);
  213. // TextBox
  214. view_dispatcher_remove_view(weebo->view_dispatcher, WeeboViewTextBox);
  215. text_box_free(weebo->text_box);
  216. furi_string_free(weebo->text_box_store);
  217. // Custom Widget
  218. view_dispatcher_remove_view(weebo->view_dispatcher, WeeboViewWidget);
  219. widget_free(weebo->widget);
  220. // View Dispatcher
  221. view_dispatcher_free(weebo->view_dispatcher);
  222. // Scene Manager
  223. scene_manager_free(weebo->scene_manager);
  224. // GUI
  225. furi_record_close(RECORD_GUI);
  226. weebo->gui = NULL;
  227. // Notifications
  228. furi_record_close(RECORD_NOTIFICATION);
  229. weebo->notifications = NULL;
  230. furi_string_free(weebo->load_path);
  231. furi_record_close(RECORD_STORAGE);
  232. furi_record_close(RECORD_DIALOGS);
  233. free(weebo);
  234. }
  235. void weebo_text_store_set(Weebo* weebo, const char* text, ...) {
  236. va_list args;
  237. va_start(args, text);
  238. vsnprintf(weebo->text_store, sizeof(weebo->text_store), text, args);
  239. va_end(args);
  240. }
  241. void weebo_text_store_clear(Weebo* weebo) {
  242. memset(weebo->text_store, 0, sizeof(weebo->text_store));
  243. }
  244. static const NotificationSequence weebo_sequence_blink_start_blue = {
  245. &message_blink_start_10,
  246. &message_blink_set_color_blue,
  247. &message_do_not_reset,
  248. NULL,
  249. };
  250. static const NotificationSequence weebo_sequence_blink_stop = {
  251. &message_blink_stop,
  252. NULL,
  253. };
  254. void weebo_blink_start(Weebo* weebo) {
  255. notification_message(weebo->notifications, &weebo_sequence_blink_start_blue);
  256. }
  257. void weebo_blink_stop(Weebo* weebo) {
  258. notification_message(weebo->notifications, &weebo_sequence_blink_stop);
  259. }
  260. void weebo_show_loading_popup(void* context, bool show) {
  261. Weebo* weebo = context;
  262. if(show) {
  263. // Raise timer priority so that animations can play
  264. furi_timer_set_thread_priority(FuriTimerThreadPriorityElevated);
  265. view_dispatcher_switch_to_view(weebo->view_dispatcher, WeeboViewLoading);
  266. } else {
  267. // Restore default timer priority
  268. furi_timer_set_thread_priority(FuriTimerThreadPriorityNormal);
  269. }
  270. }
  271. int32_t weebo_app(void* p) {
  272. UNUSED(p);
  273. Weebo* weebo = weebo_alloc();
  274. weebo->keys_loaded = weebo_load_key_retail(weebo);
  275. if(weebo->keys_loaded) {
  276. scene_manager_next_scene(weebo->scene_manager, WeeboSceneMainMenu);
  277. } else {
  278. scene_manager_next_scene(weebo->scene_manager, WeeboSceneKeysMissing);
  279. }
  280. view_dispatcher_run(weebo->view_dispatcher);
  281. weebo_free(weebo);
  282. return 0;
  283. }