weebo.c 13 KB

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