mirror of
https://github.com/DarkFlippers/unleashed-firmware.git
synced 2024-12-26 14:51:52 +03:00
217 lines
6.6 KiB
C
217 lines
6.6 KiB
C
|
#include "test_runner.h"
|
||
|
|
||
|
#include "tests/test_api.h"
|
||
|
|
||
|
#include <cli/cli.h>
|
||
|
#include <toolbox/path.h>
|
||
|
#include <loader/loader.h>
|
||
|
#include <storage/storage.h>
|
||
|
#include <notification/notification_messages.h>
|
||
|
|
||
|
#include <loader/firmware_api/firmware_api.h>
|
||
|
#include <flipper_application/flipper_application.h>
|
||
|
#include <flipper_application/api_hashtable/api_hashtable.h>
|
||
|
#include <flipper_application/plugins/composite_resolver.h>
|
||
|
|
||
|
extern const ElfApiInterface* const unit_tests_api_interface;
|
||
|
|
||
|
#define TAG "TestRunner"
|
||
|
|
||
|
#define PLUGINS_PATH "/ext/apps_data/unit_tests/plugins"
|
||
|
|
||
|
struct TestRunner {
|
||
|
Storage* storage;
|
||
|
Loader* loader;
|
||
|
NotificationApp* notification;
|
||
|
|
||
|
// Temporary used things
|
||
|
Cli* cli;
|
||
|
FuriString* args;
|
||
|
|
||
|
// ELF related stuff
|
||
|
CompositeApiResolver* composite_resolver;
|
||
|
|
||
|
// Report data
|
||
|
int minunit_run;
|
||
|
int minunit_assert;
|
||
|
int minunit_fail;
|
||
|
int minunit_status;
|
||
|
};
|
||
|
|
||
|
TestRunner* test_runner_alloc(Cli* cli, FuriString* args) {
|
||
|
TestRunner* instance = malloc(sizeof(TestRunner));
|
||
|
|
||
|
instance->storage = furi_record_open(RECORD_STORAGE);
|
||
|
instance->loader = furi_record_open(RECORD_LOADER);
|
||
|
instance->notification = furi_record_open(RECORD_NOTIFICATION);
|
||
|
|
||
|
instance->cli = cli;
|
||
|
instance->args = args;
|
||
|
|
||
|
instance->composite_resolver = composite_api_resolver_alloc();
|
||
|
composite_api_resolver_add(instance->composite_resolver, firmware_api_interface);
|
||
|
composite_api_resolver_add(instance->composite_resolver, unit_tests_api_interface);
|
||
|
|
||
|
return instance;
|
||
|
}
|
||
|
|
||
|
void test_runner_free(TestRunner* instance) {
|
||
|
furi_assert(instance);
|
||
|
|
||
|
composite_api_resolver_free(instance->composite_resolver);
|
||
|
|
||
|
furi_record_close(RECORD_NOTIFICATION);
|
||
|
instance->notification = NULL;
|
||
|
|
||
|
furi_record_close(RECORD_LOADER);
|
||
|
instance->loader = NULL;
|
||
|
|
||
|
furi_record_close(RECORD_STORAGE);
|
||
|
instance->storage = NULL;
|
||
|
|
||
|
free(instance);
|
||
|
}
|
||
|
|
||
|
static bool test_runner_run_plugin(TestRunner* instance, const char* path) {
|
||
|
furi_assert(instance);
|
||
|
|
||
|
FURI_LOG_D(TAG, "Loading %s", path);
|
||
|
FlipperApplication* lib = flipper_application_alloc(
|
||
|
instance->storage, composite_api_resolver_get(instance->composite_resolver));
|
||
|
|
||
|
bool result = false;
|
||
|
instance->minunit_fail = -1;
|
||
|
do {
|
||
|
FlipperApplicationPreloadStatus preload_res = flipper_application_preload(lib, path);
|
||
|
|
||
|
if(preload_res != FlipperApplicationPreloadStatusSuccess) {
|
||
|
FURI_LOG_E(TAG, "Failed to preload %s, %d", path, preload_res);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if(!flipper_application_is_plugin(lib)) {
|
||
|
FURI_LOG_E(TAG, "Not a plugin %s", path);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
FlipperApplicationLoadStatus load_status = flipper_application_map_to_memory(lib);
|
||
|
if(load_status != FlipperApplicationLoadStatusSuccess) {
|
||
|
FURI_LOG_E(TAG, "Failed to load %s", path);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
const FlipperAppPluginDescriptor* app_descriptor =
|
||
|
flipper_application_plugin_get_descriptor(lib);
|
||
|
|
||
|
const TestApi* test = app_descriptor->entry_point;
|
||
|
|
||
|
instance->minunit_fail = test->run();
|
||
|
|
||
|
instance->minunit_run += test->get_minunit_run();
|
||
|
instance->minunit_assert += test->get_minunit_assert();
|
||
|
instance->minunit_status += test->get_minunit_status();
|
||
|
|
||
|
result = (instance->minunit_fail == 0);
|
||
|
} while(false);
|
||
|
|
||
|
flipper_application_free(lib);
|
||
|
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
static void test_runner_run_internal(TestRunner* instance) {
|
||
|
furi_assert(instance);
|
||
|
|
||
|
char file_name_buffer[256];
|
||
|
FuriString* file_name = furi_string_alloc();
|
||
|
FuriString* file_basename = furi_string_alloc();
|
||
|
File* directory = storage_file_alloc(instance->storage);
|
||
|
|
||
|
do {
|
||
|
if(!storage_dir_open(directory, PLUGINS_PATH)) {
|
||
|
FURI_LOG_E(TAG, "Failed to open directory %s", PLUGINS_PATH);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
while(true) {
|
||
|
if(cli_cmd_interrupt_received(instance->cli)) {
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if(!storage_dir_read(directory, NULL, file_name_buffer, sizeof(file_name_buffer))) {
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
furi_string_set(file_name, file_name_buffer);
|
||
|
if(!furi_string_end_with_str(file_name, ".fal")) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
path_concat(PLUGINS_PATH, file_name_buffer, file_name);
|
||
|
|
||
|
path_extract_filename(file_name, file_basename, true);
|
||
|
const char* file_basename_cstr = furi_string_get_cstr(file_basename);
|
||
|
|
||
|
bool result = true;
|
||
|
if(furi_string_size(instance->args)) {
|
||
|
if(furi_string_cmp_str(instance->args, file_basename_cstr) == 0) {
|
||
|
result = test_runner_run_plugin(instance, furi_string_get_cstr(file_name));
|
||
|
} else {
|
||
|
printf("Skipping %s\r\n", file_basename_cstr);
|
||
|
}
|
||
|
} else {
|
||
|
result = test_runner_run_plugin(instance, furi_string_get_cstr(file_name));
|
||
|
}
|
||
|
|
||
|
if(!result) {
|
||
|
printf("Failed to execute test: %s\r\n", file_basename_cstr);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
} while(false);
|
||
|
|
||
|
storage_dir_close(directory);
|
||
|
storage_file_free(directory);
|
||
|
furi_string_free(file_name);
|
||
|
furi_string_free(file_basename);
|
||
|
}
|
||
|
|
||
|
void test_runner_run(TestRunner* instance) {
|
||
|
furi_assert(instance);
|
||
|
|
||
|
// TODO FL-3491: lock device while test running
|
||
|
if(loader_is_locked(instance->loader)) {
|
||
|
printf("RPC: stop all applications to run tests\r\n");
|
||
|
notification_message(instance->notification, &sequence_blink_magenta_100);
|
||
|
} else {
|
||
|
notification_message_block(instance->notification, &sequence_set_only_blue_255);
|
||
|
|
||
|
uint32_t heap_before = memmgr_get_free_heap();
|
||
|
uint32_t cycle_counter = furi_get_tick();
|
||
|
|
||
|
test_runner_run_internal(instance);
|
||
|
|
||
|
if(instance->minunit_run != 0) {
|
||
|
printf("\r\nFailed tests: %d\r\n", instance->minunit_fail);
|
||
|
|
||
|
// Time report
|
||
|
cycle_counter = (furi_get_tick() - cycle_counter);
|
||
|
printf("Consumed: %lu ms\r\n", cycle_counter);
|
||
|
|
||
|
// Wait for tested services and apps to deallocate memory
|
||
|
furi_delay_ms(200);
|
||
|
uint32_t heap_after = memmgr_get_free_heap();
|
||
|
printf("Leaked: %ld\r\n", heap_before - heap_after);
|
||
|
|
||
|
// Final Report
|
||
|
if(instance->minunit_fail == 0) {
|
||
|
notification_message(instance->notification, &sequence_success);
|
||
|
printf("Status: PASSED\r\n");
|
||
|
} else {
|
||
|
notification_message(instance->notification, &sequence_error);
|
||
|
printf("Status: FAILED\r\n");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|