HAL STM32基于系统滴答定时器(SysTick)实现多任务时间片轮询
📑RTOS(实时操作系统)和定时器时间片轮询是两种不同的任务调度和执行方式的差异简介
🔖 以下部分内容,由AI给出的解答:
- 🔖RTOS(实时操作系统):
- 🌿RTOS是一种专门设计用于实时系统的操作系统,它可以有效地管理多个任务,提供任务调度、同步和通信等功能。
- 🌿STM32可以使用多种RTOS,例如FreeRTOS、ChibiOS等,它们都提供了任务管理、信号量、消息队列等功能。
- 🌿在RTOS中,每个任务都有自己的优先级,并且RTOS会根据任务的优先级进行调度。高优先级的任务将获得更多的CPU时间。
- 🌿RTOS提供了更结构化的任务管理,使得编写和维护多任务应用程序变得更容易。
- 🔖定时器时间片轮询:
- 🌾定时器时间片轮询是一种基于定时器中断的任务调度方法,而不涉及RTOS的复杂性。
- 🌾在这种方法中,任务的执行由定时器中断触发,每个任务都有一个预定的时间片来执行。
- 🌾当定时器触发时,控制权将转移到下一个任务,如果当前任务没有执行完,它将在下一个时间片继续执行。
- 🌾这种方式的调度对于简单的应用来说可能足够,但在复杂的多任务系统中,可能会导致任务之间的优先级管理和调度变得复杂。
- 📺演示运行效果:
🛠STM32CubeMX工程配置
- 🔧在STM32CubeMX配置工程时,系统时基是默认配置的SysTick定时器。
- 🔱软件默认配置的系统滴答定时器的优先级是最低的,可以根据个人使用情况,进行调整。- 🌿其他外设使用可以根据个人需求配置。
⛳业务代码完善
-
✨STM32CubeMX所创建的工程,系统滴答定时器默认是没有启用中断的需要自行添加和补充。
-
🪓滴答定时器中断回调函数完善。
void HAL_SYSTICK_Callback(void)
{
Sys_Tick_Count();
}
📗时间片轮询驱动
-
🍁时间片轮询,主要有3部分组成:时基(sys_time)、任务管理(sys_task)、任务对象(TASK)组成:
-
🌿sys_time.c
#include "sys_time.h"
static unsigned short int sys_tick = 0;
/**
* @brief 系统时基
*
*/
void Sys_Tick_Count(void)
{
sys_tick += 1;
}
/**
* @description: 获取系统滴答计时
* @param {*}
* @return {*}
*/
unsigned short int Get_Sys_Tick()
{
return sys_tick;
}
/**
* @description: 判断是否超时
* @param {unsigned long int} start 计算开始的时间
* @param {unsigned long int} timeout 超时时长
* @return {*}
*/
unsigned short int Is_Timeout(unsigned short int start, unsigned short int timeout)
{
return ((unsigned short int)(Get_Sys_Tick() - start)) > timeout ? 1: 0;
}
- 🌿sys_task.c
#include "sys_task.h"
#include "sys_time.h"
#include "string.h"
sys_task_t *sys_task_head = NULL;
/**
* @brief 系统任务
*
* @param task 任务
* @param handler 任务轮询函数
* @param interval 轮询间隔
*/
void sys_task_create(sys_task_t *task, void (*handler)(void), unsigned int interval)
{
sys_task_t *sys_task_tail = NULL;
memset(task, 0, sizeof(sys_task_t));
task->enable = 0;
task->interval = interval;
task->tick_cnt = Get_Sys_Tick();
task->task_handler = handler;
task->sys_task_next = NULL;
if (sys_task_head == NULL)
{
sys_task_head = task;
return ;
}
sys_task_tail = sys_task_head;
while (sys_task_tail->sys_task_next != NULL)
{
sys_task_tail = sys_task_tail->sys_task_next;
}
sys_task_tail->sys_task_next = task;
}
/**
* @brief 启动任务
*
* @param task 任务句柄
*/
void sys_task_start(sys_task_t *task)
{
task->enable = 1;
}
/**
* @brief 停止任务
*
* @param task 任务句柄
*/
void sys_task_stop(sys_task_t *task)
{
task->enable = 0;
}
/**
* @brief 系统任务轮询
*
*/
void sys_task_process()
{
sys_task_t *task = NULL;
for (task = sys_task_head; task != NULL; task = task->sys_task_next)
{
if (task->enable && Is_Timeout(task->tick_cnt, task->interval))
{
task->task_handler(); // 运行
task->tick_cnt = Get_Sys_Tick();
}
}
}
- 🌿Blink_TASK.c:(具体执行的任务可以根据个人实际使用进行添加配置,这里以驱动3个led对象为例)
#include "Blink_TASK.h"
#include "usb_printf.h"
void Blink_Task1(void){
HAL_GPIO_TogglePin(LED_GPIO_Port, LED_Pin);
usb_printf("This Blink_Task1\r\n");
HAL_Delay(1000);
}
void Blink_Task2(void){
HAL_GPIO_TogglePin(LED2_GPIO_Port, LED2_Pin);
usb_printf("This Blink_Task2\r\n");
HAL_Delay(800);
}
void Blink_Task3(void){
HAL_GPIO_TogglePin(LED3_GPIO_Port, LED3_Pin);
usb_printf("This Blink_Task3\r\n");
HAL_Delay(600);
}
- main.c中的内容:
- 🍃添加所需运行的任务句柄,需要运行多少个任务就创建多少个对象。
// 控制LED1任务
sys_task_t Task1_Blink;
// 控制LED2任务
sys_task_t Task2_Blink;
// 控制LED3任务
sys_task_t Task3_Blink;
- 🍃创建任务对象
void task_start()
{
sys_task_create(&Task1_Blink, Blink_Task1, 50);
sys_task_start(&Task1_Blink);
sys_task_create(&Task2_Blink, Blink_Task2, 20);
sys_task_start(&Task2_Blink);
sys_task_create(&Task3_Blink, Blink_Task3, 30);
sys_task_start(&Task3_Blink);
}
- 📝main.c所有代码
/* USER CODE BEGIN Header */
/**
******************************************************************************
* @file : main.c
* @brief : Main program body
******************************************************************************
* @attention
*
* Copyright (c) 2024 STMicroelectronics.
* All rights reserved.
*
* This software is licensed under terms that can be found in the LICENSE file
* in the root directory of this software component.
* If no LICENSE file comes with this software, it is provided AS-IS.
*
******************************************************************************
*/
/* USER CODE END Header */
/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include "usart.h"
#include "usb_device.h"
#include "gpio.h"
/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */
#include "usb_printf.h"
//Timer Task Manage
#include "sys_time.h"
#include "sys_task.h"
#include "Blink_TASK.h"
/* USER CODE END Includes */
/* Private typedef -----------------------------------------------------------*/
/* USER CODE BEGIN PTD */
/* USER CODE END PTD */
/* Private define ------------------------------------------------------------*/
/* USER CODE BEGIN PD */
/* USER CODE END PD */
/* Private macro -------------------------------------------------------------*/
/* USER CODE BEGIN PM */
/* USER CODE END PM */
/* Private variables ---------------------------------------------------------*/
/* USER CODE BEGIN PV */
// 控制LED1任务
sys_task_t Task1_Blink;
// 控制LED2任务
sys_task_t Task2_Blink;
// 控制LED3任务
sys_task_t Task3_Blink;
/* USER CODE END PV */
/* Private function prototypes -----------------------------------------------*/
void SystemClock_Config(void);
/* USER CODE BEGIN PFP */
/* USER CODE END PFP */
/* Private user code ---------------------------------------------------------*/
/* USER CODE BEGIN 0 */
// 任务创建
void task_start()
{
sys_task_create(&Task1_Blink, Blink_Task1, 50);
sys_task_start(&Task1_Blink);
sys_task_create(&Task2_Blink, Blink_Task2, 20);
sys_task_start(&Task2_Blink);
sys_task_create(&Task3_Blink, Blink_Task3, 30);
sys_task_start(&Task3_Blink);
}
/* USER CODE END 0 */
/**
* @brief The application entry point.
* @retval int
*/
int main(void)
{
/* USER CODE BEGIN 1 */
/* USER CODE END 1 */
/* MCU Configuration--------------------------------------------------------*/
/* Reset of all peripherals, Initializes the Flash interface and the Systick. */
HAL_Init();
/* USER CODE BEGIN Init */
/* USER CODE END Init */
/* Configure the system clock */
SystemClock_Config();
/* USER CODE BEGIN SysInit */
/* USER CODE END SysInit */
/* Initialize all configured peripherals */
MX_GPIO_Init();
MX_USART1_UART_Init();
MX_USB_DEVICE_Init();
/* USER CODE BEGIN 2 */
uint32_t Main_Fosc = HAL_RCC_GetSysClockFreq();
// HAL_GetTick();
usb_printf("Main_Fosc:%d \r\n", Main_Fosc);
task_start();
/* USER CODE END 2 */
/* Infinite loop */
/* USER CODE BEGIN WHILE */
while(1) {
/* USER CODE END WHILE */
/* USER CODE BEGIN 3 */
sys_task_process();
}
/* USER CODE END 3 */
}
/**
* @brief System Clock Configuration
* @retval None
*/
void SystemClock_Config(void)
{
RCC_OscInitTypeDef RCC_OscInitStruct = {0};
RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};
/** Configure the main internal regulator output voltage
*/
__HAL_RCC_PWR_CLK_ENABLE();
__HAL_PWR_VOLTAGESCALING_CONFIG(PWR_REGULATOR_VOLTAGE_SCALE2);
/** Initializes the RCC Oscillators according to the specified parameters
* in the RCC_OscInitTypeDef structure.
*/
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 = 4;
RCC_OscInitStruct.PLL.PLLN = 168;
RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV4;
RCC_OscInitStruct.PLL.PLLQ = 7;
if(HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK) {
Error_Handler();
}
/** Initializes the CPU, AHB and APB buses clocks
*/
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_DIV2;
RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;
if(HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK) {
Error_Handler();
}
}
/* USER CODE BEGIN 4 */
void HAL_SYSTICK_Callback(void)
{
Sys_Tick_Count();
}
/* USER CODE END 4 */
/**
* @brief This function is executed in case of error occurrence.
* @retval None
*/
void Error_Handler(void)
{
/* USER CODE BEGIN Error_Handler_Debug */
/* User can add his own implementation to report the HAL error return state */
__disable_irq();
while(1) {
}
/* USER CODE END Error_Handler_Debug */
}
#ifdef USE_FULL_ASSERT
/**
* @brief Reports the name of the source file and the source line number
* where the assert_param error has occurred.
* @param file: pointer to the source file name
* @param line: assert_param error line source number
* @retval None
*/
void assert_failed(uint8_t *file, uint32_t line)
{
/* USER CODE BEGIN 6 */
/* User can add his own implementation to report the file name and line number,
ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
/* USER CODE END 6 */
}
#endif /* USE_FULL_ASSERT */
- 👉本示例中所使用的是USB CDC作为调试信息输出,
usb_printf.c、h
代码:
#include "usb_printf.h"
void usb_printf(const char *fmt, ...) {
char buf[128];//自定义缓冲区大小
va_list args;
va_start(args, fmt);
vsnprintf(buf, sizeof(buf), fmt, args);
va_end(args);
CDC_Transmit_FS((uint8_t *)buf, strlen(buf));
}
#ifndef _USB_PRINTF_H
#define _USB_PRINTF_H
#include "stdio.h"
#include <stdarg.h>
#include "usbd_cdc_if.h"
void usb_printf(const char *fmt, ...);
#endif
📚示例工程源码
- 🔖基于stm32f401创建。固件版本:
STM32Cube FW_F4 V1.28.0
链接:https://pan.baidu.com/s/1ouPVpfv9E_2paunmgrOMRg?pwd=r3xk
提取码:r3xk