FreeRTOS的任务优先级、Tick以及状态讲解(尊敬的嵌入式工程师,不妨进来喝杯茶)

任务优先级和Tick

        在FreeRTOS中,任务的优先级和Tick是两个关键的概念,它们直接影响任务的调度和执行。

任务优先级

        每个任务都被分配一个优先级,用于决定任务在系统中的调度顺序。

        优先级是一个无符号整数,通常从0开始,数值越小,优先级越高。最高优先级是0,最低是configMAX_PRIORITIES - 1。

        优先级的取值范围是:0~(configMAX_PRIORITIES – 1),数值越大优先级越高。

        FreeRTOS的调度器可以使用2种方法来快速找出优先级最高的、可以运行的任务。使用不同的方法时,configMAX_PRIORITIES 的取值有所不同。

(1)通用方法
        使用C函数实现,对所有的架构都是同样的代码。对configMAX_PRIORITIES的取值没有限制。但是configMAX_PRIORITIES的取值还是尽量小,因为取值越大越浪费内存,也浪费时间。configUSE_PORT_OPTIMISED_TASK_SELECTION被定义为0、或者未定义时,使用此方法。
(2)架构相关的优化的方法
        架构相关的汇编指令,可以从一个32位的数里快速地找出为1的最高位。使用这些指令,可以快速找出优先级最高的、可以运行的任务。使用这种方法时,configMAX_PRIORITIES的取值不能超过32。configUSE_PORT_OPTIMISED_TASK_SELECTION被定义为1时,使用此方法。

        在FreeRTOS中,优先级为0的任务是IDLE任务,用于在没有其他任务执行时占用CPU。

        任务的创建时通过指定参数设置任务的优先级,如下所示:

xTaskCreate(vTask1, "Task1", configMINIMAL_STACK_SIZE, NULL, 1, &Task1_Handle);

Tick

        Tick是FreeRTOS内核用于计时和任务调度的基本时间单元。

        FreeRTOS通过硬件定时器或者软件定时器,每隔一定时间(即Tick间隔)产生一次中断。

        Tick的间隔由configTICK_RATE_HZ定义,它表示每秒钟产生的Tick数。

关系

        任务的调度是基于优先级的,具有更高优先级的任务将在具有较低优先级的任务之前执行。

        Tick间隔决定了系统时钟的精度,同时也影响了任务的延时和时间控制。

        任务在等待一定时间或进行延时时,使用vTaskDelay()等函数,参数是以Tick为单位的时间。

vTaskDelay(100 / portTICK_PERIOD_MS); // 暂停任务100ms

        任务的优先级和Tick的概念结合在一起,形成了FreeRTOS中任务调度和时间控制的基础。通过调整任务的优先级和配置Tick的间隔,可以灵活地控制系统中任务的执行顺序和时间行为。

        以下是一个基于FreeRTOS的STM32F103芯片的简单优先级实验案例代码。该例子创建了两个任务,分别以不同的优先级运行,以演示优先级如何影响任务的调度。

#include "stm32f1xx.h"
#include "FreeRTOS.h"
#include "task.h"

// 函数原型
void SystemClock_Config(void);
static void MX_GPIO_Init(void);

// 任务函数
TaskHandle_t Task1_Handle, Task2_Handle;

void vTask1(void *pvParameters) {
    while (1) {
        // 任务1的处理逻辑
        HAL_GPIO_TogglePin(GPIOC, GPIO_PIN_13); // 切换LED状态
        vTaskDelay(1000 / portTICK_PERIOD_MS); // 每隔1秒执行一次
    }
}

void vTask2(void *pvParameters) {
    while (1) {
        // 任务2的处理逻辑
        vTaskDelay(2000 / portTICK_PERIOD_MS); // 每隔2秒执行一次
    }
}

int main(void) {
    // 硬件初始化
    HAL_Init();
    SystemClock_Config();
    MX_GPIO_Init();

    // 创建任务1(优先级1)
    xTaskCreate(vTask1, "Task1", configMINIMAL_STACK_SIZE, NULL, 1, &Task1_Handle);

    // 创建任务2(优先级2)
    xTaskCreate(vTask2, "Task2", configMINIMAL_STACK_SIZE, NULL, 2, &Task2_Handle);

    // 启动调度器
    vTaskStartScheduler();

    // 代码不应该运行到这里,如果运行到这里说明发生了错误
    while (1) {
        // 处理错误
    }
}

// 系统时钟配置
void SystemClock_Config(void) {
}

// GPIO初始化
static void MX_GPIO_Init(void) {
}

        在这个例子中,任务1和任务2分别在不同的时间间隔内切换LED的状态。任务1的优先级较高,它每隔1秒执行一次,而任务2的优先级较低,它每隔2秒执行一次。

