mirror of
https://github.com/DarkFlippers/unleashed-firmware.git
synced 2024-12-26 23:05:05 +03:00
204 lines
5.1 KiB
C
204 lines
5.1 KiB
C
#include "furi.h"
|
|
#include "cmsis_os.h"
|
|
|
|
// TODO: this file contains printf, that not implemented on uC target
|
|
|
|
#ifdef FURI_DEBUG
|
|
#include <stdio.h>
|
|
#endif
|
|
|
|
#include <string.h>
|
|
|
|
#define DEFAULT_STACK_SIZE 1024 // Stack size in bytes
|
|
#define MAX_TASK_COUNT 8
|
|
#define INVALID_TASK_ID UINT16_MAX
|
|
|
|
static StaticTask_t task_info_buffer[MAX_TASK_COUNT];
|
|
static StackType_t stack_buffer[MAX_TASK_COUNT][DEFAULT_STACK_SIZE / 4];
|
|
static FuriApp task_buffer[MAX_TASK_COUNT];
|
|
|
|
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
|
|
FuriApp* find_task(TaskHandle_t handler) {
|
|
FuriApp* res = NULL;
|
|
for(size_t i = 0; i < MAX_TASK_COUNT; i++) {
|
|
if(task_equal(task_buffer[i].handler, handler)) {
|
|
res = &task_buffer[i];
|
|
}
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
FuriApp* furiac_start(FlipperApplication app, const char* name, void* param) {
|
|
#ifdef FURI_DEBUG
|
|
printf("[FURIAC] start %s\n", name);
|
|
#endif
|
|
|
|
// TODO check first free item (.handler == NULL) and use it
|
|
|
|
if(current_buffer_idx >= MAX_TASK_COUNT) {
|
|
// max task count exceed
|
|
#ifdef FURI_DEBUG
|
|
printf("[FURIAC] max task count exceed\n");
|
|
#endif
|
|
return NULL;
|
|
}
|
|
|
|
// application ready
|
|
task_buffer[current_buffer_idx].ready = false;
|
|
|
|
// create task on static stack memory
|
|
task_buffer[current_buffer_idx].handler = xTaskCreateStatic(
|
|
(TaskFunction_t)app,
|
|
(const char* const)name,
|
|
DEFAULT_STACK_SIZE / 4, // freertos specify stack size in words
|
|
(void* const)param,
|
|
tskIDLE_PRIORITY + 3, // normal priority
|
|
stack_buffer[current_buffer_idx],
|
|
&task_info_buffer[current_buffer_idx]);
|
|
|
|
// save task
|
|
task_buffer[current_buffer_idx].application = app;
|
|
task_buffer[current_buffer_idx].prev_name = NULL;
|
|
task_buffer[current_buffer_idx].prev = NULL;
|
|
task_buffer[current_buffer_idx].records_count = 0;
|
|
task_buffer[current_buffer_idx].name = name;
|
|
|
|
current_buffer_idx++;
|
|
|
|
return &task_buffer[current_buffer_idx - 1];
|
|
}
|
|
|
|
bool furiac_kill(FuriApp* app) {
|
|
#ifdef FURI_DEBUG
|
|
printf("[FURIAC] kill %s\n", app->name);
|
|
#endif
|
|
|
|
// check handler
|
|
if(app == NULL || app->handler == NULL) return false;
|
|
|
|
// kill task
|
|
vTaskDelete(app->handler);
|
|
|
|
// cleanup its registry
|
|
// TODO realy free memory
|
|
app->handler = NULL;
|
|
|
|
return true;
|
|
}
|
|
|
|
void furiac_exit(void* param) {
|
|
// get current task handler
|
|
FuriApp* current_task = find_task(xTaskGetCurrentTaskHandle());
|
|
|
|
// run prev
|
|
if(current_task != NULL) {
|
|
#ifdef FURI_DEBUG
|
|
printf("[FURIAC] exit %s\n", current_task->name);
|
|
#endif
|
|
|
|
if(current_task->prev != NULL) {
|
|
furiac_start(current_task->prev, current_task->prev_name, param);
|
|
} else {
|
|
#ifdef FURI_DEBUG
|
|
printf("[FURIAC] no prev\n");
|
|
#endif
|
|
}
|
|
|
|
// cleanup registry
|
|
// TODO realy free memory
|
|
current_task->handler = NULL;
|
|
}
|
|
|
|
// kill itself
|
|
vTaskDelete(NULL);
|
|
}
|
|
|
|
void furiac_switch(FlipperApplication app, char* name, void* param) {
|
|
// get current task handler
|
|
FuriApp* current_task = find_task(xTaskGetCurrentTaskHandle());
|
|
|
|
if(current_task == NULL) {
|
|
#ifdef FURI_DEBUG
|
|
printf("[FURIAC] no current task found\n");
|
|
#endif
|
|
}
|
|
|
|
#ifdef FURI_DEBUG
|
|
printf("[FURIAC] switch %s to %s\n", current_task->name, name);
|
|
#endif
|
|
|
|
// run next
|
|
FuriApp* next = furiac_start(app, name, param);
|
|
|
|
if(next != NULL) {
|
|
// save current application pointer as prev
|
|
next->prev = current_task->application;
|
|
next->prev_name = current_task->name;
|
|
|
|
// kill itself
|
|
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;
|
|
}
|
|
} |