STM32F1系列移植freertos

FreeRTOS 是一个可裁剪的小型 RTOS 系统,其特点包括:

● FreeRTOS 的内核支持抢占式,合作式和时间片调度。

● SafeRTOS 衍生自 FreeRTOS,SafeRTOS 在代码完整性上相比 FreeRTOS 更胜一筹。

● 提供了一个用于低功耗的 Tickless 模式。

● 系统的组件在创建时可以选择动态或者静态的 RAM,比如任务、消息队列、信号量、 软件定时器等等。

● 已经在超过 30 种架构的芯片上进行了移植。

● FreeRTOS-MPU 支持 Corex-M 系列中的 MPU 单元,如 STM32F103。

● FreeRTOS 系统简单、小巧、易用,通常情况下内核占用 4k-9k 字节的空间。

● 高可移植性,代码主要 C 语言编写。

● 支持实时任务和协程(co-routines 也有称为合作式、协同程序,本教程均成为协程)。

● 任务与任务、任务与中断之间可以使用任务通知、消息队列、二值信号量、数值型信 号量、递归互斥信号量和互斥信号量进行通信和同步。

● 创新的事件组(或者事件标志)。

● 具有优先级继承特性的互斥信号量。

● 高效的软件定时器。

● 强大的跟踪执行功能。

● 堆栈溢出检测功能。

● 任务数量不限。

● 任务优先级不限。

STM32F1系列移植freertos

以STM32F103ZET6为例:

1. 准备空白工程

这里使用STM32cudemx生成示例代码(点灯代码):

生成代码如下:

注意事项:

由于FreeRTOS使用系统嘀嗒定时器当作心跳,HAL库也使用了这个定时器当作超时计数用,FreeRTOS和HAL库不能同时使用SysTick定时器,所在在这里将HAL库的定时器改为其他的定时器,这里改成了定时器4:

配置中断,注意这里一定要选择组4,也就是全部为抢占优先级:

2. 准备freertos源码

去freertos官网下载源码:https://www.freertos.org/zh-cn-cmn-s/a00104.html

内容如下:





3.进行代码裁剪:

删除不必要的文件:
portable文件夹:


MemMang文件夹:

4. 将文件添加到工程:

在项目里新建文件夹:根目录/Middlewares /FreeRTOS

将 FreeRTOS 内核源码的 Source 文件夹下的所有文件添加到工程的 FreeRTOS 文件夹中

在工程分组里添加新的分组,分 别 为 Middlewares/FreeRTOS_CORE 和 Middlewares/FreeRTOS_PORT,如下图所示:

Middlewares/FreeRTOS_CORE 分组用于存放 FreeRTOS 的内核 C 源码文件, 将 FreeRTOS 目录下所有的 FreeRTOS 的内核 C 源文件添加到 Middlewares/FreeRTOS_CORE 分组中。

Middlewares/FreeRTOS_PORT 分组用于存放 FreeRTOS 内核的移植文件,需要添加两个文件到这个分组,分别为 heap_x.c 和 port.c。
首先是 heap_x.c, 在路径 FreeRTOS/portable/MemMang 下有五个 C 源文件,这五个 C 源文 件对应了五种 FreeRTOS 提供的内存管理算法,读者在进行 FreeRTOS 移植的时候可以根据需 求选择合适的方法,具体这五种内存管理的算法,在后续 FreeRTOS 内存管理章节会具体分析, 这里就先使用 heap_4.c,将 heap_4.c 添加到 Middlewares/FreeRTOS_PORT 分组中。
接着是 port.c, port.c 是 FreeRTOS 这个软件与 MCU 这个硬件连接的桥梁,因此对于 STM32 系列不同的开发板,所使用的 port.c 文件是不同的。 port.c 文件的路径在 FreeRTOS/portable/RVDS 下。进入到 FreeRTOS/portable/RVDS,可以看到 FreeRTOS 针对不同 的 MCU 提供了不同的 port.c 文件,具体STM32 系列开发板与不同 port.c 的对应关 系如下表所示:

将所有 FreeRTOS 相关的所需文件添加到工程后,如下图所示:

5.添加头文件路径

6.添加并修改 FreeRTOSConfig.h 文件

FreeRTOSConfig.h 是 FreeRTOS 操作系统的配置文件, FreeRTOS 操作系统是可裁剪的,用 户可以根据需求对 FreeRTOS 进行裁剪,裁剪掉不需要用到的 FreeRTOS 功能.

新建空白的FreeRTOSConfig.h到core/inc文件夹;

修改FreeRTOSConfig.h
添加代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107

#ifndef FREERTOS_CONFIG_H__
#define FREERTOS_CONFIG_H__

/*
* 关于本文件宏的详细说明可参考:https://blog.csdn.net/zhzht19861011/article/details/50134883#
*/

// 设置为1使用抢占式,为0使用时间片轮转调度。
#define configUSE_PREEMPTION 1

// 设置为1使能低功耗tickless模式,为0保持系统节拍(tick)中断一直运行。
#define configUSE_TICKLESS_IDLE 0

// 系统时钟主频
#define configCPU_CLOCK_HZ 72000000

// 系统节拍中断的频率,即1s进中断的次数,配置为1000就是一秒进1000次中断,系统节拍就是1s。
#define configTICK_RATE_HZ 1000

// 任务最大优先级,对于STM32来说最大不要超过32
#define configMAX_PRIORITIES 32

