* fap-loader: load all code and data sections * fap-loader: relocate all code and data sections * fap-loader: remove old elf loader * fap-loader: new jmp call relocation * openocd: resume on detach * fap-loader: trampoline for big jumps * fap-loader: rename cache * fap-loader: init_array support * fap-loader: untangled flipper_application into separate entities * fap-loader: fix debug * fap-loader: optimize section container * fap-loader: optimize key for section container * fap-loader: disable debug log * documentation * F7: bump api symbols version * Lib: cleanup elf_file.c Co-authored-by: あく <alleteam@gmail.com>
		
			
				
	
	
		
			145 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			145 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
#include "flipper_application.h"
 | 
						|
#include "elf/elf_file.h"
 | 
						|
 | 
						|
#define TAG "fapp"
 | 
						|
 | 
						|
struct FlipperApplication {
 | 
						|
    ELFDebugInfo state;
 | 
						|
    FlipperApplicationManifest manifest;
 | 
						|
    ELFFile* elf;
 | 
						|
    FuriThread* thread;
 | 
						|
};
 | 
						|
 | 
						|
/* For debugger access to app state */
 | 
						|
FlipperApplication* last_loaded_app = NULL;
 | 
						|
 | 
						|
FlipperApplication*
 | 
						|
    flipper_application_alloc(Storage* storage, const ElfApiInterface* api_interface) {
 | 
						|
    FlipperApplication* app = malloc(sizeof(FlipperApplication));
 | 
						|
    app->elf = elf_file_alloc(storage, api_interface);
 | 
						|
    app->thread = NULL;
 | 
						|
    return app;
 | 
						|
}
 | 
						|
 | 
						|
void flipper_application_free(FlipperApplication* app) {
 | 
						|
    furi_assert(app);
 | 
						|
 | 
						|
    if(app->thread) {
 | 
						|
        furi_thread_join(app->thread);
 | 
						|
        furi_thread_free(app->thread);
 | 
						|
    }
 | 
						|
 | 
						|
    last_loaded_app = NULL;
 | 
						|
 | 
						|
    elf_file_clear_debug_info(&app->state);
 | 
						|
    elf_file_free(app->elf);
 | 
						|
    free(app);
 | 
						|
}
 | 
						|
 | 
						|
static FlipperApplicationPreloadStatus
 | 
						|
    flipper_application_validate_manifest(FlipperApplication* app) {
 | 
						|
    if(!flipper_application_manifest_is_valid(&app->manifest)) {
 | 
						|
        return FlipperApplicationPreloadStatusInvalidManifest;
 | 
						|
    }
 | 
						|
 | 
						|
    if(!flipper_application_manifest_is_compatible(
 | 
						|
           &app->manifest, elf_file_get_api_interface(app->elf))) {
 | 
						|
        return FlipperApplicationPreloadStatusApiMismatch;
 | 
						|
    }
 | 
						|
 | 
						|
    return FlipperApplicationPreloadStatusSuccess;
 | 
						|
}
 | 
						|
 | 
						|
/* Parse headers, load manifest */
 | 
						|
FlipperApplicationPreloadStatus
 | 
						|
    flipper_application_preload_manifest(FlipperApplication* app, const char* path) {
 | 
						|
    if(!elf_file_open(app->elf, path) || !elf_file_load_manifest(app->elf, &app->manifest)) {
 | 
						|
        return FlipperApplicationPreloadStatusInvalidFile;
 | 
						|
    }
 | 
						|
 | 
						|
    return flipper_application_validate_manifest(app);
 | 
						|
}
 | 
						|
 | 
						|
/* Parse headers, load full file */
 | 
						|
FlipperApplicationPreloadStatus
 | 
						|
    flipper_application_preload(FlipperApplication* app, const char* path) {
 | 
						|
    if(!elf_file_open(app->elf, path) || !elf_file_load_section_table(app->elf, &app->manifest)) {
 | 
						|
        return FlipperApplicationPreloadStatusInvalidFile;
 | 
						|
    }
 | 
						|
 | 
						|
    return flipper_application_validate_manifest(app);
 | 
						|
}
 | 
						|
 | 
						|
