1
0
mirror of https://github.com/lvgl/lvgl.git synced 2025-01-21 06:53:01 +08:00
lvgl/src/lv_misc/lv_task.c

373 lines
10 KiB
C
Raw Normal View History

2017-11-23 20:42:14 +01:00
/**
2017-11-24 17:48:47 +01:00
* @file lv_task.c
* An 'lv_task' is a void (*fp) (void* param) type function which will be called periodically.
2018-06-19 09:49:58 +02:00
* A priority (5 levels + disable) can be assigned to lv_tasks.
2017-11-23 20:42:14 +01:00
*/
/*********************
* INCLUDES
*********************/
2017-11-26 11:38:28 +01:00
#include <stddef.h>
2017-11-23 20:42:14 +01:00
#include "lv_task.h"
#include "../lv_hal/lv_hal_tick.h"
#include "lv_gc.h"
#if defined(LV_GC_INCLUDE)
2019-04-04 07:15:40 +02:00
#include LV_GC_INCLUDE
#endif /* LV_ENABLE_GC */
2017-11-23 20:42:14 +01:00
/*********************
* DEFINES
*********************/
2019-04-04 07:15:40 +02:00
#define IDLE_MEAS_PERIOD 500 /*[ms]*/
2019-06-06 06:05:40 +02:00
#define DEF_PRIO LV_TASK_PRIO_MID
#define DEF_PERIOD 500
2017-11-23 20:42:14 +01:00
/**********************
* TYPEDEFS
**********************/
/**********************
* STATIC PROTOTYPES
**********************/
2019-05-17 08:29:38 +02:00
static bool lv_task_exec(lv_task_t * task);
2017-11-23 20:42:14 +01:00
/**********************
* STATIC VARIABLES
**********************/
2019-04-04 07:15:40 +02:00
static bool lv_task_run = false;
2017-11-23 20:42:14 +01:00
static uint8_t idle_last = 0;
2018-10-15 19:00:03 +02:00
static bool task_deleted;
static bool task_created;
2017-11-23 20:42:14 +01:00
/**********************
* MACROS
**********************/
/**********************
* GLOBAL FUNCTIONS
**********************/
/**
2017-11-24 17:48:47 +01:00
* Init the lv_task module
2017-11-23 20:42:14 +01:00
*/
2019-05-17 07:58:47 +02:00
void lv_task_core_init(void)
2017-11-23 20:42:14 +01:00
{
lv_ll_init(&LV_GC_ROOT(_lv_task_ll), sizeof(lv_task_t));
2018-06-19 09:49:58 +02:00
2017-11-24 17:48:47 +01:00
/*Initially enable the lv_task handling*/
lv_task_enable(true);
2017-11-23 20:42:14 +01:00
}
/**
2017-11-24 17:48:47 +01:00
* Call it periodically to handle lv_tasks.
2017-11-23 20:42:14 +01:00
*/
LV_ATTRIBUTE_TASK_HANDLER void lv_task_handler(void)
2017-11-23 20:42:14 +01:00
{
2018-10-05 17:22:49 +02:00
LV_LOG_TRACE("lv_task_handler started");
2018-07-25 17:57:08 +02:00
2018-10-05 17:22:49 +02:00
/*Avoid concurrent running of the task handler*/
static bool task_handler_mutex = false;
if(task_handler_mutex) return;
2018-10-05 17:22:49 +02:00
task_handler_mutex = true;
2017-12-20 00:47:37 +01:00
static uint32_t idle_period_start = 0;
2019-04-04 07:15:40 +02:00
static uint32_t handler_start = 0;
static uint32_t busy_time = 0;
2017-12-20 00:47:37 +01:00
if(lv_task_run == false) {
task_handler_mutex = false; /*Release mutex*/
return;
}
2017-11-23 20:42:14 +01:00
2018-06-19 09:49:58 +02:00
handler_start = lv_tick_get();
2017-12-20 00:47:37 +01:00
2018-06-19 09:49:58 +02:00
/* Run all task from the highest to the lowest priority
* If a lower priority task is executed check task again from the highest priority
* but on the priority of executed tasks don't run tasks before the executed*/
2018-07-22 22:03:22 +02:00
lv_task_t * task_interrupter = NULL;
2018-04-18 13:23:19 +02:00
lv_task_t * next;
2018-06-19 09:49:58 +02:00
bool end_flag;
do {
2019-04-04 07:15:40 +02:00
end_flag = true;
task_deleted = false;
task_created = false;
2019-01-18 22:18:56 +02:00
LV_GC_ROOT(_lv_task_act) = lv_ll_get_head(&LV_GC_ROOT(_lv_task_ll));
while(LV_GC_ROOT(_lv_task_act)) {
2018-06-19 09:49:58 +02:00
/* The task might be deleted if it runs only once ('once = 1')
* So get next element until the current is surely valid*/
2019-01-18 22:18:56 +02:00
next = lv_ll_get_next(&LV_GC_ROOT(_lv_task_ll), LV_GC_ROOT(_lv_task_act));
2018-06-19 09:49:58 +02:00
/*We reach priority of the turned off task. There is nothing more to do.*/
2019-01-18 22:18:56 +02:00
if(((lv_task_t *)LV_GC_ROOT(_lv_task_act))->prio == LV_TASK_PRIO_OFF) {
2018-06-19 09:49:58 +02:00
break;
}
/*Here is the interrupter task. Don't execute it again.*/
2019-01-18 22:18:56 +02:00
if(LV_GC_ROOT(_lv_task_act) == task_interrupter) {
2019-04-04 07:15:40 +02:00
task_interrupter = NULL; /*From this point only task after the interrupter comes, so
the interrupter is not interesting anymore*/
2019-01-18 22:18:56 +02:00
LV_GC_ROOT(_lv_task_act) = next;
2019-04-04 07:15:40 +02:00
continue; /*Load the next task*/
2018-06-19 09:49:58 +02:00
}
/*Just try to run the tasks with highest priority.*/
2019-01-18 22:18:56 +02:00
if(((lv_task_t *)LV_GC_ROOT(_lv_task_act))->prio == LV_TASK_PRIO_HIGHEST) {
lv_task_exec(LV_GC_ROOT(_lv_task_act));
2018-06-19 09:49:58 +02:00
}
/*Tasks with higher priority then the interrupted shall be run in every case*/
2018-07-22 22:03:22 +02:00
else if(task_interrupter) {
2019-01-18 22:18:56 +02:00
if(((lv_task_t *)LV_GC_ROOT(_lv_task_act))->prio > task_interrupter->prio) {
if(lv_task_exec(LV_GC_ROOT(_lv_task_act))) {
2019-06-06 06:05:40 +02:00
task_interrupter =
LV_GC_ROOT(_lv_task_act); /*Check all tasks again from the highest priority */
2018-06-19 09:49:58 +02:00
end_flag = false;
break;
}
}
}
/* It is no interrupter task or we already reached it earlier.
* Just run the remaining tasks*/
else {
2019-01-18 22:18:56 +02:00
if(lv_task_exec(LV_GC_ROOT(_lv_task_act))) {
2019-06-06 06:05:40 +02:00
task_interrupter = LV_GC_ROOT(_lv_task_act); /*Check all tasks again from the highest priority */
end_flag = false;
2018-06-19 09:49:58 +02:00
break;
}
}
2019-06-06 06:05:40 +02:00
if(task_deleted) break; /*If a task was deleted then this or the next item might be corrupted*/
if(task_created) break; /*If a task was created then this or the next item might be corrupted*/
2019-04-04 07:15:40 +02:00
LV_GC_ROOT(_lv_task_act) = next; /*Load the next task*/
2018-06-19 09:49:58 +02:00
}
} while(!end_flag);
2017-12-20 00:47:37 +01:00
busy_time += lv_tick_elaps(handler_start);
uint32_t idle_period_time = lv_tick_elaps(idle_period_start);
if(idle_period_time >= IDLE_MEAS_PERIOD) {
2019-06-06 06:05:40 +02:00
idle_last = (uint32_t)((uint32_t)busy_time * 100) / IDLE_MEAS_PERIOD; /*Calculate the busy percentage*/
idle_last = idle_last > 100 ? 0 : 100 - idle_last; /*But we need idle time*/
2019-04-04 07:15:40 +02:00
busy_time = 0;
2017-12-20 00:47:37 +01:00
idle_period_start = lv_tick_get();
}
2018-07-25 17:57:08 +02:00
2019-04-04 07:15:40 +02:00
task_handler_mutex = false; /*Release the mutex*/
2018-10-05 17:22:49 +02:00
LV_LOG_TRACE("lv_task_handler ready");
2017-11-23 20:42:14 +01:00
}
/**
2019-05-17 08:29:38 +02:00
* Create an "empty" task. It needs to initialzed with at least
* `lv_task_set_cb` and `lv_task_set_period`
* @return pointer to the craeted task
2017-11-23 20:42:14 +01:00
*/
2019-05-17 08:29:38 +02:00
lv_task_t * lv_task_create_basic(void)
2017-11-23 20:42:14 +01:00
{
2019-05-17 08:29:38 +02:00
lv_task_t * new_task = NULL;
2018-06-19 09:49:58 +02:00
lv_task_t * tmp;
2018-02-15 13:24:56 +01:00
/*Create task lists in order of priority from high to low*/
tmp = lv_ll_get_head(&LV_GC_ROOT(_lv_task_ll));
2019-05-17 08:29:38 +02:00
/*It's the first task*/
if(NULL == tmp) {
new_task = lv_ll_ins_head(&LV_GC_ROOT(_lv_task_ll));
lv_mem_assert(new_task);
if(new_task == NULL) return NULL;
}
/*Insert the new task to proper place according to its priority*/
else {
2018-06-19 09:49:58 +02:00
do {
2019-05-17 08:29:38 +02:00
if(tmp->prio <= DEF_PRIO) {
new_task = lv_ll_ins_prev(&LV_GC_ROOT(_lv_task_ll), tmp);
lv_mem_assert(new_task);
if(new_task == NULL) return NULL;
2018-02-15 13:24:56 +01:00
break;
}
tmp = lv_ll_get_next(&LV_GC_ROOT(_lv_task_ll), tmp);
2018-06-19 09:49:58 +02:00
} while(tmp != NULL);
2018-02-15 13:24:56 +01:00
2019-05-17 08:29:38 +02:00
/*Only too high priority tasks were found. Add the task to the end*/
if(tmp == NULL) {
new_task = lv_ll_ins_tail(&LV_GC_ROOT(_lv_task_ll));
lv_mem_assert(new_task);
if(new_task == NULL) return NULL;
2018-02-15 13:24:56 +01:00
}
}
2019-06-06 06:05:40 +02:00
new_task->period = DEF_PERIOD;
new_task->task_cb = NULL;
new_task->prio = DEF_PRIO;
2019-05-17 08:29:38 +02:00
new_task->once = 0;
new_task->last_run = lv_tick_get();
2017-11-24 17:48:47 +01:00
2019-06-06 06:05:40 +02:00
new_task->user_data = NULL;
task_created = true;
2019-05-17 08:29:38 +02:00
return new_task;
}
/**
* Create a new lv_task
* @param task a function which is the task itself
* @param period call period in ms unit
* @param prio priority of the task (LV_TASK_PRIO_OFF means the task is stopped)
* @param user_data custom parameter
* @return pointer to the new task
*/
lv_task_t * lv_task_create(lv_task_cb_t task_cb, uint32_t period, lv_task_prio_t prio, void * user_data)
{
lv_task_t * new_task = lv_task_create_basic();
lv_mem_assert(new_task);
if(new_task == NULL) return NULL;
lv_task_set_cb(new_task, task_cb);
lv_task_set_period(new_task, period);
lv_task_set_prio(new_task, prio);
new_task->user_data = user_data;
return new_task;
2017-11-23 20:42:14 +01:00
}
2019-05-17 07:58:47 +02:00
/**
* Set the callback the task (the function to call periodically)
* @param task pointer to a task
* @param task_cb teh function to call periodically
*/
void lv_task_set_cb(lv_task_t * task, lv_task_cb_t task_cb)
{
task->task_cb = task_cb;
}
2017-11-23 20:42:14 +01:00
/**
2017-11-24 17:48:47 +01:00
* Delete a lv_task
2019-05-17 08:29:38 +02:00
* @param task pointer to task created by task
2017-11-23 20:42:14 +01:00
*/
2019-05-17 08:29:38 +02:00
void lv_task_del(lv_task_t * task)
2017-11-23 20:42:14 +01:00
{
2019-05-17 08:29:38 +02:00
lv_ll_rem(&LV_GC_ROOT(_lv_task_ll), task);
2018-06-19 09:49:58 +02:00
2019-05-17 08:29:38 +02:00
lv_mem_free(task);
2018-10-15 19:00:03 +02:00
2019-05-17 08:29:38 +02:00
if(LV_GC_ROOT(_lv_task_act) == task) task_deleted = true; /*The active task was deleted*/
2017-11-23 20:42:14 +01:00
}
/**
2017-11-24 17:48:47 +01:00
* Set new priority for a lv_task
2019-05-17 08:29:38 +02:00
* @param task pointer to a lv_task
2017-11-23 20:42:14 +01:00
* @param prio the new priority
*/
2019-05-17 08:29:38 +02:00
void lv_task_set_prio(lv_task_t * task, lv_task_prio_t prio)
2017-11-23 20:42:14 +01:00
{
2019-05-17 08:29:38 +02:00
if(task->prio == prio) return;
2018-04-18 13:23:19 +02:00
/*Find the tasks with new priority*/
lv_task_t * i;
2019-04-04 07:15:40 +02:00
LV_LL_READ(LV_GC_ROOT(_lv_task_ll), i)
{
2018-04-18 13:23:19 +02:00
if(i->prio <= prio) {
2019-05-17 08:29:38 +02:00
if(i != task) lv_ll_move_before(&LV_GC_ROOT(_lv_task_ll), task, i);
2018-04-18 13:23:19 +02:00
break;
}
}
2018-02-15 13:24:56 +01:00
2018-04-18 13:23:19 +02:00
/*There was no such a low priority so far then add the node to the tail*/
if(i == NULL) {
2019-05-17 08:29:38 +02:00
lv_ll_move_before(&LV_GC_ROOT(_lv_task_ll), task, NULL);
2018-04-18 13:23:19 +02:00
}
2019-05-17 08:29:38 +02:00
task->prio = prio;
2017-11-23 20:42:14 +01:00
}
/**
2017-11-24 17:48:47 +01:00
* Set new period for a lv_task
2019-05-17 08:29:38 +02:00
* @param task pointer to a lv_task
2017-11-23 20:42:14 +01:00
* @param period the new period
*/
2019-05-17 08:29:38 +02:00
void lv_task_set_period(lv_task_t * task, uint32_t period)
2017-11-23 20:42:14 +01:00
{
2019-05-17 08:29:38 +02:00
task->period = period;
2017-11-23 20:42:14 +01:00
}
/**
2017-11-24 17:48:47 +01:00
* Make a lv_task ready. It will not wait its period.
2019-05-17 08:29:38 +02:00
* @param task pointer to a lv_task.
2017-11-23 20:42:14 +01:00
*/
2019-05-17 08:29:38 +02:00
void lv_task_ready(lv_task_t * task)
2017-11-23 20:42:14 +01:00
{
2019-05-17 08:29:38 +02:00
task->last_run = lv_tick_get() - task->period - 1;
2017-11-23 20:42:14 +01:00
}
/**
2017-11-24 17:48:47 +01:00
* Delete the lv_task after one call
2019-05-17 08:29:38 +02:00
* @param task pointer to a lv_task.
2017-11-23 20:42:14 +01:00
*/
2019-05-17 08:29:38 +02:00
void lv_task_once(lv_task_t * task)
2017-11-23 20:42:14 +01:00
{
2019-05-17 08:29:38 +02:00
task->once = 1;
2017-11-23 20:42:14 +01:00
}
/**
2018-06-19 09:49:58 +02:00
* Reset a lv_task.
2017-11-23 20:42:14 +01:00
* It will be called the previously set period milliseconds later.
2019-05-17 08:29:38 +02:00
* @param task pointer to a lv_task.
2017-11-23 20:42:14 +01:00
*/
2019-05-17 08:29:38 +02:00
void lv_task_reset(lv_task_t * task)
2017-11-23 20:42:14 +01:00
{
2019-05-17 08:29:38 +02:00
task->last_run = lv_tick_get();
2017-11-23 20:42:14 +01:00
}
/**
2017-11-24 17:48:47 +01:00
* Enable or disable the whole lv_task handling
* @param en: true: lv_task handling is running, false: lv_task handling is suspended
2017-11-23 20:42:14 +01:00
*/
2017-11-24 17:48:47 +01:00
void lv_task_enable(bool en)
2017-11-23 20:42:14 +01:00
{
2018-06-19 09:49:58 +02:00
lv_task_run = en;
2017-11-23 20:42:14 +01:00
}
/**
* Get idle percentage
2017-11-24 17:48:47 +01:00
* @return the lv_task idle in percentage
2017-11-23 20:42:14 +01:00
*/
2017-11-24 17:48:47 +01:00
uint8_t lv_task_get_idle(void)
2017-11-23 20:42:14 +01:00
{
return idle_last;
}
/**********************
* STATIC FUNCTIONS
**********************/
/**
2018-06-19 09:49:58 +02:00
* Execute task if its the priority is appropriate
2019-05-17 08:29:38 +02:00
* @param task pointer to lv_task
2017-11-23 20:42:14 +01:00
* @return true: execute, false: not executed
*/
2019-05-17 08:29:38 +02:00
static bool lv_task_exec(lv_task_t * task)
2017-11-23 20:42:14 +01:00
{
bool exec = false;
2018-06-19 09:49:58 +02:00
2018-02-15 13:24:56 +01:00
/*Execute if at least 'period' time elapsed*/
2019-05-17 08:29:38 +02:00
uint32_t elp = lv_tick_elaps(task->last_run);
if(elp >= task->period) {
task->last_run = lv_tick_get();
2019-06-06 06:05:40 +02:00
task_deleted = false;
task_created = false;
2019-05-17 08:29:38 +02:00
if(task->task_cb) task->task_cb(task);
2018-02-15 13:24:56 +01:00
/*Delete if it was a one shot lv_task*/
2019-04-04 07:15:40 +02:00
if(task_deleted == false) { /*The task might be deleted by itself as well*/
2019-05-17 08:29:38 +02:00
if(task->once != 0) {
lv_task_del(task);
2019-04-04 07:15:40 +02:00
}
2018-10-15 19:00:03 +02:00
}
2018-02-15 13:24:56 +01:00
exec = true;
2017-11-23 20:42:14 +01:00
}
2018-06-19 09:49:58 +02:00
2017-11-23 20:42:14 +01:00
return exec;
}