IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 嵌入式 -> FreeRTOS学习记录 -> 正文阅读

[嵌入式]FreeRTOS学习记录

前言

本人小白,最近学习了FreeRTOS操作系统,打算做一点记录。
学习的过程中虽然做了点练习,不过都是跟着例程来的,大部分没什么记录必要,本文更多的是零零散散地记录我在学习过程中的一些疑惑与个人见解,可能会有错误还请指出。
这只是一个初学者的简单学习记录,并没有全面系统的知识介绍,学习的话还是要看对应的教程啦。

使用到的工具及版本:
FreeRTOS:V9.0.0

STM32CubeMX版本:6.3.0
HAL库:STM32CubeF4 Firmware Package V1.26.2
MDK-ARM:V5.32.0.0
开发板:野火的霸天虎开发板V2(主控芯片是STM32F407ZGT6)

参考的资料:
野火的《FreeRTOS内核实现与应用开发实战》
FreeRTOS入门手册_中文
FreeRTOS官网提供的介绍。

FreeRTOS学习记录

本文对port部分(移植层面)源码的探究查看的是FreeRTOSv9.0.0\FreeRTOS\Source\portable\RVDS\ARM_CM4_MPU,这是我所使用的开发工具和硬件决定的。其他工具或者硬件上的实现可能不一样。

任务会在什么情况下切换?
1、 我们自己编写的程序所引起的任务切换。比如我们自己调用taskYIELD(),或者我们调用的API函数里面包含了任务切换(这种时候往往是当前任务进入阻塞/挂起,又或者是有更高优先级的任务加入了就绪列表,API函数里面会在需要的时候去调用任务切换)。
2、 tick中断里面会触发任务切换。tick由内核的Systick产生,属于内核的定时器/计数器(其设计的本意可能就是供操作系统使用的,为操作系统提供时间度量,我猜的)。
注意: 抢占式调度上面两种情况都存在。协作式调度不存在第2种情况的切换(tick中断的切换),协作式调度中所有的切换都是“显式的”。
显而易见协作式调度比抢占式调度有着更高的CPU效率,毕竟不需要每个tick中断就跑去看看是不是要切换任务。但是它存在着高优先级中断可能得不到快速响应的风险,这取决于我们开发者的程序设计,得看我们是不是在合适的时机调用了任务切换,如果程序结构复杂的话我们可能得时时刻刻考虑是不是该在什么地方进行一次任务切换,这可太麻烦了,有的时候甚至不现实。
通过FreeRTOSConfig.h里面的configUSE_PREEMPTION(0是协作式调度,非0是抢占式调度)决定调度类型。
tick中断源码与任务切换源码
系统tick依赖于Cotex-M4的SysTick硬件中断资源,任务切换依赖于PendSV硬件中断资源(注:下图的SysTick及之前的中断是属于内核的,后面的是芯片厂商的)
在这里插入图片描述
来看看SysTick_Handler(systick中断函数)。

void SysTick_Handler(void)
{	
    #if (INCLUDE_xTaskGetSchedulerState  == 1 )
      if (xTaskGetSchedulerState() != taskSCHEDULER_NOT_STARTED)
      {
    #endif  /* INCLUDE_xTaskGetSchedulerState */  
        xPortSysTickHandler();
    #if (INCLUDE_xTaskGetSchedulerState  == 1 )
      }
    #endif  /* INCLUDE_xTaskGetSchedulerState */
}

如果调度器此时压根还没开启,或者调用了vTaskEndScheduler( void ),那么这里的xPortSysTickHandler()并不会执行,系统节拍则不会增加(当然前提是宏INCLUDE_xTaskGetSchedulerState定义为1)。调度器如果挂起,系统时间依然会增加。
接下来看看xPortSysTickHandler()里面是啥。