const FlipperApplicationManifest* flipper_application_get_manifest(FlipperApplication* app) {
 | 
						|
    return &app->manifest;
 | 
						|
}
 | 
						|
 | 
						|
FlipperApplicationLoadStatus flipper_application_map_to_memory(FlipperApplication* app) {
 | 
						|
    last_loaded_app = app;
 | 
						|
    ELFFileLoadStatus status = elf_file_load_sections(app->elf);
 | 
						|
 | 
						|
    switch(status) {
 | 
						|
    case ELFFileLoadStatusSuccess:
 | 
						|
        elf_file_init_debug_info(app->elf, &app->state);
 | 
						|
        return FlipperApplicationLoadStatusSuccess;
 | 
						|
    case ELFFileLoadStatusNoFreeMemory:
 | 
						|
        return FlipperApplicationLoadStatusNoFreeMemory;
 | 
						|
    case ELFFileLoadStatusMissingImports:
 | 
						|
        return FlipperApplicationLoadStatusMissingImports;
 | 
						|
    default:
 | 
						|
        return FlipperApplicationLoadStatusUnspecifiedError;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static int32_t flipper_application_thread(void* context) {
 | 
						|
    elf_file_pre_run(last_loaded_app->elf);
 | 
						|
    int32_t result = elf_file_run(last_loaded_app->elf, context);
 | 
						|
    elf_file_post_run(last_loaded_app->elf);
 | 
						|
    return result;
 | 
						|
}
 | 
						|
 | 
						|
FuriThread* flipper_application_spawn(FlipperApplication* app, void* args) {
 | 
						|
    furi_check(app->thread == NULL);
 | 
						|
 | 
						|
    const FlipperApplicationManifest* manifest = flipper_application_get_manifest(app);
 | 
						|
    furi_check(manifest->stack_size > 0);
 | 
						|
 | 
						|
    app->thread = furi_thread_alloc();
 | 
						|
    furi_thread_set_stack_size(app->thread, manifest->stack_size);
 | 
						|
    furi_thread_set_name(app->thread, manifest->name);
 | 
						|
    furi_thread_set_callback(app->thread, flipper_application_thread);
 | 
						|
    furi_thread_set_context(app->thread, args);
 | 
						|
 | 
						|
    return app->thread;
 | 
						|
}
 | 
						|
 | 
						|
static const char* preload_status_strings[] = {
 | 
						|
    [FlipperApplicationPreloadStatusSuccess] = "Success",
 | 
						|
    [FlipperApplicationPreloadStatusUnspecifiedError] = "Unknown error",
 | 
						|
    [FlipperApplicationPreloadStatusInvalidFile] = "Invalid file",
 | 
						|
    [FlipperApplicationPreloadStatusInvalidManifest] = "Invalid file manifest",
 | 
						|
    [FlipperApplicationPreloadStatusApiMismatch] = "API version mismatch",
 | 
						|
    [FlipperApplicationPreloadStatusTargetMismatch] = "Hardware target mismatch",
 | 
						|
};
 | 
						|
 | 
						|
static const char* load_status_strings[] = {
 | 
						|
    [FlipperApplicationLoadStatusSuccess] = "Success",
 | 
						|
    [FlipperApplicationLoadStatusUnspecifiedError] = "Unknown error",
 | 
						|
    [FlipperApplicationLoadStatusNoFreeMemory] = "Out of memory",
 | 
						|
    [FlipperApplicationLoadStatusMissingImports] = "Found unsatisfied imports",
 | 
						|
};
 | 
						|
 | 
						|
const char* flipper_application_preload_status_to_string(FlipperApplicationPreloadStatus status) {
 | 
						|
    if(status >= COUNT_OF(preload_status_strings) || preload_status_strings[status] == NULL) {
 | 
						|
        return "Unknown error";
 | 
						|
    }
 | 
						|
    return preload_status_strings[status];
 | 
						|
}
 | 
						|
 | 
						|
const char* flipper_application_load_status_to_string(FlipperApplicationLoadStatus status) {
 | 
						|
    if(status >= COUNT_OF(load_status_strings) || load_status_strings[status] == NULL) {
 | 
						|
        return "Unknown error";
 | 
						|
    }
 | 
						|
    return load_status_strings[status];
 | 
						|
}
 |