FreeRTOS学习笔记(二)任务基础篇

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档

文章目录

  • 前言
  • 一、 任务的基本内容
    • 1.1 任务的基本特点
    • 1.2 任务的状态
    • 1.3 任务控制块——任务的“身份证”
  • 二、 任务的实现
    • 2.1 定义任务函数
    • 2.2 创建任务
    • 2.3 启动任务调度器
    • 2.4 任务的运行与切换
      • 2.4.1 利用延时函数
      • 2.4.2 利用中断
    • 2.5 任务的通信与同步
    • 2.6 任务的删除
    • 2.7 任务的通知(可选)
    • 2.8 任务挂起和恢复
  • 三、相关辅助API函数


前言

  在FreeRTOS中,任务(Task) 是基本的执行单位,每个任务代表一个独立的线程,可以并行执行,管理系统的各项操作。任务是FreeRTOS的核心概念,了解任务的工作原理和管理方式是开发实时多任务系统的基础。


一、 任务的基本内容

1.1 任务的基本特点

  在 FreeRTOS 中,任务是系统运行的基本单位,每个任务都有自己独立的执行逻辑和特点。以下是任务的一些主要特点:

  1. 独立性
      每个任务在 FreeRTOS 中是独立的。任务之间不共享堆栈和程序计数器,每个任务都有自己的上下文,包括寄存器状态和堆栈。由于任务是独立的,任务切换不会影响其他任务的执行状态。
  2. 优先级
      FreeRTOS 支持多任务的优先级调度。每个任务在创建时可以设置优先级,高优先级的任务可以抢占低优先级任务的执行。优先级决定了任务的调度顺序,高优先级任务会优先执行,但不会一直独占 CPU。
  3. 周期性和响应性
      任务可以设置为周期性执行(例如,每隔一定时间运行一次),也可以设置为响应外部事件(如中断、信号量释放等)。
      响应时间可以通过任务的优先级和延时函数(如 vTaskDelay)来控制。
  4. 上下文切换
      FreeRTOS 通过上下文切换来调度任务。上下文切换是指从一个任务切换到另一个任务时,保存当前任务的执行状态(如寄存器内容)并恢复另一个任务的执行状态。上下文切换可以由任务自身调用(例如调用 taskYIELD),也可以由操作系统根据调度策略自动执行。

1.2 任务的状态

  一般来说 FreeRTOS中的任务永远是处于运行态、就绪态、阻塞态、挂起态和终止态四种状态。

  • 运行态(Running):当前被调度器选择执行的任务。
  • 就绪态(Ready):任务可以执行,但调度器尚未选择它。
  • 阻塞态(Suspended):是指一个任务当前正在等待某个外部事件,比如说如果某个任务调用了函数 vTaskDelay()的话就会进入阻塞态,直到延时周期完成。任务在等待队列、信号量、事件组、通知或互斥信号量的时候也会进入阻塞态。任务进入阻塞态会有一个超时时间,当超过这个超时时间任务就会退出阻塞态,即使所等待的事件还没有来临!
  • 挂起态(Suspended):是当任务进入挂起态以后也不能被调度器调用,但是进入挂起态的任务没有超时时间。任务进入和退出挂起态通过调用函数 vTaskSuspend()和 xTaskResume()。
    在这里插入图片描述

1.3 任务控制块——任务的“身份证”

  FreeRTOS 的每个任务都有一些属性需要存储,FreeRTOS 把这些属性集合到一起用一个结构体来表示,这个结构体叫做任务控制块:TCB_t,在使用函数 xTaskCreate()创建任务的时候就会自动的给每个任务分配一个任务控制块。此结构体在文件 tasks.c 中有定义,如下

typedef struct tskTaskControlBlock
{
volatile StackType_t *pxTopOfStack; // 任务堆栈栈顶,必须是TCB的第一个成员
ListItem_t xStateListItem; 			// 状态列表项
ListItem_t xEventListItem;			// 事件列表项
UBaseType_t uxPriority; 			// 任务优先级
StackType_t *pxStack; 				// 任务堆栈起始地...
} tskTCB;