void xPortSysTickHandler( void )
{
	/* The SysTick runs at the lowest interrupt priority, so when this interrupt
	executes all interrupts must be unmasked.  There is therefore no need to
	save and then restore the interrupt mask value as its value is already
	known - therefore the slightly faster vPortRaiseBASEPRI() function is used
	in place of portSET_INTERRUPT_MASK_FROM_ISR(). */
	vPortRaiseBASEPRI();
	{
		/* Increment the RTOS tick. */
		if( xTaskIncrementTick() != pdFALSE )
		{
			/* A context switch is required.  Context switching is performed in
			the PendSV interrupt.  Pend the PendSV interrupt. */
			portNVIC_INT_CTRL_REG = portNVIC_PENDSVSET_BIT;
		}
	}
	vPortClearBASEPRIFromISR();
}

首先调用了vPortRaiseBASEPRI()把中断屏蔽,它的实现利用了Cotex-M4内核的BASEPRI register(这张图是从《STM32F4xx-Cortex_-M4内核参考手册》截的)。
在这里插入图片描述
然后调用的xTaskIncrementTick() 则是实现了:
1、系统tick加1(无论调度器是否挂起)
2、如果tick溢出,把延时列表和延时溢出列表交换
3、处理延时列表的事宜
4、如果是抢占式调度,会返回pdTRUE,这样接下来就会把PendSV中断标志位置位,进入PendSV中断服务完成任务切换。如果是协作式调度则返回pdFALSE,不会置位PendSV中断标志。

接下来看看PendSV_Handler(PendSV中断函数,用于任务切换)。
在这里插入图片描述
在这里插入图片描述
有关R14与R15(PC)寄存器的描述(我在网上找的)

R14称为子程序链接寄存器LR(Link Register),当执行子程序调用指令(BL)时,R14可得到R15(程序计数器PC)的备份。在每一种运行模式下,都可用R14保存子程序的返回地址,当用BL或BLX指令调用子程序时,将PC的当前值复制给R14,执行完子程序后,又将R14的值复制回PC,即可完成子程序的调用返回。

寄存器R15用作程序计数器(PC),在ARM状态下,位[1:0]为0,位[31:2]用于保存PC,在Thumb状态下,位[0]为0,位[31:1]用于保存PC。由于ARM体系结构采用了多级流水线技术,对于ARM指令集而言,PC总是指向当前指令的下两条指令的地址,即PC的值为当前指令的地址值加8个字节程序状态寄存器。

一开始我只找了R14的描述,然后觉得很奇怪,上图中的程序怎么会跳到R14(PC的内容)所指的地址呢,不应该是PC+1吗(我只了解一点51内核,cortex-M内核还没怎么了解,以后有时间得看看它的内容),所以才查了PC寄存器的内容。
阅读这些汇编代码需要对照ARM与Thumb指令集说明(除非特殊需要,不然应该没什么人会去背这个吧)。可以在keil提供的“帮助”窗口查阅。
在这里插入图片描述
在这里插入图片描述
任务优先级与中断优先级
中断优先级用于硬件中断管理,属于微处理器内核的内容。cortex-M4的中断优先级数字越大优先级越低。
任务优先级用于调度器的任务管理,属于操作系统的内容。FreeRTOS中任务优先级数字越大优先级越高。
vTaskDelay相对延时与vTaskDelayUntil绝对延时
我试着写了一段代码来进行说明

void Task_1( void* parameter )
{
	/* 用于保存上次时间。调用后自动更新 */
	static portTickType PreviousWakeTime;
	...
	PreviousWakeTime = xTaskGetTickCount();	
	for( ;; )
	{
		...				/*假设此处可能切换成了更高优先级的任务并去执行,
						并假设执行该任务要花费5个tick才会把CPU释放给Task_1*/
		vTaskDelay(50)		or		vTaskDelayUntil(&PreviousWakeTime,50)	
	}	
}