任务状态

        在FreeRTOS中,较高优先级的任务将优先于较低优先级的任务执行。任务优先级是通过任务创建时指定的参数来设置的(在xTaskCreate函数的第5个参数)。在这个案例中,任务1的优先级为1,任务2的优先级为2。

        FreeRTOS中的任务可以处于不同的状态,这些状态反映了任务在系统中的执行阶段。任务的状态是通过eTaskState枚举类型表示的,定义在task.h头文件中。以下是任务可能的状态:

        eRunning(运行中): 任务正在执行。这是任务处于活动状态的时候。

        eReady(就绪): 任务已经准备好执行,但由于有其他高优先级任务在运行,该任务暂时没有被调度。

        eBlocked(阻塞): 任务由于等待某些事件而被阻塞,例如等待定时器、消息队列、信号量等。

        eSuspended(挂起): 任务被显式地挂起,不再参与调度。

        eDeleted(已删除): 任务已被删除,但其资源尚未被释放。

        eInvalid(无效): 无效状态,通常用于表示错误或未知状态。

你可以使用eTaskGetState函数来获取任务的当前状态。以下是一个示例代码:

#include "FreeRTOS.h"
#include "task.h"

TaskHandle_t xTaskHandle;

void vTaskFunction(void *pvParameters) {
    while (1) {
        // 任务的处理逻辑
    }
}

int main(void) {
    // 创建任务
    xTaskCreate(vTaskFunction, "Task", configMINIMAL_STACK_SIZE, NULL, 1, &xTaskHandle);
    // 启动调度器
    vTaskStartScheduler();
    // 代码不应该运行到这里,如果运行到这里说明发生了错误
    while (1) {
        // 处理错误
    }
}

void vApplicationIdleHook(void) {
    // 空闲钩子函数
    eTaskState taskState = eTaskGetState(xTaskHandle);

    switch (taskState) {
        case eRunning:
            // 任务正在运行
            break;
        case eReady:
            // 任务就绪
            break;
        case eBlocked:
            // 任务被阻塞
            break;
        case eSuspended:
            // 任务被挂起
            break;
        case eDeleted:
            // 任务已删除
            break;
        case eInvalid:
            // 无效状态
            break;
    }
}

        在这个例子中,vApplicationIdleHook是一个空闲钩子函数,用于在系统空闲时检查任务的状态。函数eTaskGetState返回任务的当前状态,然后可以根据任务的状态进行相应的处理。

阻塞状态(Blocked State)

        在FreeRTOS中,任务的阻塞状态(Blocked State)表示任务由于等待某些事件而无法执行。任务可以因为多种原因而被阻塞,包括等待定时器、等待消息队列、等待信号量等。当任务处于阻塞状态时,它将不会被调度执行,直到满足了其阻塞条件。

        以下是几个常见的阻塞状态的示例:

        等待定时器:

// 创建定时器
TimerHandle_t xTimer = xTimerCreate("MyTimer", pdMS_TO_TICKS(1000), pdTRUE, 0, vTimerCallback);

// 启动定时器,在任务中使用xTimerPendFunctionCallFromISR等函数
xTimerStart(xTimer, 0);

        在任务中可能会使用xTimerPendFunctionCallFromISR等函数等待定时器的超时事件。

        等待消息队列:

// 创建消息队列
QueueHandle_t xQueue = xQueueCreate(5, sizeof(int));

// 在任务中等待消息队列
xQueueReceive(xQueue, &data, portMAX_DELAY);

        任务通过xQueueReceive函数等待消息队列中的消息,如果队列为空,任务将被阻塞,直到队列中有数据。

        等待信号量:

// 创建信号量
SemaphoreHandle_t xSemaphore = xSemaphoreCreateBinary();

// 在任务中等待信号量
xSemaphoreTake(xSemaphore, portMAX_DELAY);

        任务通过xSemaphoreTake函数等待信号量,如果信号量被其他任务取得,当前任务将被阻塞,直到信号量可用。

        在上述示例中,portMAX_DELAY表示任务将一直等待,直到满足其阻塞条件为止。任务也可以使用超时值来设置阻塞的最大等待时间。一旦阻塞条件得到满足,任务将被置为就绪状态,等待调度器调度执行。

暂停状态(Suspended State)

        在FreeRTOS中,任务的暂停状态(Suspended State)表示任务被显式地挂起,使得该任务不再参与调度,即不会被执行。任务在创建后可以通过调用vTaskSuspend()函数将其挂起,然后通过调用vTaskResume()函数将其恢复执行。

以下是一个简单的示例代码,演示了任务的暂停和恢复:

#include "FreeRTOS.h"
#include "task.h"

TaskHandle_t xTaskHandle;

void vTaskFunction(void *pvParameters) {
    while (1) {
        // 任务的处理逻辑
    }
}

int main(void) {
    // 创建任务
    xTaskCreate(vTaskFunction, "Task", configMINIMAL_STACK_SIZE, NULL, 1, &xTaskHandle);

    // 启动调度器
    vTaskStartScheduler();

    // 代码不应该运行到这里,如果运行到这里说明发生了错误
    while (1) {
        // 处理错误
    }
}

// 在某个事件或条件下挂起任务
void vSuspendTask(void) {
    vTaskSuspend(xTaskHandle);
}

// 在某个事件或条件下恢复任务
void vResumeTask(void) {
    vTaskResume(xTaskHandle);
}

        在这个例子中,vTaskSuspend()函数用于挂起任务,vTaskResume()函数用于恢复任务。通常,这样的操作可以用于在某些条件满足或事件发生时挂起任务,然后在其他条件下恢复任务的执行。

        需要注意的是,挂起任务不会立即停止任务的执行,而是在任务下一次被调度执行时生效。此外,挂起任务并不会释放任务所占用的资源,因此在使用这些函数时需要谨慎,以免引起资源泄漏。

就绪状态(Ready)

        在FreeRTOS中,任务的就绪状态(Ready State)表示任务已经准备好被调度执行,但由于有其他高优先级的任务正在运行,该任务暂时还未被调度。

        在FreeRTOS中,任务被创建后,它的状态一开始就是就绪状态,等待调度器来选择合适的时机执行它。任务的就绪状态可以由以下几种情况触发:

        任务创建: 任务被创建后,会进入就绪状态,等待调度执行。

xTaskCreate(vTaskFunction, "Task", configMINIMAL_STACK_SIZE, NULL, 1, &xTaskHandle);

        任务挂起状态解除: 任务从挂起状态(Suspended State)被恢复后,会进入就绪状态。

vTaskResume(xTaskHandle);

        等待事件结束: 任务等待某个事件的发生,一旦事件发生,任务从阻塞状态切换到就绪状态。

xSemaphoreTake(xSemaphore, portMAX_DELAY);

        任务等待定时器超时: 任务等待一个定时器超时,一旦超时,任务从阻塞状态切换到就绪状态。

xTaskDelay(1000 / portTICK_PERIOD_MS);

        在以上例子中,xTaskCreate创建了一个任务,vTaskResume恢复了一个挂起的任务,xSemaphoreTake和xTaskDelay是任务等待事件或超时的示例。

        任务的就绪状态是由FreeRTOS调度器根据任务的优先级和调度算法来决定的。当调度器判定某个任务处于就绪状态时,该任务将会被调度器选中,并执行相应的任务函数。

任务状态转换图

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/233739.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

使用大语言模型生成自动驾驶指令代码可行吗?

大语言模型最近太火了,大家都在各个方向上应用它。自动驾驶也是当下一个热门领域,两个热门领域的结合自然非常令人期待。AIGCer在读了一些相关文献后,感觉在自动驾驶这个热点方向上应用大语言模型,也将是一个很有前途的方向。 这…

设计模式篇---享元模式

文章目录 概念结构实例总结 概念 享元模式:运用共享技术有效地支持大量细粒度对象的复用。 当系统中出现大量相同或者相似的对象时,可以考虑使用享元模式。享元模式中提供了一个享元池用于存储已经创建好的对象。享元对象能做到共享的关键是区分了内部状…

Diary22-全网最全的CSS3.0讲解

CSS学习 1.认识CSS 1.1什么是CSS CSS:Cascading Style Sheet——层叠级联样式表 CSS:表现(美化网页) 字体;颜色;边距;高度;宽度;背景图片;网页定位&…

Nginx按指定格式记录访问日志以及利用logrotate做日志轮转

今天突然想起来一个日志的一个东西,因为拉项目无意中看到了日志文件的一些东西,现在不经常做后端了,加上其他的一些原因吧.有时候有些问题也没想太多,马马虎虎就过了,后来想想还是要记录一下这方面的处理过程吧: 一般我们作为开发人员关注的日志只是在应用程序层面的,我们称它…

springboot095学生宿舍信息的系统

springboot095学生宿舍信息的系统 源码获取: https://docs.qq.com/doc/DUXdsVlhIdVlsemdX

使用MfgTool烧写工具烧写自制系统

一. 简介 本文我们就来学习,如何将我们编译的 uboot,zImage(内核镜像),xxx.dtb设备树文件,还有制作的根文件系统,这四个文件烧写到开发板中,最后 开发板能正常启动。 上一篇文章说…