二、 任务的实现

  在FreeRTOS中,任务的实现过程涉及从任务的定义到任务的调度运行,通常包括以下几个关键步骤。

函数名描述
vTaskFunction( )定义任务函数
xTaskCreate( )动态创建任务
xTaskCreateStatic( )静态创建任务
vTaskStartScheduler( )启动任务调度器
vTaskDelete( )删除创建的任务
vTaskSuspend( )挂起一个任务
vTaskResume( )恢复一个任务
xTaskResumeFromISR( )中断服务函数中恢复一个任务

2.1 定义任务函数

  任务函数本质也是函数,所以肯定有任务名什么的,不过这里我们要注意:任务函数的返回类型一定要为 void 类型,也就是无返回值,而且任务的参数也是 void 指针类型的!任务函数名可以根据实际情况定义。任务的具体执行过程是一个大循环,for(; ; )就代表一个循环,作用和 while(1)一样,循环里面就是真正的任务代码了,此任务具体要干的活就在这里实现!

void vTaskFunction(void *pvParameters) {for (;;) {// 执行任务的操作// 例如:读取传感器数据,处理输入,控制输出等}
}

2.2 创建任务

  任务的创建通常是在系统初始化时完成的。你可以通过调用 xTaskCreate 函数来动态创建任务,任务控制块及任务栈空间均由freertos调度。

// configSUPPORT_DYNAMIC_ALLOCATION需要设置为1
// 成功:pdPASS
BaseType_t xTaskCreate(TaskFunction_t pvTaskCode,   // 任务函数的指针const char * const pcName,   // 任务名称(仅用于调试)uint16_t usStackDepth,       // 任务堆栈大小(以字为单位)void *pvParameters,          // 传递给任务的参数UBaseType_t uxPriority,      // 任务优先级(数值越大,优先级越高)TaskHandle_t *pxCreatedTask  // 任务句柄(用于管理任务)
);

  在一些内存资源有限的系统中,使用函数xTaskCreateStatic( )静态创建任务,允许你预先分配任务所需的内存,并将其传递给函数。但需要程序员自行定义任务堆栈,然后堆栈首地址作为函数的参数 puxStackBuffer 传递给函数。

// configSUPPORT_STATIC_ALLOCATION需要设置为1
// 成功:任务句柄
// 失败:NULL
TaskHandle_t xTaskCreateStatic(TaskFunction_t pvTaskCode,    // 任务函数的指针const char * const pcName,    // 任务名称(仅用于调试)uint32_t ulStackDepth,        // 任务堆栈大小(以字为单位)void *pvParameters,           // 传递给任务的参数UBaseType_t uxPriority,       // 任务优先级StackType_t *pxStackBuffer,   // 指向任务堆栈的指针StaticTask_t *pxTaskBuffer    // 指向任务控制块的指针
);

你 不需要手动配置 PendSV 和 SysTick 寄存器,因为 FreeRTOS 的移植层代码会在调度器启动时为你配置它们。你只需要确保时钟频率和中断优先级配置正确,系统即可正常启动任务调度器。

2.3 启动任务调度器

  通常情况下,我们都是在main()函数中先创建一个开始任务 start_task,后面紧接着调用函数 vTaskStartScheduler()。这个函数的功能就是开启任务调度器的。调度器负责管理任务的执行,决定哪个任务将被CPU执行。当调度器启动后,系统将进入多任务模式,各个任务根据优先级和调度策略被执行。

vTaskStartScheduler( );

  值得注意的是,在实际启动 FreeRTOS 任务调度器时,移植层代码会在调度器启动时为你配置PendSV 和 SysTick 寄存器。你只需要确保时钟频率和中断优先级配置正确,系统即可正常启动任务调度器。即使你没有手动处理就绪列表中断管理,调度器也会在背后自动管理这些任务调度工作。只要调用了 vTaskStartScheduler(),FreeRTOS 内核就会处理所有的中断、任务状态转换和任务切换,因此不会影响系统的正常运行。这种自动化特性使得 FreeRTOS 使用起来更加方便。

2.4 任务的运行与切换

  任务切换是通过上下文切换机制实现的,即在不同任务之间切换 CPU 控制权,使得多个任务可以看起来像是并行运行。任务切换由任务调度器(Scheduler)管理,根据任务的优先级、时间片和状态决定下一个要运行的任务。任务被调度器选择执行时,将进入运行状态(Running)。在FreeRTOS中,任务的运行通常是一个无限循环。在这个循环中,任务可以执行各种操作,如处理输入、控制输出、进行计算等。
  当任务执行FreeRTOS 的延时函数 vTaskDelay、等待信号量、等待队列消息或发生其他阻塞条件时,任务将进入阻塞状态(Blocked)。在阻塞状态下,任务会释放CPU,允许其他任务执行。

2.4.1 利用延时函数

  当函数 vTaskDelay()是相对模式(相对延时函数),在文件 tasks.c 中有定义,要使用此函数的话宏 INCLUDE_vTaskDelay 必须为 1。

void vTaskFunction(void *pvParameters) {for (;;) {// 执行任务操作// 延时时间由参数 xTicksToDelay 来确定,为要延时的时间节拍数vTaskDelay(pdMS_TO_TICKS(1000));  // 延迟1秒}
}

  函数 vTaskDelayUntil()是绝对模式(绝对延时函数),阻塞时间是一个绝对时间,那些需要按照一定的频率运行的任务可以使用函数 vTaskDelayUntil(),其原型如下:

void vTaskDelayUntil( TickType_t * const pxPreviousWakeTime,  const TickType_t xTimeIncrement )
  • 参数:
    • pxPreviousWakeTime: 上一次任务延时结束被唤醒的时间点,任务中第一次调用函数vTaskDelayUntil 的话需要将 pxPreviousWakeTime 初始化进入任务的while()循环体的时间点值。在以后的运行中函数 vTaskDelayUntil()会自动更新 pxPreviousWakeTime。
    • xTimeIncrement:任务需要延时的时间节拍数(相对于 pxPreviousWakeTime 本次延时的节拍数)。

2.4.2 利用中断

  除此之外,我们也可以利用中断触发的任务切换,利用FreeRTOS 中的一个宏portEND_SWITCHING_ISR() ,在中断服务程序(ISR)结束时请求任务切换。它的主要作用是在处理完中断后,如果有更高优先级的任务进入就绪状态,通知调度器进行任务切换,确保高优先级任务能够及时执行。
  在中断服务程序(ISR)中,FreeRTOS 通常会进行某些操作,例如发送信号量、消息队列或者处理事件。这些操作可能会使某个更高优先级的任务变为就绪状态。在这种情况下,你需要通过 portEND_SWITCHING_ISR() 来检查是否需要任务切换。

void ISR_Handler(void)
{/* 该参数用于指示是否有更高优先级的任务变为就绪状态;如有更高优先级的任务被唤醒,该变量会被设置为pdTRUE。*/BaseType_t xHigherPriorityTaskWoken = pdFALSE;// 处理中断,可能会唤醒某个任务xSemaphoreGiveFromISR(xSemaphore, &xHigherPriorityTaskWoken);// 如果有高优先级任务需要执行,它会触发 PendSV 中断,进行上下文切换portEND_SWITCHING_ISR(xHigherPriorityTaskWoken);
}

  这里顺便提一下xSemaphoreGiveFromISR( ),它用于从中断服务程序(ISR)中释放信号量的函数。如果有任务因为等待该信号量而阻塞(调用了xSemaphoreTake()),该函数会将最高优先级的等待任务移入就绪状态;此外通过 xHigherPriorityTaskWoken 参数,指示在中断结束后是否需要进行任务切换,以确保高优先级任务能及时执行。

BaseType_t xSemaphoreGiveFromISR( SemaphoreHandle_t xSemaphore, BaseType_t *pxHigherPriorityTaskWoken );
  • 参数说明:
    • xSemaphore:这是要释放的信号量句柄,它是由 xSemaphoreCreateBinary() 或其他信号量创建函数生成的;
    • pxHigherPriorityTaskWoken:这是一个指向 BaseType_t 类型变量的指针。当信号量释放后,如果有一个更高优先级的任务被唤醒,FreeRTOS 会将该变量设置为 pdTRUE,指示调度器在中断结束时需要进行任务切换。
  • 返回值:
    • 成功:pdTRUE;
    • 失败:pdFALSE。

  值得注意的是,xSemaphoreGiveFromISR 不会直接返回是否需要进行上下文切换,而是通过一个 BaseType_t 类型的变量 xHigherPriorityTaskWoken 传递这个信息。如果在 ISR 中释放信号量后,有一个比当前任务优先级更高的任务被唤醒xHigherPriorityTaskWoken 会被设置为 pdTRUE,你需要根据这个变量来决定是否切换到高优先级任务。

  在 FreeRTOS 中,我们除了可以使用xSemaphoreGiveFromISR,也可以使用xTaskResumeFromISR 函数,二者都是从 ISR 中唤醒任务的常见方法,但它们的机制稍有不同。xTaskResumeFromISR 的返回值是一个布尔值 pdTRUE 或 pdFALSE,表示是否需要进行上下文切换,你可以直接使用该返回值来决定是否调用 portYIELD_FROM_ISR() 或 portEND_SWITCHING_ISR()。

BaseType_t xTaskResumeFromISR(TaskHandle_t xTaskToResume);
  • 参数说明
    • xTaskToResume:任务句柄,即要恢复的任务。这个任务必须已经被挂起(通过调用 vTaskSuspend 挂起)。
  • 返回值
    • pdTRUE:如果恢复任务后需要进行上下文切换(即被恢复的任务优先级高于当前任务优先级),返回 pdTRUE。
    • pdFALSE:如果恢复任务后不需要进行上下文切换(即当前任务的优先级不低于被恢复的任务优先级),返回 pdFALSE。

2.5 任务的通信与同步

  在多任务系统中,任务之间需要通信和同步,以协调各个任务的操作。FreeRTOS提供了多种机制来实现任务间的通信与同步,包括队列(Queue)、信号量(Semaphore)、事件组(Event Group)等。本节仅作简单了解,后续会详细介绍这部分。

void vSenderTask(void *pvParameters) {for (;;) {// 向队列发送数据xQueueSend(queueHandle, &data, portMAX_DELAY);vTaskDelay(pdMS_TO_TICKS(500)); // 延迟500毫秒}
}void vReceiverTask(void *pvParameters) {for (;;) {// 从队列接收数据xQueueReceive(queueHandle, &data, portMAX_DELAY);// 处理接收到的数据}
}

2.6 任务的删除

  任务函数一般不允许跳出循环,当任务完成其使命或不再需要时,如果一定要跳出循环的话在跳出循环以后一定要调用函数 vTaskDelete(NULL)删除此任务!任务删除后,所占用的资源(如堆栈内存)将被释放。

void vTaskFunction(void *pvParameters) {for (;;) {// 执行任务操作if (某些条件满足) {vTaskDelete(NULL);  // 删除任务自身}}
}

2.7 任务的通知(可选)

  FreeRTOS提供了一种轻量级的任务通知机制,允许任务之间发送通知。任务通知可以用来触发任务执行特定的操作。本节仅作简单了解,后续会详细介绍这部分。

void vTaskFunction(void *pvParameters) {for (;;) {// 等待通知ulTaskNotifyTake(pdTRUE, portMAX_DELAY);// 收到通知后执行操作}
}// 另一任务或中断可以发送通知
xTaskNotify(taskHandle, 0x01, eSetBits);

2.8 任务挂起和恢复

  任务的挂起和恢复常用于让某个任务在等待某种外部事件(例如接收到信号量或数据)时暂停运行,直到事件发生后再恢复。

  • 任务挂起:调用 vTaskSuspend(TaskHandle_t xTask) 可以挂起某个指定任务。挂起的任务将不会被任务调度器分配 CPU 时间,直到明确调用 vTaskResume(TaskHandle_t xTask) 恢复该任务。
  • 任务恢复:调用 vTaskResume() 恢复被挂起的任务,该任务将重新参与调度。
函数描述
vTaskSuspend( )挂起一个任务
vTaskResume( )恢复一个任务
xTaskResumeFromISR( )中断服务函数中恢复一个任务

三、相关辅助API函数

  FreeRTOS 还有很多与任务相关的 API 函数,不过这些 API函数大多都是辅助函数了,本小节我们就来看一下这些与任务相关的其他的 API 函数。
在这里插入图片描述

免责声明:本文参考了网上公开资料,仅用于学习交流,若有错误或侵权请联系笔者。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

python安装包的三种区别

python安装包的三种区别: Download Windows x86 web-based installer Download Windows x86 executable installerDownload Windows x86 embeddable zip fileDownload Windows x86-64 web-based installerDownload Windows x86-64 executable installerDownload W…

使用mingw64 编译 QT开发流程

1. 安装QT5 QT5.12.12 安装时选择mingw的开发包 2. 使用qtdesigner 进行ui设计 生成ui文件 3. 将ui文件转换为.h 文件 uic mywindow.ui -o ui_mywindow.h代码中指向生成的 UI 对象的地方 要改成这个Form 4. 编译 创建mainwindow.cpp #include "mainwindow.h"…

Python Flask_APScheduler定时任务的正确(最佳)使用

描述 APScheduler基于Quartz的一个Python定时任务框架,实现了Quartz的所有功能。最近使用Flask框架使用Flask_APScheduler来做定时任务,在使用过程当中也遇到很多问题,例如在定时任务调用的方法中需要用到flask的app.app_context()时&#…

绍兴视角下的广州温暖:星贝育园——自闭症儿童的关怀之家

在绍兴这座充满人文情怀的城市里,人们对自闭症儿童的关注与关怀如同涓涓细流,汇聚成爱的海洋。当谈及为这些特殊孩子寻找一个温馨、专业的成长环境时,广州的星贝育园自闭症儿童寄宿制学校无疑是众多家庭心中的理想之选。这所学校以其独特的关…

代码随想录Day 42|leetcode题目:188.买卖股票的最佳时机IV、309.最佳买卖股票时机含冷冻期、714.买卖股票的最佳时机含手续费

提示:DDU,供自己复习使用。欢迎大家前来讨论~ 文章目录 题目题目一:188.买卖股票的最佳时机IV解题思路: 题目二:309.最佳买卖股票时机含冷冻期解题思路: 题目三: 714.买卖股票的最佳时机含手续…

摊牌了!一文教会你轻松上手豆包MarsCode 编程助手!

豆包MarsCode 编程助手是豆包旗下的 AI 编程助手,提供以智能代码补全为代表的 AI 功能。豆包MarsCode 编程助手支持主流的编程语言和 IDE,在开发过程中提供单行代码或整个函数的编写建议。此外,它还支持代码解释、单测生成和问题修复等功能&a…

有关采用parallelStream并行流处理List并使用自定义线程池和lettuce redis客户端一起使用的问题

在使用parallelStream进行处理list时,如不指定线程池,默认的并行度采用cpu核数进行并行,这里采用ForJoinPool来指定线程池,但循环中使用了luttuce 来获取redis的key时,出现没有控制住线程池的线程数问题。具体上代码。…

SAP B1 学习笔记 - 易混淆字段名(持续更新中)

背景 在 SAP B1 的单据中,由于同一单据时常对应着多个后台表单,且后台表单内包含的字段信息往往远大于单据显示出来的,在配置时经常出现多个字段混淆、无系统信息提示字段名模糊的情况,这里总结常见的易混淆难查找的后台字段名。…

AIGC6: 走进腾讯数字盛会

图中是一个程序员,去参加一个技术盛会。AI大潮下,五颜六色,各种不确定。 背景 AI对各行各业的冲击越来越大,身处职场的我也能清晰的感受到。 我所在的行业为全球客服外包行业。 业务模式为: 为国际跨境公司提供不同…

使用C++编写一个语音播报时钟(Qt)

要求:当系统时间达到输入的时间时,语音播报对话框中的内容。定时可以取消。qt界面如上图所示。组件如下: countdownEdit作为书写目标时间的line_edit start_btn作为开始和停止的按钮 stop_btn作为取消的按钮 systimelab显示系统时间的lab tex…

Java后端程序员简单操作Linux系统命令

Linux系统概述 Linux 内核最初是由芬兰人林纳斯托瓦兹(Linus Torvalds)在赫尔辛基大学上 学时而编写的一个开源的操作系统。 Linux(管理计算机硬件资源,任务调度)支持多用户,支持网络,支持多线…

Vue3:el-table实现日期的格式化

后端如果返回的是时间戳,需要我们进行日期格式化 例如:2024-09-11T14:19:14 定义一个日期解析的工具组件 export function formatDateAsYYYYMMDDHHMMSS(dateStr: any) {const date new Date(dateStr);const year date.getFullYear();const month S…

踩最深的坑,教会自己找到需求

目录 引言 1. 寻找合适的需求 2. 海外市场选择 3. 线下热点判断 4. 线上关注度分析 5. 当前竞争分析 6. 未来潜力分析 引言 在经历了刻骨铭心的合伙创业经历后,我意识到是时候该独立出海了。 捡起早已深埋在心里的创业想法,开始独自创业。 这次…

[笔记] 电机工作制以及软硬特性的本质推导

原始资料来源:某电机厂商 1.电机非常规操作术语和许可次数 1.1 电机操作术语 点动:通电后立即关停,最终速度不到额定转速的1/4电制动:制动到额定转速的1/3逆转:也就是打反车,不等停车,立即翻…

stm32之硬件SPI读写W25Q64存储器应用案例

系列文章目录 1. stm32之SPI通信协议 2. stm32之软件SPI读写W25Q64存储器应用案例 3. stm32之SPI通信外设 文章目录 系列文章目录前言一、电路接线图二、应用案例代码三、应用案例代码分析3.1 基本思路3.2 相关库函数介绍3.3 MySPI模块3.3.1 模块初始化3.3.2 SPI基本时序单元模…

01_Python基本语法

Hello Python 与其他编程一样,第一个程序都是输出Hello World!。 print("Hello World!")print() 函数由两部分构成 : 指令:print指令的执行对象,在 print 后面的括号里的内容 Hello World!。 Python程序的执行流程如…

机器学习中最常见的50个问题(进阶篇)

机器学习中最常见的50个问题 进阶篇 1.解释SVM的工作原理。 SVM,全称支持向量机(Support Vector Machine),是一种有监督学习算法,主要用于解决数据挖掘或模式识别领域中的数据分类问题。 SVM的工作原理是建立一个最…

【排序算法】之基数排序

一、算法介绍 基数排序是一种非比较型整数排序算法,其原理是将整数按低位到高位或者高位到低位的顺序,依次根据每一位的数值进行排序。通常情况下,基数排序会使用桶排序来处理每一位上的数值。 实现方法主要有如下: 最高位优先(…

echarts实现湖南省地图并且定时轮询

1、在HTML页面引入echarts.min.js <script src"https://cdn.jsdelivr.net/npm/echarts5/dist/echarts.min.js"></script> 2、实现代码 <!DOCTYPE html> <html lang"en"> <head><meta charset"utf-8"><…

如何搞定日语翻译?试试这四款工具

写一篇字数800-1000字的软文&#xff0c;用翻译新手的角度分享福昕翻译在线、福昕翻译客户端、海鲸AI翻译以及彩云翻译在翻译日语时候的表现&#xff0c;要求口语化表达。 最近对于一些轻小说突然感兴趣了&#xff0c;所以我开始尝试各种翻译工具来帮助我搞定日语翻译。今天&am…