FreeRTOS学习
Posted Caramel_biscuit
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了FreeRTOS学习相关的知识,希望对你有一定的参考价值。
裸机与RTOS对比
裸机:又称为前后台系统,前台系统指的是中断服务函数,后台系统指的大循环,即应用程序。
- 实时性差:(应用程序轮流执行)
- delay:空等待,CPU不执行其它代码
- 结构臃肿:实现功能都放在无限循环
RTOS:Real Time OS,实时操作系统,强调“实时性”
- 分而治之:实现功能划分为多个任务
- 延时函数:任务调度
- 抢占式:高优先级任务抢占低优先级任务
- 任务堆栈:每个任务都有自己的栈空间,用于保存局部变量以及任务的上下文信息。
中断可以打断任意任务
FreeRTOS基础知识
调度器:使用相关的调度算法来决定当前需要执行哪个任务。
FreeRTOS一共支持三种任务调度方式:
- 抢占式调度:主要针对优先级不同的任务,每个任务都有一个优先级,优先级高的任务可以抢占优先级低的任务。(数值越大,优先级越大)
- 时间片调度:主要针对优先级相同的任务,当多个任务的优先级相同时,任务调度器会在每一次系统时钟节拍到的时候切换任务。
- 协程式调度:当前执行任务将会一直执行,同时高优先级的任务不会抢占低优先级任务。FreeRTOS现在虽然还支持,但是官方表示已经不再更新协程式调度。
抢占式调度
运行过程如下:
1、首先Task1在运行中,在这个过程中Task2就绪了,在抢占式调度器的作用下Task2会抢占Task1的运行
2、Task2运行过程中,Task3就绪了,在抢占式调度器的作用下Task3会抢占Task2的运行
3、Task3运行过程中,Task3阻塞了(系统延时或等待信号量等),此时就绪态中,优先级最高的任务Task2执行
4、Task3阻塞解除了(延时到了或者接收到信号量),此时Task3恢复到就绪态中,抢占TasK2的运行
高优先级任务,优先执行;高优先级任务不停止,低优先级任务无法执行;被抢占的任务将会进入就绪态。
时间片调度
同等优先级任务轮流地享有相同的CPU时间(可设置:设置滴答定时器的中断周期),叫时间片,在FreeRTOS中,一个时间片就等于SysTick中断周期。
Task3运行过程中(还不到一个时间片),Task3阻塞了(系统延时或等待信号量等),此时直接切换到下一个任务Task1。
同等优先级任务,轮流执行;时间片轮转。
一个时间片大小,取决于滴答定时器中断周期。
没有用完的时间片不会再使用,下次任务Task3得到执行还是按照一个时间片的时钟节拍运行。
任务状态
FreeRTOS中任务共存在4种状态:
- 运行态:正在执行的任务,该任务处于运行态,注意在STM32中,同一个时间仅一个任务处于运行态。
- 就绪态:如果该任务已经能够被执行,但当前还未被执行,那么该任务处于就绪态。
- 阻塞态:任务因延时或等待外部事件发生,那么这个任务就处于阻塞态。
- 挂起态:类似暂停,调度函数vTaskSuspend()进入挂起态,需要调用解挂函数vTaskResume()才可以进入就绪态。
- 仅就绪态可以转变成运行态。
系统配置文件详解
FreeRTOSConfig.h配置文件作用:对FreeRTOS进行功能配置和裁剪,以及API函数使能。
相关宏可分为三类:
- INCLUDE:配置FreeRTOS可选的API函数
- config:完成FreeRTOS的功能配置和裁剪
- 其它配置项:PendSV宏定义、SVC宏定义(宏定义给Port.c调用)
#define configUSE_PREEMPTION 1 //1使用抢占式内核,0使用协程
#define configUSE_PORT_OPTIMISED_TASK_SELECTION 1 //1启用特殊方法来选择下一个要运行的任务,1:使用硬件计算下一个要运行的任务(使用特殊方法,STM32支持,任务优先级的最大值有限制32(0~31))。0:使用软件算法计算下一个要运行的任务(通用方式,不限制优先级的最大值,效率较低)
#define configUSE_TICKLESS_IDLE 0 //1启用低功耗tickless模式
#define configCPU_CLOCK_HZ (SystemCoreClock) //CPU频率
#define configIDLE_SHOULD_YIELD 1 //为1时空闲任务放弃CPU使用权给其他同优先级的用户任务
#define configUSE_TASK_NOTIFICATIONS 1 //为1时开启任务通知功能,默认开启
/***************************************************************************************************************/
/* FreeRTOS与内存申请有关配置选项 */
/***************************************************************************************************************/
#define configSUPPORT_DYNAMIC_ALLOCATION 1 //支持动态内存申请
#define configTOTAL_HEAP_SIZE ((size_t)(20*1024)) //系统所有总的堆大小
/* FreeRTOS与钩子函数有关的配置选项 */
/***************************************************************************************************************/
#define configUSE_IDLE_HOOK 0 //1,使用空闲钩子;0,不使用
#define configUSE_TICK_HOOK 0 //1,使用时间片钩子;0,不使用
FreeRTOS学习笔记1—FreeRTOS移植
1. 新建完整的可以点亮LED灯的工程
2.添加FreeRTOS源码
- 2.1 .将FreeRTOS的源码拷贝到新建工程的FreeRTOS的文件加下
- 2.2.Protable文件下只保留Keil、MemMang、RVDS三个文件夹,其余的全部删除
- 2.3.在keil工程新建FreeRTOS_CORE、FreeRTOS_PORTABLE两个分组,分别添加如下图所示的文件
- 2.4.添加头文件路径,如下图所示
- 添加头文件FreeRTOSConfig.h,在…\\FreeRTOSv10.4.1\\FreeRTOS\\Demo\\CORTEX_M4F_STM32F407ZG-SK路径下把FreeRTOSConfig.h拷贝到工程中,位置自定义,本文将其移植到… \\FreeRTOS\\User路径下。
到此为止,我们已经将FreeRTOS的源码添加完毕,接下来我们开始编译修改
3.修改文件编译
- 3.1.将FreeRTOSConfig.h中
#ifdef __ICCARM__
#include <stdint.h>
extern uint32_t SystemCoreClock;
#endif
修改为:
#if defined(__ICCARM__)||defined(__CC_ARM) ||defined(__GNUC__)
#include <stdint.h>
extern uint32_t SystemCoreClock;
#endif
- 3.2.将stm32f4xx_it.c中的void SVC_Handler(void)、void PendSV_Handler(void)、void SysTick_Handler(void)三个函数屏蔽,因为在port.c中也定义了这三个函数;
- 3.3.将FreeRTOSConfig.h中的configUSE_IDLE_HOOK、configUSE_TICK_HOOK、configCHECK_FOR_STACK_OVERFLOW、configUSE_MALLOC_FAILED_HOOK四个钩子函数启用的宏定义为0,至此编译应该为0 Error(s), 0 Warning(s),如有错误在仔细检查
4.新建实验任务,验证移植情况
//任务优先级
#define START_TASK_PRIO 1
//任务堆栈大小
#define START_STK_SIZE 128
//任务句柄
TaskHandle_t StartTask_Handler;
//任务函数
void start_task(void *pvParameters);
//任务优先级
#define LED0_TASK_PRIO 2
//任务堆栈大小
#define LED0_STK_SIZE 50
//任务句柄
TaskHandle_t LED0Task_Handler;
//任务函数
void led0_task(void *pvParameters);
//任务优先级
#define LED1_TASK_PRIO 3
//任务堆栈大小
#define LED1_STK_SIZE 50
//任务句柄
TaskHandle_t LED1Task_Handler;
//任务函数
void led1_task(void *pvParameters);
//开始任务任务函数
void start_task(void *pvParameters)
taskENTER_CRITICAL(); //进入临界区
//创建LED0任务
xTaskCreate((TaskFunction_t )led0_task,
(const char* )"led0_task",
(uint16_t )LED0_STK_SIZE,
(void* )NULL,
(UBaseType_t )LED0_TASK_PRIO,
(TaskHandle_t* )&LED0Task_Handler);
//创建LED1任务
xTaskCreate((TaskFunction_t )led1_task,
(const char* )"led1_task",
(uint16_t )LED1_STK_SIZE,
(void* )NULL,
(UBaseType_t )LED1_TASK_PRIO,
(TaskHandle_t* )&LED1Task_Handler);
vTaskDelete(StartTask_Handler); //删除开始任务
taskEXIT_CRITICAL(); //退出临界区
//LED0任务函数
void led0_task(void *pvParameters)
while(1)
Led_State_Set(LED_Gren,LED_ON);
vTaskDelay(500);
Led_State_Set(LED_Gren,LED_OFF);
vTaskDelay(500);
//LED1任务函数
void led1_task(void *pvParameters)
while(1)
vTaskDelay(200);
Led_State_Set(LED_Blue,LED_ON);
vTaskDelay(800);
Led_State_Set(LED_Blue,LED_OFF);
/**
* @brief The application entry point.
* @param None
* @retval int
*/
int main(void)
/* Reset of all peripherals, Initializes the Flash interface and the Systick*/
HAL_Init();
/* Configure the system clock */
SystemClock_Config();
Led_Port_Init();
//创建开始任务
xTaskCreate((TaskFunction_t )start_task, //任务函数
(const char* )"start_task", //任务名称
(uint16_t )START_STK_SIZE, //任务堆栈大小
(void* )NULL, //传递给任务函数的参数
(UBaseType_t )START_TASK_PRIO, //任务优先级
(TaskHandle_t* )&StartTask_Handler); //任务句柄
vTaskStartScheduler(); //开启任务调度
以上是关于FreeRTOS学习的主要内容,如果未能解决你的问题,请参考以下文章
韦东山freeRTOS系列教程:入门文档教程+进阶视频教程(全部免费的freeRTOS系列教程freeRTOS学习路线)
韦东山freeRTOS系列教程:入门文档教程+进阶视频教程(全部免费的freeRTOS系列教程freeRTOS学习路线)