STM32F407 是 ST 公司推出的一款高性能微控制器,具有丰富的外设资源和强大的处理能力。HAL(Hardware Abstraction Layer)库是 ST 为其微控制器提供的硬件抽象层,它简化了硬件操作,提高了开发效率。FreeRTOS 是一个开源的实时操作系统,具有轻量级、可移植性强等特点,广泛应用于嵌入式系统开发中。本文将详细介绍如何基于 STM32F407 HAL 库和 FreeRTOS 创建多任务。
在开始开发之前,需要搭建好相应的开发环境。以下是具体步骤:
在 STM32CubeMX 中,通过 “Help” -> “Manage embedded software packages” 安装 STM32F4 系列的 HAL 库。
在 STM32CubeMX 中,通过 “Middleware” -> “RTOS” 选择 FreeRTOS 进行安装。
打开 STM32CubeMX,创建一个新的项目,选择 STM32F407 芯片。以下是具体的配置步骤:
在 “RCC” 选项中,选择外部晶振作为时钟源,并配置系统时钟频率为 168MHz。
在 “System Core” -> “SYS” 中,选择调试接口为 “Serial Wire”。
在 “Middleware” -> “RTOS” 中,选择 FreeRTOS 的版本和内核选项。可以根据需要配置任务栈大小、优先级等参数。
配置完成后,点击 “Project Manager”,选择生成代码的工具链为 “MDK-ARM”,然后点击 “Generate Code” 生成初始化代码。
打开生成的 Keil 项目,在main.c
文件中可以看到 FreeRTOS 的初始化代码。以下是创建多任务的详细步骤:
在main.c
文件中定义任务函数,每个任务函数都有一个特定的功能。例如,以下是两个简单的任务函数:
#include "main.h"
#include "stm32f4xx_hal.h"
#include "cmsis_os.h"
// 任务句柄
osThreadId Task1Handle;
osThreadId Task2Handle;
// 任务1函数
void Task1(void const * argument)
{
for(;;)
{
// 任务1的具体操作
HAL_GPIO_TogglePin(GPIOA, GPIO_PIN_5); // 翻转LED引脚
osDelay(1000); // 延时1秒
}
}
// 任务2函数
void Task2(void const * argument)
{
for(;;)
{
// 任务2的具体操作
HAL_UART_Transmit(&huart1, (uint8_t *)"Task 2 is running!\r\n", 20, 100); // 发送串口信息
osDelay(2000); // 延时2秒
}
}
在main.c
文件的MX_FREERTOS_Init
函数中创建任务。代码如下:
void MX_FREERTOS_Init(void) {
/* 创建任务 */
osThreadDef(Task1, Task1, osPriorityNormal, 0, 128);
Task1Handle = osThreadCreate(osThread(Task1), NULL);
osThreadDef(Task2, Task2, osPriorityNormal, 0, 128);
Task2Handle = osThreadCreate(osThread(Task2), NULL);
}
在main
函数中启动 FreeRTOS 调度器。代码如下:
int main(void)
{
HAL_Init();
SystemClock_Config();
MX_GPIO_Init();
MX_USART1_UART_Init();
MX_FREERTOS_Init();
/* 启动调度器 */
osKernelStart();
while (1)
{
// 主循环一般为空
}
}
下面是一个基于 STM32F407 HAL 库和 FreeRTOS 创建多任务的示例代码:
#include "main.h"
#include "stm32f4xx_hal.h"
#include "cmsis_os.h"
// 任务句柄
osThreadId Task1Handle;
osThreadId Task2Handle;
// 任务1函数
void Task1(void const * argument)
{
for(;;)
{
// 任务1的具体操作
HAL_GPIO_TogglePin(GPIOA, GPIO_PIN_5); // 翻转LED引脚
osDelay(1000); // 延时1秒
}
}
// 任务2函数
void Task2(void const * argument)
{
for(;;)
{
// 任务2的具体操作
HAL_UART_Transmit(&huart1, (uint8_t *)"Task 2 is running!\r\n", 20, 100); // 发送串口信息
osDelay(2000); // 延时2秒
}
}
void SystemClock_Config(void);
static void MX_GPIO_Init(void);
static void MX_USART1_UART_Init(void);
static void MX_FREERTOS_Init(void);
int main(void)
{
HAL_Init();
SystemClock_Config();
MX_GPIO_Init();
MX_USART1_UART_Init();
MX_FREERTOS_Init();
/* 启动调度器 */
osKernelStart();
while (1)
{
// 主循环一般为空
}
}
void SystemClock_Config(void)
{
RCC_OscInitTypeDef RCC_OscInitStruct = {0};
RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};
/** 初始化RCC振荡器
*/
RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
RCC_OscInitStruct.HSEState = RCC_HSE_ON;
RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
RCC_OscInitStruct.PLL.PLLM = 8;
RCC_OscInitStruct.PLL.PLLN = 336;
RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV2;
RCC_OscInitStruct.PLL.PLLQ = 7;
if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
{
Error_Handler();
}
/** 初始化RCC时钟
*/
RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
|RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV4;
RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV2;
if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_5) != HAL_OK)
{
Error_Handler();
}
}
static void MX_GPIO_Init(void)
{
GPIO_InitTypeDef GPIO_InitStruct = {0};
/* GPIO Ports Clock Enable */
__HAL_RCC_GPIOA_CLK_ENABLE();
/*Configure GPIO pin Output Level */
HAL_GPIO_WritePin(GPIOA, GPIO_PIN_5, GPIO_PIN_RESET);
/*Configure GPIO pin : PA5 */
GPIO_InitStruct.Pin = GPIO_PIN_5;
GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
GPIO_InitStruct.Pull = GPIO_NOPULL;
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);
}
static void MX_USART1_UART_Init(void)
{
huart1.Instance = USART1;
huart1.Init.BaudRate = 115200;
huart1.Init.WordLength = UART_WORDLENGTH_8B;
huart1.Init.StopBits = UART_STOPBITS_1;
huart1.Init.Parity = UART_PARITY_NONE;
huart1.Init.Mode = UART_MODE_TX_RX;
huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE;
huart1.Init.OverSampling = UART_OVERSAMPLING_16;
if (HAL_UART_Init(&huart1) != HAL_OK)
{
Error_Handler();
}
}
static void MX_FREERTOS_Init(void)
{
/* 创建任务 */
osThreadDef(Task1, Task1, osPriorityNormal, 0, 128);
Task1Handle = osThreadCreate(osThread(Task1), NULL);
osThreadDef(Task2, Task2, osPriorityNormal, 0, 128);
Task2Handle = osThreadCreate(osThread(Task2), NULL);
}
void Error_Handler(void)
{
while(1)
{
}
}
Task1
和Task2
是两个任务函数,在for(;;)
无限循环里执行各自的操作。Task1
每隔 1 秒翻转一次 LED 引脚,Task2
每隔 2 秒通过串口发送一条信息。MX_FREERTOS_Init
函数中,借助osThreadDef
宏定义任务,再用osThreadCreate
函数创建任务。main
函数中调用osKernelStart
函数启动 FreeRTOS 调度器。FreeRTOS 支持两种主要的任务调度算法:抢占式调度和时间片轮转调度。
Task1
的优先级设为高,Task2
的优先级设为低,当Task2
正在执行时,Task1
就绪,调度器会暂停Task2
,开始执行Task1
,等Task1
阻塞或者完成后,再恢复执行Task2
。Task1
和Task2
优先级相同,调度器会给每个任务分配一个时间片,比如 10ms。Task1
执行 10ms 后,调度器切换到Task2
执行 10ms,如此循环。osDelay
函数)或者被高优先级任务抢占时,调度器会保存当前任务的上下文(寄存器值等),然后恢复下一个要执行任务的上下文,实现任务切换。通过以上的配置和机制,FreeRTOS 能够有效地管理多个任务的执行,保证系统的实时性和稳定性。