// 任务最小栈大小
#define configMINIMAL_STACK_SIZE 64

// FreeRTOR堆空间大小
#define configTOTAL_HEAP_SIZE 8192

// 任务名称最大长度
#define configMAX_TASK_NAME_LEN 16

// 系统节拍计数器的变量类型,即定义portTickType是表示16位变量还是32位变量。
#define configUSE_16_BIT_TICKS 0

// 设置为1允许任务调度,为0不允许(时间片耗尽才让出CPU使用权),该参数抢占式方式下才生效
#define configIDLE_SHOULD_YIELD 1

// 设置是否使用互斥量
#define configUSE_MUTEXES 1

// 设置是否使用递归互斥量
#define configUSE_RECURSIVE_MUTEXES 0

// 设置是否使用计数信号量
#define configUSE_COUNTING_SEMAPHORES 0

// 设置可以记录的队列和信号量的最大数目
#define configQUEUE_REGISTRY_SIZE 10

// 是否使用空闲钩子函数
#define configUSE_IDLE_HOOK 0

// 是否使用TICK嘀嗒钩子函数
#define configUSE_TICK_HOOK 0

// 是否使用栈溢出检查
#define configCHECK_FOR_STACK_OVERFLOW 0

// 是否使用内存申请失败钩子函数
#define configUSE_MALLOC_FAILED_HOOK 0

// 是否使用软件定时器
#define configUSE_TIMERS 1

// 设置软件定时器服务/守护进程的优先级
#define configTIMER_TASK_PRIORITY 3

// 设置软件定时器命令队列的长度
#define configTIMER_QUEUE_LENGTH 10

// 设置软件定时器服务/守护进程任务的堆栈深度
#define configTIMER_TASK_STACK_DEPTH configMINIMAL_STACK_SIZE

// STM32的最低优先级
#define configLIBRARY_LOWEST_INTERRUPT_PRIORITY 15

// 能够在中断服务函数中安全调用FreeRTOS API的中断最低优先级
#define configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY 1

#define configKERNEL_INTERRUPT_PRIORITY ( configLIBRARY_LOWEST_INTERRUPT_PRIORITY << 4 )
#define configMAX_SYSCALL_INTERRUPT_PRIORITY ( configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY << 4 )


// 将以下定义设置为1以包含API函数,或设置为0排除API函数
#define INCLUDE_vTaskPrioritySet 1
#define INCLUDE_uxTaskPriorityGet 1
#define INCLUDE_vTaskDelete 1
#define INCLUDE_vTaskSuspend 1
#define INCLUDE_xResumeFromISR 1
#define INCLUDE_vTaskDelayUntil 1
#define INCLUDE_vTaskDelay 1
#define INCLUDE_xTaskGetSchedulerState 1
#define INCLUDE_xTaskGetCurrentTaskHandle 1
#define INCLUDE_uxTaskGetStackHighWaterMark 0
#define INCLUDE_xTaskGetIdleTaskHandle 0
#define INCLUDE_xTimerGetTimerDaemonTaskHandle 0
#define INCLUDE_pcTaskGetTaskName 0
#define INCLUDE_eTaskGetState 0
#define INCLUDE_xEventGroupSetBitFromISR 1
#define INCLUDE_xTimerPendFunctionCall 0

#endif /* FREERTOS_CONFIG_H__ */




7.修改 stm32f1xx_it.c文件

打开stm32f1xx_it.c文件,在头部添加FreeRTOS的三个函数导入:

1
2
3
extern void xPortPendSVHandler(void);
extern void xPortSysTickHandler(void);
extern void vPortSVCHandler(void);


在函数SVC_Handler中添加函数调用:

在函数PendSV_Handler中添加函数调用:

在函数SysTick_Handler中添加函数调用:

8.建立工程进行测试

main.c里添加头文件:

1
2
3
#include "FreeRTOSConfig.h" 
#include "FreeRTOS.h"
#include "task.h"

新建两个点灯任务:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
void LED1Tesk(void *p)
{
while(1)
{
//功能
HAL_GPIO_WritePin(GPIOB,GPIO_PIN_5,GPIO_PIN_SET);
vTaskDelay(1000);
HAL_GPIO_WritePin(GPIOB,GPIO_PIN_5,GPIO_PIN_RESET);
vTaskDelay(1000);
}
}

void LED2Tesk(void *p)
{
while(1)
{
//功能
HAL_GPIO_WritePin(GPIOE,GPIO_PIN_5,GPIO_PIN_SET);
vTaskDelay(500);
HAL_GPIO_WritePin(GPIOE,GPIO_PIN_5,GPIO_PIN_RESET);
vTaskDelay(500);
}
}

开启任务:

两个灯交替闪烁,移植成功!

总结:移植FreeRTOS还是相对简单的,主要是配置文件FreeRTOSConfig.h的配置,每个宏定义所代表的意思需要好好查阅!

参考链接

FreeRTOS之在STM32F103移植
https://blog.csdn.net/m0_37895576/article/details/120435482

STM32CubeMX | STM32使用HAL库手动移植FreeRTOS10.4.1
https://blog.csdn.net/qq153471503/article/details/108999523


STM32F1系列移植freertos
https://www.duruofu.xyz/posts/43973/
作者
DuRuofu
发布于
2023年6月6日
更新于
2025年1月10日
许可协议