fap_loader_app.c 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. #include <furi.h>
  2. #include <gui/gui.h>
  3. #include <gui/view_dispatcher.h>
  4. #include <gui/modules/loading.h>
  5. #include <storage/storage.h>
  6. #include <dialogs/dialogs.h>
  7. #include "elf_cpp/elf_hashtable.h"
  8. #include <flipper_application/flipper_application.h>
  9. #define TAG "fap_loader_app"
  10. typedef struct {
  11. FlipperApplication* app;
  12. Storage* storage;
  13. DialogsApp* dialogs;
  14. Gui* gui;
  15. string_t fap_path;
  16. } FapLoader;
  17. static bool
  18. fap_loader_item_callback(string_t path, void* context, uint8_t** icon_ptr, string_t item_name) {
  19. FapLoader* loader = context;
  20. furi_assert(loader);
  21. FlipperApplication* app = flipper_application_alloc(loader->storage, &hashtable_api_interface);
  22. FlipperApplicationPreloadStatus preload_res =
  23. flipper_application_preload(app, string_get_cstr(path));
  24. bool load_success = false;
  25. if(preload_res == FlipperApplicationPreloadStatusSuccess) {
  26. const FlipperApplicationManifest* manifest = flipper_application_get_manifest(app);
  27. if(manifest->has_icon) {
  28. memcpy(*icon_ptr, manifest->icon, FAP_MANIFEST_MAX_ICON_SIZE);
  29. }
  30. string_set_str(item_name, manifest->name);
  31. load_success = true;
  32. } else {
  33. FURI_LOG_E(TAG, "FAP Loader failed to preload %s", string_get_cstr(path));
  34. load_success = false;
  35. }
  36. flipper_application_free(app);
  37. return load_success;
  38. }
  39. static bool fap_loader_run_selected_app(FapLoader* loader) {
  40. furi_assert(loader);
  41. string_t error_message;
  42. string_init_set(error_message, "unknown error");
  43. bool file_selected = false;
  44. bool show_error = true;
  45. do {
  46. file_selected = true;
  47. loader->app = flipper_application_alloc(loader->storage, &hashtable_api_interface);
  48. FURI_LOG_I(TAG, "FAP Loader is loading %s", string_get_cstr(loader->fap_path));
  49. FlipperApplicationPreloadStatus preload_res =
  50. flipper_application_preload(loader->app, string_get_cstr(loader->fap_path));
  51. if(preload_res != FlipperApplicationPreloadStatusSuccess) {
  52. const char* err_msg = flipper_application_preload_status_to_string(preload_res);
  53. string_printf(error_message, "Preload failed: %s", err_msg);
  54. FURI_LOG_E(
  55. TAG,
  56. "FAP Loader failed to preload %s: %s",
  57. string_get_cstr(loader->fap_path),
  58. err_msg);
  59. break;
  60. }
  61. FURI_LOG_I(TAG, "FAP Loader is mapping");
  62. FlipperApplicationLoadStatus load_status = flipper_application_map_to_memory(loader->app);
  63. if(load_status != FlipperApplicationLoadStatusSuccess) {
  64. const char* err_msg = flipper_application_load_status_to_string(load_status);
  65. string_printf(error_message, "Load failed: %s", err_msg);
  66. FURI_LOG_E(
  67. TAG,
  68. "FAP Loader failed to map to memory %s: %s",
  69. string_get_cstr(loader->fap_path),
  70. err_msg);
  71. break;
  72. }
  73. FURI_LOG_I(TAG, "FAP Loader is staring app");
  74. FuriThread* thread = flipper_application_spawn(loader->app, NULL);
  75. furi_thread_start(thread);
  76. furi_thread_join(thread);
  77. show_error = false;
  78. int ret = furi_thread_get_return_code(thread);
  79. FURI_LOG_I(TAG, "FAP app returned: %i", ret);
  80. } while(0);
  81. if(show_error) {
  82. DialogMessage* message = dialog_message_alloc();
  83. dialog_message_set_header(message, "Error", 64, 0, AlignCenter, AlignTop);
  84. dialog_message_set_buttons(message, NULL, NULL, NULL);
  85. string_t buffer;
  86. string_init(buffer);
  87. string_printf(buffer, "%s", string_get_cstr(error_message));
  88. string_replace_str(buffer, ":", "\n");
  89. dialog_message_set_text(
  90. message, string_get_cstr(buffer), 64, 32, AlignCenter, AlignCenter);
  91. dialog_message_show(loader->dialogs, message);
  92. dialog_message_free(message);
  93. string_clear(buffer);
  94. }
  95. string_clear(error_message);
  96. if(file_selected) {
  97. flipper_application_free(loader->app);
  98. }
  99. return file_selected;
  100. }
  101. static bool fap_loader_select_app(FapLoader* loader) {
  102. const DialogsFileBrowserOptions browser_options = {
  103. .extension = ".fap",
  104. .skip_assets = true,
  105. .icon = &I_badusb_10px,
  106. .hide_ext = true,
  107. .item_loader_callback = fap_loader_item_callback,
  108. .item_loader_context = loader,
  109. };
  110. return dialog_file_browser_show(
  111. loader->dialogs, loader->fap_path, loader->fap_path, &browser_options);
  112. }
  113. int32_t fap_loader_app(void* p) {
  114. FapLoader* loader = malloc(sizeof(FapLoader));
  115. loader->storage = furi_record_open(RECORD_STORAGE);
  116. loader->dialogs = furi_record_open(RECORD_DIALOGS);
  117. loader->gui = furi_record_open(RECORD_GUI);
  118. ViewDispatcher* view_dispatcher = view_dispatcher_alloc();
  119. Loading* loading = loading_alloc();
  120. view_dispatcher_enable_queue(view_dispatcher);
  121. view_dispatcher_attach_to_gui(view_dispatcher, loader->gui, ViewDispatcherTypeFullscreen);
  122. view_dispatcher_add_view(view_dispatcher, 0, loading_get_view(loading));
  123. if(p) {
  124. string_init_set(loader->fap_path, (const char*)p);
  125. fap_loader_run_selected_app(loader);
  126. } else {
  127. string_init_set(loader->fap_path, EXT_PATH("apps"));
  128. while(fap_loader_select_app(loader)) {
  129. view_dispatcher_switch_to_view(view_dispatcher, 0);
  130. fap_loader_run_selected_app(loader);
  131. };
  132. }
  133. view_dispatcher_remove_view(view_dispatcher, 0);
  134. loading_free(loading);
  135. view_dispatcher_free(view_dispatcher);
  136. string_clear(loader->fap_path);
  137. furi_record_close(RECORD_GUI);
  138. furi_record_close(RECORD_DIALOGS);
  139. furi_record_close(RECORD_STORAGE);
  140. free(loader);
  141. return 0;
  142. }