#include #include "loader/loader.h" #include "loader_i.h" #define TAG "LoaderSrv" #define LOADER_THREAD_FLAG_SHOW_MENU (1 << 0) #define LOADER_THREAD_FLAG_ALL (LOADER_THREAD_FLAG_SHOW_MENU) static Loader* loader_instance = NULL; static bool loader_start_application(const FlipperApplication* application, const char* arguments) { loader_instance->application = application; furi_assert(loader_instance->application_arguments == NULL); if(arguments && strlen(arguments) > 0) { loader_instance->application_arguments = strdup(arguments); } FURI_LOG_I(TAG, "Starting: %s", loader_instance->application->name); furi_thread_set_name(loader_instance->application_thread, loader_instance->application->name); furi_thread_set_stack_size( loader_instance->application_thread, loader_instance->application->stack_size); furi_thread_set_context( loader_instance->application_thread, loader_instance->application_arguments); furi_thread_set_callback( loader_instance->application_thread, loader_instance->application->app); bool result = furi_thread_start(loader_instance->application_thread); if(!result) { loader_instance->application = NULL; } return result; } static void loader_menu_callback(void* _ctx, uint32_t index) { const FlipperApplication* application = _ctx; furi_assert(application->app); furi_assert(application->name); if(!loader_lock(loader_instance)) { FURI_LOG_E(TAG, "Loader is locked"); return; } loader_start_application(application, NULL); } static void loader_submenu_callback(void* context, uint32_t index) { uint32_t view_id = (uint32_t)context; view_dispatcher_switch_to_view(loader_instance->view_dispatcher, view_id); } static void loader_cli_print_usage() { printf("Usage:\r\n"); printf("loader \r\n"); printf("Cmd list:\r\n"); printf("\tlist\t - List available applications\r\n"); printf("\topen \t - Open application by name\r\n"); } const FlipperApplication* loader_find_application_by_name(const char* name) { const FlipperApplication* application = NULL; for(size_t i = 0; i < FLIPPER_APPS_COUNT; i++) { if(strcmp(name, FLIPPER_APPS[i].name) == 0) { application = &FLIPPER_APPS[i]; } } for(size_t i = 0; i < FLIPPER_PLUGINS_COUNT; i++) { if(strcmp(name, FLIPPER_PLUGINS[i].name) == 0) { application = &FLIPPER_PLUGINS[i]; } } if(furi_hal_rtc_is_flag_set(FuriHalRtcFlagDebug)) { for(size_t i = 0; i < FLIPPER_DEBUG_APPS_COUNT; i++) { if(strcmp(name, FLIPPER_DEBUG_APPS[i].name) == 0) { application = &FLIPPER_DEBUG_APPS[i]; } } } return application; } void loader_cli_open(Cli* cli, string_t args, Loader* instance) { string_t application_name; string_init(application_name); do { if(!args_read_probably_quoted_string_and_trim(args, application_name)) { printf("No application provided\r\n"); break; } const FlipperApplication* application = loader_find_application_by_name(string_get_cstr(application_name)); if(!application) { printf("%s doesn't exists\r\n", string_get_cstr(application_name)); break; } string_strim(args); if(!loader_start_application(application, string_get_cstr(args))) { printf("Can't start, furi application is running"); return; } else { // We must to increment lock counter to keep balance // TODO: rewrite whole thing, it's complex as hell FURI_CRITICAL_ENTER(); instance->lock_count++; FURI_CRITICAL_EXIT(); } } while(false); string_clear(application_name); } void loader_cli_list(Cli* cli, string_t args, Loader* instance) { printf("Applications:\r\n"); for(size_t i = 0; i < FLIPPER_APPS_COUNT; i++) { printf("\t%s\r\n", FLIPPER_APPS[i].name); } printf("Plugins:\r\n"); for(size_t i = 0; i < FLIPPER_PLUGINS_COUNT; i++) { printf("\t%s\r\n", FLIPPER_PLUGINS[i].name); } if(furi_hal_rtc_is_flag_set(FuriHalRtcFlagDebug)) { printf("Debug:\r\n"); for(size_t i = 0; i < FLIPPER_DEBUG_APPS_COUNT; i++) { printf("\t%s\r\n", FLIPPER_DEBUG_APPS[i].name); } } } void loader_cli(Cli* cli, string_t args, void* _ctx) { furi_assert(_ctx); Loader* instance = _ctx; string_t cmd; string_init(cmd); do { if(!args_read_string_and_trim(args, cmd)) { loader_cli_print_usage(); break; } if(string_cmp_str(cmd, "list") == 0) { loader_cli_list(cli, args, instance); break; } if(string_cmp_str(cmd, "open") == 0) { loader_cli_open(cli, args, instance); break; } loader_cli_print_usage(); } while(false); string_clear(cmd); } LoaderStatus loader_start(Loader* instance, const char* name, const char* args) { furi_assert(name); const FlipperApplication* application = loader_find_application_by_name(name); if(!application) { FURI_LOG_E(TAG, "Can't find application with name %s", name); return LoaderStatusErrorUnknownApp; } if(!loader_lock(loader_instance)) { FURI_LOG_E(TAG, "Loader is locked"); return LoaderStatusErrorAppStarted; } if(!loader_start_application(application, args)) { return LoaderStatusErrorInternal; } return LoaderStatusOk; } bool loader_lock(Loader* instance) { FURI_CRITICAL_ENTER(); bool result = false; if(instance->lock_count == 0) { instance->lock_count++; result = true; } FURI_CRITICAL_EXIT(); return result; } void loader_unlock(Loader* instance) { FURI_CRITICAL_ENTER(); if(instance->lock_count > 0) instance->lock_count--; FURI_CRITICAL_EXIT(); } bool loader_is_locked(Loader* instance) { return instance->lock_count > 0; } static void loader_thread_state_callback(FuriThreadState thread_state, void* context) { furi_assert(context); Loader* instance = context; LoaderEvent event; if(thread_state == FuriThreadStateRunning) { event.type = LoaderEventTypeApplicationStarted; furi_pubsub_publish(loader_instance->pubsub, &event); furi_hal_power_insomnia_enter(); // Snapshot current memory usage instance->free_heap_size = memmgr_get_free_heap(); } else if(thread_state == FuriThreadStateStopped) { /* * Current Leak Sanitizer assumes that memory is allocated and freed * inside one thread. Timers are allocated in one task, but freed in * Timer-Task thread, and xTimerDelete() just put command to queue. * To avoid some bad cases there are few fixes: * 1) delay for Timer to process commands * 2) there are 'heap diff' which shows difference in heap before task * started and after task completed. In process of leakage monitoring * both values should be taken into account. */ delay(20); int heap_diff = instance->free_heap_size - memmgr_get_free_heap(); FURI_LOG_I( TAG, "Application thread stopped. Heap allocation balance: %d. Thread allocation balance: %d.", heap_diff, furi_thread_get_heap_size(instance->application_thread)); if(loader_instance->application_arguments) { free(loader_instance->application_arguments); loader_instance->application_arguments = NULL; } furi_hal_power_insomnia_exit(); loader_unlock(instance); event.type = LoaderEventTypeApplicationStopped; furi_pubsub_publish(loader_instance->pubsub, &event); } } static uint32_t loader_hide_menu(void* context) { return VIEW_NONE; } static uint32_t loader_back_to_primary_menu(void* context) { furi_assert(context); Submenu* submenu = context; submenu_set_selected_item(submenu, 0); return LoaderMenuViewPrimary; } static Loader* loader_alloc() { Loader* instance = furi_alloc(sizeof(Loader)); instance->application_thread = furi_thread_alloc(); furi_thread_enable_heap_trace(instance->application_thread); furi_thread_set_state_context(instance->application_thread, instance); furi_thread_set_state_callback(instance->application_thread, loader_thread_state_callback); instance->pubsub = furi_pubsub_alloc(); #ifdef SRV_CLI instance->cli = furi_record_open("cli"); cli_add_command(instance->cli, "loader", CliCommandFlagDefault, loader_cli, instance); #endif instance->loader_thread = osThreadGetId(); // Gui instance->gui = furi_record_open("gui"); instance->view_dispatcher = view_dispatcher_alloc(); view_dispatcher_attach_to_gui( instance->view_dispatcher, instance->gui, ViewDispatcherTypeFullscreen); // Primary menu instance->primary_menu = menu_alloc(); view_set_previous_callback(menu_get_view(instance->primary_menu), loader_hide_menu); view_dispatcher_add_view( instance->view_dispatcher, LoaderMenuViewPrimary, menu_get_view(instance->primary_menu)); // Plugins menu instance->plugins_menu = submenu_alloc(); view_set_context(submenu_get_view(instance->plugins_menu), instance->plugins_menu); view_set_previous_callback( submenu_get_view(instance->plugins_menu), loader_back_to_primary_menu); view_dispatcher_add_view( instance->view_dispatcher, LoaderMenuViewPlugins, submenu_get_view(instance->plugins_menu)); // Debug menu instance->debug_menu = submenu_alloc(); view_set_context(submenu_get_view(instance->debug_menu), instance->debug_menu); view_set_previous_callback( submenu_get_view(instance->debug_menu), loader_back_to_primary_menu); view_dispatcher_add_view( instance->view_dispatcher, LoaderMenuViewDebug, submenu_get_view(instance->debug_menu)); // Settings menu instance->settings_menu = submenu_alloc(); view_set_context(submenu_get_view(instance->settings_menu), instance->settings_menu); view_set_previous_callback( submenu_get_view(instance->settings_menu), loader_back_to_primary_menu); view_dispatcher_add_view( instance->view_dispatcher, LoaderMenuViewSettings, submenu_get_view(instance->settings_menu)); view_dispatcher_enable_queue(instance->view_dispatcher); return instance; } static void loader_free(Loader* instance) { furi_assert(instance); if(instance->cli) { furi_record_close("cli"); } furi_pubsub_free(instance->pubsub); furi_thread_free(instance->application_thread); menu_free(loader_instance->primary_menu); view_dispatcher_remove_view(loader_instance->view_dispatcher, LoaderMenuViewPrimary); submenu_free(loader_instance->plugins_menu); view_dispatcher_remove_view(loader_instance->view_dispatcher, LoaderMenuViewPlugins); submenu_free(loader_instance->debug_menu); view_dispatcher_remove_view(loader_instance->view_dispatcher, LoaderMenuViewDebug); submenu_free(loader_instance->settings_menu); view_dispatcher_remove_view(loader_instance->view_dispatcher, LoaderMenuViewSettings); view_dispatcher_free(loader_instance->view_dispatcher); furi_record_close("gui"); free(instance); instance = NULL; } static void loader_build_menu() { FURI_LOG_I(TAG, "Building main menu"); size_t i; for(i = 0; i < FLIPPER_APPS_COUNT; i++) { menu_add_item( loader_instance->primary_menu, FLIPPER_APPS[i].name, FLIPPER_APPS[i].icon, i, loader_menu_callback, (void*)&FLIPPER_APPS[i]); } if(FLIPPER_PLUGINS_COUNT != 0) { menu_add_item( loader_instance->primary_menu, "Plugins", &A_Plugins_14, i++, loader_submenu_callback, (void*)LoaderMenuViewPlugins); } if(furi_hal_rtc_is_flag_set(FuriHalRtcFlagDebug)) { menu_add_item( loader_instance->primary_menu, "Debug tools", &A_Debug_14, i++, loader_submenu_callback, (void*)LoaderMenuViewDebug); } menu_add_item( loader_instance->primary_menu, "Settings", &A_Settings_14, i++, loader_submenu_callback, (void*)LoaderMenuViewSettings); } static void loader_build_submenu() { FURI_LOG_I(TAG, "Building plugins menu"); size_t i; for(i = 0; i < FLIPPER_PLUGINS_COUNT; i++) { submenu_add_item( loader_instance->plugins_menu, FLIPPER_PLUGINS[i].name, i, loader_menu_callback, (void*)&FLIPPER_PLUGINS[i]); } FURI_LOG_I(TAG, "Building debug menu"); for(i = 0; i < FLIPPER_DEBUG_APPS_COUNT; i++) { submenu_add_item( loader_instance->debug_menu, FLIPPER_DEBUG_APPS[i].name, i, loader_menu_callback, (void*)&FLIPPER_DEBUG_APPS[i]); } FURI_LOG_I(TAG, "Building settings menu"); for(i = 0; i < FLIPPER_SETTINGS_APPS_COUNT; i++) { submenu_add_item( loader_instance->settings_menu, FLIPPER_SETTINGS_APPS[i].name, i, loader_menu_callback, (void*)&FLIPPER_SETTINGS_APPS[i]); } } void loader_show_menu() { furi_assert(loader_instance); osThreadFlagsSet(loader_instance->loader_thread, LOADER_THREAD_FLAG_SHOW_MENU); } void loader_update_menu() { menu_reset(loader_instance->primary_menu); loader_build_menu(); } int32_t loader_srv(void* p) { FURI_LOG_I(TAG, "Starting"); loader_instance = loader_alloc(); loader_build_menu(); loader_build_submenu(); // Call on start hooks for(size_t i = 0; i < FLIPPER_ON_SYSTEM_START_COUNT; i++) { FLIPPER_ON_SYSTEM_START[i](); } FURI_LOG_I(TAG, "Started"); furi_record_create("loader", loader_instance); #ifdef LOADER_AUTOSTART loader_start(loader_instance, LOADER_AUTOSTART, NULL); #endif while(1) { uint32_t flags = osThreadFlagsWait(LOADER_THREAD_FLAG_ALL, osFlagsWaitAny, osWaitForever); if(flags & LOADER_THREAD_FLAG_SHOW_MENU) { menu_set_selected_item(loader_instance->primary_menu, 0); view_dispatcher_switch_to_view( loader_instance->view_dispatcher, LoaderMenuViewPrimary); view_dispatcher_run(loader_instance->view_dispatcher); } } furi_record_destroy("loader"); loader_free(loader_instance); return 0; } FuriPubSub* loader_get_pubsub(Loader* instance) { return instance->pubsub; }