我画了个非常潦草的图来说明Task_1的运行情况(这里假设Task_1处于运行状态的时间非常短,没有画出来)
在这里插入图片描述
可以看到用相对延时的话任务每次被唤醒的时间间隔可能是55个tick,也可能是50个tick,绝对延时则会让任务每次进入就绪的时间是确定的。
关于消息队列、信号量与互斥量
用于进程间通信或同步的手段。信号量与互斥量的创建利用了消息队列的数据结构。它们可以实现任务间的通信,本质上就是函数利用全局变量来对其他函数产生影响(进行通信)。它们可以让等待它们的任务进入阻塞状态。
事件位
相比起消息队列,它的结构更加简单。不能传递值,只传递事件标志。
任务通知
相比起上述的时间和消息队列,它更更简单,不需要创建该对象,也就不需要额外花费RAM空间,因为它是在任务控制块(TCB)里定义了uint32_t类型的变量(用来传递值)和uint8_t类型的变量(用来传递状态)来实现的,它的功能函数的实现也更加简单。(FreeRTOS V8.2.0 才开始支持的)
内存管理方案
heap_1.c:最简单的,只能申请内存,不能进行内存释放,申请内存的时间是一个常量。
heap_2.c:可删除内存。采用了一种最佳匹配算法,但是会产生内存碎片。
heap_3.c:简单地封装了标准 C 库中的 malloc()和 free()函数。此时FreeRTOSConfig.h中的configTOTAL_HEAP_SIZE 宏定义不起作用,使用启动文件里划分的堆空间。
heap_4.c:在heap_2.c的基础上多了合并相邻内存空间的功能。
heap_5.c:在heap_4.c的基础上把外扩RAM也纳入分配空间。
关于“stdint.readme”文件
在C语言的发展过程中会添加新的标准,所以会有C89、C99、C11标准。不同的编译器对标准的支持程度不一样,有的会支持新出的标准而有的则不会。FreeRTOS希望在不同的编译器下都可以编译,所以它没有用C99及C99之后的内容,除了C99引进的stdint.h。这个文件定义了一些整数类型和宏。
如果编译器不提供stdint.h的话,就把FreeRTOS/Source/include路径下的stdint.readme改成stdint.h并放到工程指定的头文件路径下吧。
关于协程
和任务(task)相似的东西,但是对RAM的要求很低,是为存储非常小的情况设计的,但是使用起来有更多的限制,现在用的很少,FreeRTOS官方虽然没有删除协程,但并不再打算更新这部分内容。
关于互斥量的优先级继承机制
一个用来减轻优先级反转危害的措施。当更高优先级的任务(A)要获取一个被低优先级任务(B)占用的互斥量时,A不可避免的要等待B,如果B的优先级临时提高到与A一样,可以避免优先级介于两者的任务这时候掺一脚(假设有任务C,优先级顺序A>C>B,没有优先级继承的话C可能会抢占B,导致A又要等B又要等C执行)。如果不想出现优先级反转,要硬实时之类的,应该是在一开始的设计阶段就要考虑。

在STM32CubeMX中配置FreeRTOS

原本我是计划学习μC/OS的,因为查学习路径的时候网上都推荐μC/OS,不过在使用STM32CubeMX的时候发现它提供了FreeRTOS中间件并可以快速配置,所以选择了先学习FreeRTOS。
STM32CubeMX中FreeRTOS的配置界面。
在这里插入图片描述
CMSIS是什么
CMSIS是ARM和一众厂商一起决定的 Cortex-M 处理器系列的通用接口,相当于大家围在一起制定了标准并提供了各种外设函数、实时操作系统和中间设备等在该标准下的具体实现,使用通用接口的话,应用程序可以更简单地在不同厂商的Cortex-M处理器之间反复横跳(移植更方便),V2表示版本2。
版本选择不了,STM32CubeMX里面好像只能用新版本的FreeRTOS。
点击选项可以看见参数说明等。
在这里插入图片描述
创建任务
任务、队列等直接在STM32CubeMX里面就可以创建好,不必在编辑器里面写。
STM32CubeMX默认创建了一个defaultTask,这是一个啥都不做的任务,每次执行就阻塞1个tick。
我创建了两个任务
在这里插入图片描述
在这里插入图片描述
关于优先级
单从配置的名字看不出具体的优先级数字,只能看出优先级先后。我在cmsis_os2.h里面找到它的定义。

