STM32 FreeRTOS 中断管理
一、中断优先级配置
在STM32上使用FreeRTOS时,合理配置中断优先级是非常重要的。STM32使用8位宽的寄存器来配置中断的优先等级,但实际只使用了高4位(7:4),因此提供了最大16级的中断优先级。中断优先级数值越小,优先级越高。
二、中断优先级分组
STM32的中断优先级可以分为抢占优先级和子优先级:
• 抢占优先级:抢占优先级高的中断可以打断正在执行但抢占优先级低的中断。
• 子优先级:当同时发生具有相同抢占优先级的中断时,子优先级数值小的优先执行,但不能互相打断。
为了方便FreeRTOS管理,建议将所有优先级位指定为抢占优先级位,即将优先级分组设置为组4。可以通过以下代码设置:
HAL_NVIC_SetPriorityGrouping(NVIC_PRIORITYGROUP_4);
这样,4位优先级就都作为抢占优先级,没有子优先级,共有0~15共16个优先级。
三、FreeRTOS 中断管理
1. 中断优先级配置
FreeRTOS通过宏configMAX_SYSCALL_INTERRUPT_PRIORITY来定义可管理的最高中断优先级。例如:
#define configMAX_SYSCALL_INTERRUPT_PRIORITY (5 << 4)
这意味着FreeRTOS可以管理优先级数值大于等于5的中断。
2. 中断服务例程(ISR)
在中断服务例程中,如果需要调用FreeRTOS的API函数,只能使用带“FromISR”后缀的函数。例如:
xQueueSendFromISR(queue, &data, &higherPriorityTaskWoken);
这样可以确保在中断服务例程中安全地与FreeRTOS任务进行通信。
3. 临界区保护
FreeRTOS提供了宏taskENTER_CRITICAL()和taskEXIT_CRITICAL()来进入和退出临界区。这些宏通过禁用和启用中断来保护临界区代码。例如:
void vPortEnterCritical(void) {
portDISABLE_INTERRUPTS();
uxCriticalNesting++;
if (uxCriticalNesting == 1) {
configASSERT((portNVIC_INT_CTRL_REG & portVECTACTIVE_MASK) == 0);
}
}
void vPortExitCritical(void) {
configASSERT(uxCriticalNesting);
uxCriticalNesting--;
if (uxCriticalNesting == 0) {
portENABLE_INTERRUPTS();
}
}
这些宏确保在临界区代码执行期间,不会被其他中断打断,从而保证操作的原子性。
四、中断控制器配置
在使用FreeRTOS时,需要合理配置中断控制器。例如,设置SysTick和PendSV中断的优先级:
#define configMAX_SYSCALL_INTERRUPT_PRIORITY (5 << 4)
NVIC_SetPriorityGrouping(0);
NVIC_SetPriority(SysTick_IRQn, configMAX_SYSCALL_INTERRUPT_PRIORITY);
NVIC_SetPriority(PendSV_IRQn, configMAX_SYSCALL_INTERRUPT_PRIORITY);
这样可以确保FreeRTOS的系统任务切换不会阻塞其他中断的响应。
五、延迟中断处理
FreeRTOS采用延迟中断处理机制,将中断处理任务延迟到RTOS任务中执行。例如:
• t1-t2这段时间低优先级的任务Task1正在执行,高优先级的任务Task2因等待事件被阻塞。
• t2时刻发生了中断,该中断就是Task2等待的事件,因此Task2进入就绪态。
• t3时刻中断服务程序执行完毕,由于就绪任务中Task2优先级更高,因此调度器选择Task2进入运行态。
六、实际案例
以下是一个实际案例,展示了如何在STM32上使用FreeRTOS进行中断处理和任务调度:
1. 初始化代码
#include "stm32f10x.h"
#include "OLED.h"
#include "FreeRTOS.h"
#include "task.h"
#include "FreeRTOS_experiment.h"
#include "Key.h"
#include "LED.h"
#include "Timer.h"
int main(void) {
Key_Init();
LED_Init();
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);
Timer2_Init();
Timer3_Init();
FreeRTOS_Test();
while (1) {
}
}
void TIM2_IRQHandler(void) {
if (TIM_GetITStatus(TIM2, TIM_IT_Update) == SET) {
LED1_Turn();
TIM_ClearITPendingBit(TIM2, TIM_IT_Update);
}
}
void TIM3_IRQHandler(void) {
if (TIM_GetITStatus(TIM3, TIM_IT_Update) == SET) {
LED2_Turn();
TIM_ClearITPendingBit(TIM3, TIM_IT_Update);
}
}
2. 任务创建和调度
#include "FreeRTOS.h"
#include "task.h"
#include "LED.h"
#include "Key.h"
#include "Delay.h"
#define START_TASK_STACK_SIZE 128
#define START_TASK_PRIO 1
#define TASK3_STACK_SIZE 128
#define TASK3_PRIO 4
void start_task(void);
void task3(void);
TaskHandle_t start_task_handler;
TaskHandle_t task3_handler;
void FreeRTOS_Test(void) {
xTaskCreate((TaskFunction_t)start_task,
"start_task",
START_TASK_STACK_SIZE,
NULL,
START_TASK_PRIO,
(TaskHandle_t *)&start_task_handler);
vTaskStartScheduler();
}
void start_task(void) {
taskENTER_CRITICAL();
xTaskCreate((TaskFunction_t)task3,
"task3",
TASK3_STACK_SIZE,
NULL,
TASK3_PRIO,
(TaskHandle_t *)&task3_handler);
vTaskDelete(NULL);
taskEXIT_CRITICAL();
}
void task3(void) {
uint8_t key = 0;
while (1) {
key = Key_GetNum();
if (key == 1) {
portDISABLE_INTERRUPTS();
}
if (key == 2) {
portENABLE_INTERRUPTS();
}
Delay_ms(10);
}
}
七、总结
通过合理配置中断优先级、使用临界区保护、延迟中断处理等机制,FreeRTOS可以在STM32上实现高效、稳定的中断管理。这些机制不仅提高了系统的实时性,还确保了任务调度的正确性和可靠性。