Linux权限(用户角色+文件权限属性)

Linux权限 文章目录 Linux权限一.文件权限1.快速掌握修改权限的方法(修改文件权限属性)2.对比权限的有无,以及具体的体现3.修改权限的第二套方法(修改用户角色)4.文件类型(Linux下一切皆文件) 二…

六级高频词汇1

目录 高频词汇 参考连接 高频词汇 1. alter v. 改变,改动,变更 2. burst vi. n. 突然发生,爆裂 3. dispose vi. 除掉;处置;解决;处理(of) 4. blast n. 爆炸;气流 vi. 炸,炸掉 …

Python轴承故障诊断 (五)基于EMD-LSTM的故障分类

目录 前言 1 经验模态分解EMD的Python示例 2 轴承故障数据的预处理 2.1 导入数据 2.2 制作数据集和对应标签 2.3 故障数据的EMD分解可视化 2.4 故障数据的EMD分解预处理 3 基于EMD-LSTM的轴承故障诊断分类 3.1 训练数据、测试数据分组,数据分batch 3.2 定…

C++ 面向对象补充

目录 初始化列表 explicit关键字 单参数构造函数 多参数构造函数 static成员 友元 内部类(不常用) 匿名对象 初始化列表 以一个冒号开始,接着是一个以逗号分隔的数据成员列表,每个成员变量后面跟 一个放在括号中的初始值或表达式。 class Date…

头歌平台题目参考

任务描述 本关任务:获取从键盘输入3个数,要求按从大到小的顺序输出这3个数。 相关知识 程序并不会总是执行同样的处理。例如,按下某个键的时候执行 A 处理,按下其他键的时候执行 B 处理…… 像这样,程序通过条件判断…

回溯算法之N皇后

一 什么是回溯算法 回溯算法(Backtracking Algorithm)是一种用于解决组合优化问题的算法,它通过逐步构建候选解并进行验证,以寻找所有满足特定条件的解。回溯算法通常应用于在给定约束条件下枚举所有可能解的问题,如…

FaceBook推出新的翻译模型Seamless!可实现跨语言交流的无缝衔接!

FaceBook **(中文名:脸书)**近期发布了一个新的翻译模型 Seamless Communication,可实现跨语言实时"无缝"交流。 该模型可以保留跨语言的表达方式和复杂性(翻译时保留语音中的停顿和语速,以及声…

优雅草蜻蜓I即时通讯·水银版私有化部署之安卓Android端编译-02

Android 项目配置 添加图片注释,不超过 140 字(可选) 使用以上Android studio版本 添加图片注释,不超过 140 字(可选) 下载最低sdk最低版本28 完成后就可以导入项目(项目导入不能开VPN,会导致部分三方库…

Glibc之malloc实现原理

前言导入 内存管理之虚拟内存空间 详细了解这部分知识,再看下面的内容会很舒服 进程地址空间 在32位Linux系统中,进程地址空间是这样分布的。其中内核空间独占1G,不允许用户操作,其余3G由用户操作。malloc的操作对象&#xff1…

C语言之函数设计(1)

目录 没有返回值的函数 通用性 不含形参的函数 函数返回值的初始化 作用域 文件作用域 声明和定义 函数原型声明 头文件和文件包含指令 在上节中我们简单的学习了函数的创建方法(函数定义)与函数的使用方法(函数调用)&…

谈一谈网络协议中的应用层

文章目录 一,什么是HTTPHTTP的优缺点HTTPS 一,什么是HTTP 我们在通过网络进行传输数据时,我们要保证,我们在发送时构造的数据,在接收时也能够解析出来,这本质上就是一种协议,是一种应用层协议&…

python zblog API实现类似XMLRPC/发布文章

我发现python对Zblog的XML发布并不友好,虽然也有对应的模块,但是远远没有XPCRPC更直接方便,但是使用xmlRpc是直接给发布文章带来了不小的便利,但是对系统也并不友好,但是zblog也开放了Api,但是干部子弟不乐…

测试剪切板贴图,兼测试2023年12月7日更新的Bard

当前的情况好比,(居然真的可以通过剪切板把图片放进来!) 听说2023年12月7日Bard有更新,所以,再测试了一次。这下,对大语言模型应该死心了;AI替代人的传闻应该是过早危言耸听了。

SAP UI5 walkthrough step3 Controls

在上一步&#xff0c;我们是直接用index.html 中的body 里面的DIVision去输出 hello world&#xff0c; 在这个章节&#xff0c;我们将用SAP UI5 的标准控件 sap/m/Text 首先&#xff0c;我们去修改 webapp/index.html <!DOCTYPE html> <html> <head><…