/// Priority values.
typedef enum {
  osPriorityNone          =  0,         ///< No priority (not initialized).
  osPriorityIdle          =  1,         ///< Reserved for Idle thread.
  osPriorityLow           =  8,         ///< Priority: low
  osPriorityLow1          =  8+1,       ///< Priority: low + 1
  osPriorityLow2          =  8+2,       ///< Priority: low + 2
  osPriorityLow3          =  8+3,       ///< Priority: low + 3
  osPriorityLow4          =  8+4,       ///< Priority: low + 4
  osPriorityLow5          =  8+5,       ///< Priority: low + 5
  osPriorityLow6          =  8+6,       ///< Priority: low + 6
  osPriorityLow7          =  8+7,       ///< Priority: low + 7
  osPriorityBelowNormal   = 16,         ///< Priority: below normal
  osPriorityBelowNormal1  = 16+1,       ///< Priority: below normal + 1
  osPriorityBelowNormal2  = 16+2,       ///< Priority: below normal + 2
  osPriorityBelowNormal3  = 16+3,       ///< Priority: below normal + 3
  osPriorityBelowNormal4  = 16+4,       ///< Priority: below normal + 4
  osPriorityBelowNormal5  = 16+5,       ///< Priority: below normal + 5
  osPriorityBelowNormal6  = 16+6,       ///< Priority: below normal + 6
  osPriorityBelowNormal7  = 16+7,       ///< Priority: below normal + 7
  osPriorityNormal        = 24,         ///< Priority: normal
  osPriorityNormal1       = 24+1,       ///< Priority: normal + 1
  osPriorityNormal2       = 24+2,       ///< Priority: normal + 2
  osPriorityNormal3       = 24+3,       ///< Priority: normal + 3
  osPriorityNormal4       = 24+4,       ///< Priority: normal + 4
  osPriorityNormal5       = 24+5,       ///< Priority: normal + 5
  osPriorityNormal6       = 24+6,       ///< Priority: normal + 6
  osPriorityNormal7       = 24+7,       ///< Priority: normal + 7
  osPriorityAboveNormal   = 32,         ///< Priority: above normal
  osPriorityAboveNormal1  = 32+1,       ///< Priority: above normal + 1
  osPriorityAboveNormal2  = 32+2,       ///< Priority: above normal + 2
  osPriorityAboveNormal3  = 32+3,       ///< Priority: above normal + 3
  osPriorityAboveNormal4  = 32+4,       ///< Priority: above normal + 4
  osPriorityAboveNormal5  = 32+5,       ///< Priority: above normal + 5
  osPriorityAboveNormal6  = 32+6,       ///< Priority: above normal + 6
  osPriorityAboveNormal7  = 32+7,       ///< Priority: above normal + 7
  osPriorityHigh          = 40,         ///< Priority: high
  osPriorityHigh1         = 40+1,       ///< Priority: high + 1
  osPriorityHigh2         = 40+2,       ///< Priority: high + 2
  osPriorityHigh3         = 40+3,       ///< Priority: high + 3
  osPriorityHigh4         = 40+4,       ///< Priority: high + 4
  osPriorityHigh5         = 40+5,       ///< Priority: high + 5
  osPriorityHigh6         = 40+6,       ///< Priority: high + 6
  osPriorityHigh7         = 40+7,       ///< Priority: high + 7
  osPriorityRealtime      = 48,         ///< Priority: realtime
  osPriorityRealtime1     = 48+1,       ///< Priority: realtime + 1
  osPriorityRealtime2     = 48+2,       ///< Priority: realtime + 2
  osPriorityRealtime3     = 48+3,       ///< Priority: realtime + 3
  osPriorityRealtime4     = 48+4,       ///< Priority: realtime + 4
  osPriorityRealtime5     = 48+5,       ///< Priority: realtime + 5
  osPriorityRealtime6     = 48+6,       ///< Priority: realtime + 6
  osPriorityRealtime7     = 48+7,       ///< Priority: realtime + 7
  osPriorityISR           = 56,         ///< Reserved for ISR deferred thread.
  osPriorityError         = -1,         ///< System cannot determine priority or illegal priority.
  osPriorityReserved      = 0x7FFFFFFF  ///< Prevents enum down-size compiler optimization.
} osPriority_t;

