当前位置: 首页 > article >正文

FreeRTOS的中断管理

前言

        FreeRTOS的任务有优先级,MCU的硬件中断有中断优先级,这是两个不同的概念,FreeRTOS的任务管理要用到硬件中断,使用FreeRTOS时候也可以使用硬件中断,但是硬件中断ISR的设计要注意一些设计原则,在本节中我将详细介绍FreeRTOS与硬件中断的关系,以及如何正确使用硬件中断

1、FreeRTOS与中断设置

        中断是MCU的硬件特性,FreeRTOS的运行要用到中断,因为它的上下文切换就是在PendSV中断中进行的,FreeRTOS还需要一个基础时钟产生滴答信号,CubeMX中启动FreeRTOS后,系统会自动对NVIC进行一些设置,这些设置是必要的

        启用FreeRTOS后,中断优先级分组策略自动设置为4位全用于抢占优先级,所以优先级编号是0到15,这个参数在FreeRTOS中不能修改,默认为4

        FreeRTOS可管理的最高优先级默认值为5,也就是说,只有在中断优先级等于或大于5的中断ISR里,才可以调用FreeRTOS的中断安全API函数,也就是带 FromISR 后缀的函数,使用taskDISABLE_INTERRUPTS() 函数也只能屏蔽优先级数值等于或者大于5的中断

2、任务与中断的关系

        MCU的中断有中断优先级,有中断服务例程(ISR),FreeRTOS的任务有任务优先级,有任务函数,这两者具体的特点和区别如下

1、中断是MCU特有的硬件属性,由硬件时间和软件信号引起中断,运行哪个ISR是由硬件决定的,中断的优先级数字越小,表示优先级越高,所以中断的最高优先级是0

2、FreeRTOS的任务是一个纯软件概念,与硬件系统无关,任务的优先级是由开发者在软件中赋予的,任务的优先级数字越低,表示优先级越低,所以任务的最低优先级为0,FreeRTOS的任务调度器决定哪个任务处于运行状态,FreeRTOS在中断优先级15的PendSV中断里进行上下文切换,所以只要有中断ISR在运行,FreeRTOS就无法进行切换

3、任务只有在没有ISR运行的时候才能运行,即使是优先级最低的中断,也可以打断高优先级的任务去执行,而任务不能抢占ISR的运行

        注意对最后一条规则的理解,根据NVIC管理中断的原则,同等抢占优先级的中断是不能发生抢占的,一个优先级为15的RTC唤醒中断是不能抢占优先级为15的SysTick和PendSV中断的执行的,这是因为SysTick和PendSV中断的ISR运行时间很短,RTC唤醒中断的ISR能被及时执行,但如果优先级为15的RTC唤醒中断执行时间很长,那么SysTick和PendSV发生了中断也无法抢占,也就无法进行任务调度,任务函数也无法执行

        所以在实际的软件设计中,一般要尽量简化ISR的功能,使其减少占用CPU的时间,一般的硬件中断都是处理一些数据的接收或发送工作,例如采用中断方式进行ADC数据采集时,只需要把ADC的中断里将数据读取到缓冲区,而对数据进行滤波,频谱计算等耗时间的工作就转移到任务中处理。

3、中断屏蔽和临界段代码

        一个任务函数在执行时,可能会被其他高优先级的任务抢占CPU,也可能被任何一个中断的ISR抢占CPU,在某些时候,任务的某段代码可能非常关键,需要连续执行完,不希望被其他任务或者中断打断,这种程序段被成为临界段,在FreeRTOS中,有函数定义临界段代码,也可以屏蔽系统的部分中断

我们在文件task.h中定义了几个宏函数,定义代码如下,函数功能见注释

//开启临界段代码
#define taskENTER_CRITICAL()		portENTER_CRITICAL()
#define taskENTER_CRITICAL_FROM_ISR() portSET_INTERRUPT_MASK_FROM_ISR()

