Applications dependency, initial realization (#159)

This commit is contained in:
DrZlo13 2020-10-05 20:17:34 +10:00 committed by GitHub
parent 44ab04f8d6
commit ec5e5da138
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
8 changed files with 103 additions and 20 deletions

View File

@ -171,6 +171,9 @@ void display_u8g2(void* p) {
u8g2_SendBuffer(u8g2); u8g2_SendBuffer(u8g2);
furi_give(fb_record); furi_give(fb_record);
// we ready to work
furiac_ready();
while(1) { while(1) {
// wait for event // wait for event
if(xSemaphoreTake(update, 10000) == pdTRUE) { if(xSemaphoreTake(update, 10000) == pdTRUE) {

View File

@ -14,9 +14,6 @@ static void event_cb(const void* value, size_t size, void* ctx) {
} }
void application_input_dump(void* p) { void application_input_dump(void* p) {
// TODO try open record and retry on timeout (needs FURI behaviour change)
delay(1000);
// open record // open record
FuriRecordSubscriber* state_record = FuriRecordSubscriber* state_record =
furi_open("input_state", false, false, state_cb, NULL, NULL); furi_open("input_state", false, false, state_cb, NULL, NULL);

View File

@ -4,9 +4,6 @@
void u8g2_example(void* p) { void u8g2_example(void* p) {
FuriRecordSubscriber* log = get_default_log(); FuriRecordSubscriber* log = get_default_log();
// TODO try open record and retry on timeout (needs FURI behaviour change)
delay(1000);
// open record // open record
FuriRecordSubscriber* fb_record = furi_open("u8g2_fb", false, false, NULL, NULL, NULL); FuriRecordSubscriber* fb_record = furi_open("u8g2_fb", false, false, NULL, NULL, NULL);

View File

@ -40,6 +40,8 @@ void input_task(void* p) {
furiac_exit(NULL); furiac_exit(NULL);
} }
// we ready to work
furiac_ready();
initialized = true; initialized = true;
// Force state update // Force state update

View File

@ -5,6 +5,7 @@
typedef struct { typedef struct {
FlipperApplication app; FlipperApplication app;
const char* name; const char* name;
const char* libs;
} FlipperStartupApp; } FlipperStartupApp;
#ifdef TEST #ifdef TEST
@ -27,31 +28,31 @@ void coreglitch_demo_0(void* p);
const FlipperStartupApp FLIPPER_STARTUP[] = { const FlipperStartupApp FLIPPER_STARTUP[] = {
#ifndef TEST #ifndef TEST
{.app = display_u8g2, .name = "display_u8g2"}, {.app = display_u8g2, .name = "display_u8g2", .libs = ""},
{.app = u8g2_example, .name = "u8g2_example"}, {.app = u8g2_example, .name = "u8g2_example", .libs = "display_u8g2"},
#endif #endif
#ifdef USE_INPUT #ifdef USE_INPUT
{.app = input_task, .name = "input_task"}, {.app = input_task, .name = "input_task", .libs = ""},
#endif #endif
// {.app = coreglitch_demo_0, .name = "coreglitch_demo_0"}, // {.app = coreglitch_demo_0, .name = "coreglitch_demo_0", .libs = ""},
#ifdef TEST #ifdef TEST
{.app = flipper_test_app, .name = "test app"}, {.app = flipper_test_app, .name = "test app", .libs = ""},
#endif #endif
#ifdef EXAMPLE_BLINK #ifdef EXAMPLE_BLINK
{.app = application_blink, .name = "blink"}, {.app = application_blink, .name = "blink", .libs = ""},
#endif #endif
#ifdef EXAMPLE_UART_WRITE #ifdef EXAMPLE_UART_WRITE
{.app = application_uart_write, .name = "uart write"}, {.app = application_uart_write, .name = "uart write", .libs = ""},
#endif #endif
#ifdef EXAMPLE_IPC #ifdef EXAMPLE_IPC
{.app = application_ipc_display, .name = "ipc display"}, {.app = application_ipc_display, .name = "ipc display", .libs = ""},
{.app = application_ipc_widget, .name = "ipc widget"}, {.app = application_ipc_widget, .name = "ipc widget", .libs = ""},
#endif #endif
#ifdef EXAMPLE_INPUT_DUMP #ifdef EXAMPLE_INPUT_DUMP
{.app = application_input_dump, .name = "input dump"}, {.app = application_input_dump, .name = "input dump", .libs = "input_task"},
#endif #endif
}; };

View File

@ -2,9 +2,9 @@
#include <stdio.h> #include <stdio.h>
extern "C" { extern "C" {
#include "startup.h"
#include "furi.h" #include "furi.h"
#include "log.h" #include "log.h"
#include "startup.h"
#include "tty_uart.h" #include "tty_uart.h"
} }
@ -15,16 +15,19 @@ extern "C" void app() {
fuprintf(log, "\n=== Welcome to Flipper Zero! ===\n\n"); fuprintf(log, "\n=== Welcome to Flipper Zero! ===\n\n");
// FURI startup // FURI startup
FuriApp* handlers[sizeof(FLIPPER_STARTUP) / sizeof(FLIPPER_STARTUP[0])]; const size_t flipper_app_count = sizeof(FLIPPER_STARTUP) / sizeof(FLIPPER_STARTUP[0]);
FuriApp* handlers[flipper_app_count];
for(size_t i = 0; i < sizeof(FLIPPER_STARTUP) / sizeof(FLIPPER_STARTUP[0]); i++) { for(size_t i = 0; i < flipper_app_count; i++) {
// TODO create a dependency tree and run tasks in the desired order
furiac_wait_libs(FLIPPER_STARTUP[i].libs);
handlers[i] = furiac_start(FLIPPER_STARTUP[i].app, FLIPPER_STARTUP[i].name, NULL); handlers[i] = furiac_start(FLIPPER_STARTUP[i].app, FLIPPER_STARTUP[i].name, NULL);
} }
bool is_alive = false; bool is_alive = false;
do { do {
is_alive = false; is_alive = false;
for(size_t i = 0; i < sizeof(FLIPPER_STARTUP) / sizeof(FLIPPER_STARTUP[0]); i++) { for(size_t i = 0; i < flipper_app_count; i++) {
if(handlers[i]->handler != NULL) { if(handlers[i]->handler != NULL) {
is_alive = true; is_alive = true;
} }
@ -32,4 +35,6 @@ extern "C" void app() {
delay(500); delay(500);
// TODO add deferred event queue here // TODO add deferred event queue here
} while(is_alive); } while(is_alive);
fuprintf(log, "\n=== Bye from Flipper Zero! ===\n\n");
} }

View File

@ -56,6 +56,8 @@ typedef struct {
TaskHandle_t handler; TaskHandle_t handler;
uint8_t records_count; ///< count of records which task open uint8_t records_count; ///< count of records which task open
FuriRecord* records[MAX_TASK_RECORDS]; ///< list of records which task open FuriRecord* records[MAX_TASK_RECORDS]; ///< list of records which task open
bool ready;
} FuriApp; } FuriApp;
/*! /*!
@ -82,6 +84,16 @@ application registry.
*/ */
void furiac_exit(void* param); void furiac_exit(void* param);
/*!
Mark application as prepared and ready to perform actions
*/
void furiac_ready();
/*
Wait for the libraries we depend on
*/
void furiac_wait_libs(const char* libs);
/*! /*!
Stop specified app without returning to prev application. Stop specified app without returning to prev application.
*/ */

View File

@ -7,8 +7,11 @@
#include <stdio.h> #include <stdio.h>
#endif #endif
#include <string.h>
#define DEFAULT_STACK_SIZE 1024 // Stack size in bytes #define DEFAULT_STACK_SIZE 1024 // Stack size in bytes
#define MAX_TASK_COUNT 8 #define MAX_TASK_COUNT 8
#define INVALID_TASK_ID UINT16_MAX
static StaticTask_t task_info_buffer[MAX_TASK_COUNT]; static StaticTask_t task_info_buffer[MAX_TASK_COUNT];
static StackType_t stack_buffer[MAX_TASK_COUNT][DEFAULT_STACK_SIZE / 4]; static StackType_t stack_buffer[MAX_TASK_COUNT][DEFAULT_STACK_SIZE / 4];
@ -16,6 +19,45 @@ static FuriApp task_buffer[MAX_TASK_COUNT];
static size_t current_buffer_idx = 0; static size_t current_buffer_idx = 0;
uint16_t furiac_get_task_id_by_name(const char* app_name) {
for(size_t i = 0; i < MAX_TASK_RECORDS; i++) {
if(strcmp(task_buffer[i].name, app_name) == 0) return i;
}
return INVALID_TASK_ID;
}
void furiac_wait_libs(const char* libs) {
char* lib_rest = NULL;
char* lib_name = strtok_r((char*)libs, " ", &lib_rest);
while(lib_name != NULL) {
// trim library name
for(uint16_t i = 0; i < strlen(lib_name); i++) {
if(lib_name[i] == ' ') {
lib_name[i] = 0;
}
}
uint16_t app_id = furiac_get_task_id_by_name(lib_name);
if(app_id == INVALID_TASK_ID) {
#ifdef FURI_DEBUG
printf("[FURIAC] Invalid library name %s\n", lib_name);
#endif
} else {
while(!task_buffer[app_id].ready) {
#ifdef FURI_DEBUG
printf("[FURIAC] waiting for library \"%s\"\n", lib_name);
#endif
osDelay(50);
}
}
lib_name = strtok_r(NULL, " ", &lib_rest);
}
}
// find task pointer by handle // find task pointer by handle
FuriApp* find_task(TaskHandle_t handler) { FuriApp* find_task(TaskHandle_t handler) {
FuriApp* res = NULL; FuriApp* res = NULL;
@ -43,6 +85,9 @@ FuriApp* furiac_start(FlipperApplication app, const char* name, void* param) {
return NULL; return NULL;
} }
// application ready
task_buffer[current_buffer_idx].ready = false;
// create task on static stack memory // create task on static stack memory
task_buffer[current_buffer_idx].handler = xTaskCreateStatic( task_buffer[current_buffer_idx].handler = xTaskCreateStatic(
(TaskFunction_t)app, (TaskFunction_t)app,
@ -135,4 +180,25 @@ void furiac_switch(FlipperApplication app, char* name, void* param) {
// kill itself // kill itself
vTaskDelete(NULL); vTaskDelete(NULL);
} }
}
// set task to ready state
void furiac_ready() {
/*
TODO:
Currently i think that better way is to use application name
and restrict applications to "one task per application"
*/
FuriApp* app = find_task(xTaskGetCurrentTaskHandle());
if(app == NULL) {
#ifdef FURI_DEBUG
printf("[FURIAC] cannot find task to set ready state\n");
#endif
} else {
#ifdef FURI_DEBUG
printf("[FURIAC] task is ready\n");
#endif
app->ready = true;
}
} }