优先级个数超过32个,可见没有使用硬件优化任务查找(Cortex-M处理器提供了一个计算前导零的指令
CLZ可以优化优先级查找)。
在这里插入图片描述
关于Timebase Source
Timebase Source如果使用SysTick,那么在生成代码前会有这样的警告。
在这里插入图片描述
原因是使用HAL库的函数功能话会依赖一个时钟源提供周期的节拍,这和操作系统的tick(心跳)是一样的道理,这里的Timebase Source是设定HAL所依赖的时钟源,如果把时钟源设为内核的SysTick,相当于HAL库和操作系统共用一个“心脏”,如果HAL库的某个功能把“心脏”给停了那么操作系统会受到影响。
给HAL换一个时钟源就好了,我这里用的TIM1。
在这里插入图片描述
顺便一提,HAL库用来记录节拍数的全局变量叫uwTick,FreeRTOS用来记录节拍数的全局变量叫xTickCount。
编写任务函数
创建好任务后产生代码,接下来写任务功能就好了,我这里就是简单的点灯,把freetros.c里的任务函数补充好。

/* USER CODE BEGIN Header_Task_LED_Red */
/**
  * @brief  Function implementing the TASK_LED_RED thread.
  * @param  argument: Not used
  * @retval None
  */
/* USER CODE END Header_Task_LED_Red */
void Task_LED_Red(void *argument)
{
  /* USER CODE BEGIN Task_LED_Red */
  /* Infinite loop */
  for(;;)
  {
    HAL_GPIO_TogglePin(LED_RED_GPIO_Port,LED_RED_Pin);
	osDelay(3000);
  }
  /* USER CODE END Task_LED_Red */
}

/* USER CODE BEGIN Header_Task_LED_Green */
/**
* @brief Function implementing the TASK_LED_GREEN thread.
* @param argument: Not used
* @retval None
*/
/* USER CODE END Header_Task_LED_Green */
void Task_LED_Green(void *argument)
{
  /* USER CODE BEGIN Task_LED_Green */
  /* Infinite loop */
  for(;;)
  {
    HAL_GPIO_TogglePin(LED_GREEN_GPIO_Port,LED_GREEN_Pin);
	osDelay(1000);
  }
  /* USER CODE END Task_LED_Green */
}

现象
灯光颜色周期性地变化。
在这里插入图片描述

其他

最近机械硬盘坏了,换了个固态硬盘,丢失一些资料… …
接下来我打算学习lwIP。
毕业后在家自学也有3个月了…刚毕业的时候只会51单片机,STM32标准函数库的教程那个时候也看了一部分,但是没怎么上手,感觉自己好菜就没去找工作。现在把常用的外设基本都熟悉了一下,也简单地学习了一个操作系统,也是时候出去找工作了。
刚毕业在家的第一个月感觉就跟以前放暑假一样,时不时就打打游戏看看动漫,第二个月就非常地焦虑紧张害怕,第三个月倒是平静了下来。

  嵌入式 最新文章
基于高精度单片机开发红外测温仪方案
89C51单片机与DAC0832
基于51单片机宠物自动投料喂食器控制系统仿
《痞子衡嵌入式半月刊》 第 68 期
多思计组实验实验七 简单模型机实验
CSC7720
启明智显分享| ESP32学习笔记参考--PWM(脉冲
STM32初探
STM32 总结
【STM32】CubeMX例程四---定时器中断(附工
上一篇文章      下一篇文章      查看所有文章
加:2021-09-26 10:20:39  更:2021-09-26 10:21:27 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/26 3:45:49-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码