//退出临界段代码
#define taskEXIT_CRITICAL()			portEXIT_CRITICAL()
#define taskEXIT_CRITICAL_FROM_ISR( x ) portCLEAR_INTERRUPT_MASK_FROM_ISR( x )

//屏蔽MCU的部分中断 5-15 以及解除屏蔽
#define taskDISABLE_INTERRUPTS()	portDISABLE_INTERRUPTS()
#define taskENABLE_INTERRUPTS()		portENABLE_INTERRUPTS()

4、在ISR中使用FreeRTOS的API函数

        在中断的ISR里,有时会需要调用FreeRTOS的API函数,但是调用普通的API函数可能会存在问题,例如在ISR中调用vTaskDelay()就会出现问题,因为在vTaskDelay会使任务进入阻塞状态,而ISR根本就不是任务,ISR运行的时候,也不能进行任务调度

        为此,FreeRTOS的API函数分为两个版本,一个被成为“任务级”,既普通名称的API函数,另一个被成为“中断级”,既带后缀的FromISR的函数,中断API也被成为中断安全API函数

        FreeRTOS将API函数分为两个版本的好处是:在API的实现代码中,无须判断调用这个API函数的是一个ISR还是一个任务,否则需要增加额外的代码,而且不同的MCU判断ISR和任务函数的机制也不一样,所以两个版本的API函数会提升我们开发的效率

        在ISR中,绝对不能使用任务级API函数,但是在任务函数中,可以使用中断级API函数,此外在FreeRTOS不能管理的高优先级中断的ISR中,连中断级的API函数也不能调用

5、FreeRTOS中断及ISR的设计原则

根据FreeRTOS管理中断的特点,中断优先级和ISR程序设计应该遵循如下原则

1、根据参数的设置,MCU的优先级为0到15的中断,分为FreeRTOS不可屏蔽中断和可屏蔽中断,要根据中断的重要性和功能,为其设置合适的中断优先级,使其成为FreeRTOS的不可屏蔽中断和可屏蔽中断

2、ISR的代码应该尽量简短,应该将比较耗时的处理功能转移到任务函数中实现

3、可屏蔽中断的ISR中,能调用中断级的FreeRTOS API函数,绝不能调用普通的API函数,在不可屏蔽中断的ISR中,不能调用任何的FreeRTOS的API函数


http://www.kler.cn/a/325747.html

相关文章:

  • 《深度剖析算法优化:提升效率与精度的秘诀》
  • 浅谈云计算03 | 云计算的技术支撑(云使能技术)
  • 【Rust自学】12.6. 使用TDD(测试驱动开发)开发库功能
  • JAVA:利用 RabbitMQ 死信队列实现支付超时场景的技术指南
  • 计算机网络之---网络安全的基本概念
  • Http 响应状态码 前后端联调
  • AI产品经理面试题详细整理【已拿offer】
  • I/O中断处理过程
  • Linux操作系统如何定时关机?
  • 论文笔记:Gradient Episodic Memory for Continual Learning
  • Chromium 使用安全 DNS功能源码分析c++
  • windows11家庭版开启Hyper-v
  • 使用 Spring Boot 实现 JWT 生成与验证的自定义类
  • golang web笔记-3.响应ResponseWriter
  • 万界星空科技数字孪生:解锁制造业未来,重塑智慧工厂新纪元
  • 从0学习React(4)---更新组件状态setState
  • gradle的入门及kotlin的了解
  • 开箱元宇宙| 探索 Great Entertainment Group 如何利用 Web3 和数字创新重新定义活动体验
  • 【Android】多角度看handler--looper的阻塞
  • 产品管理- 互联网产品(6):产品测试
  • CSS预处理器LESS
  • 工厂模式和抽象工厂模式的实验报告
  • Could not find com.mapbox.mapboxsdk:mapbox-android-accounts:0.7.0.解决
  • 禾赛嵌入式面试题及参考答案(2万字长文)
  • 升级 Windows 后如何恢复丢失的文件
  • 51单片机的教室智能照明系统【proteus仿真+程序+报告+原理图+演示视频】