FreeRTOS基础(七):临界段代码保护及调度器挂起与恢复

2024-06-02 13:52

本文主要是介绍FreeRTOS基础(七):临界段代码保护及调度器挂起与恢复,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

      上一篇博客我们详细介绍了FreeRTOS是怎么管理中断的,其实,从本质上来讲就是将就是利用的BASEPRI这个寄存器,来屏蔽优先级低于某一个阈值的中断,当设置为0的时候,就是打开所有中断,所有中断都可以响应。这样就实现了对所有中断的管理!(中断的管理就是关闭某一个范围的中断和打开中断)。这一篇博客,我们介绍一个另外一个重要的概念,这在实际开发也有非常多的应用。

目录

一、临界段代码保护简介(熟悉)

1.1 什么是临界段代码

1.2 适用场合

1.3 什么可以打断当前程序的运行(打断临界段)?

1.3.1 中断

1.3.2  任务调度

1.4 如何避免打断临界段?

1.5 临界段代码保护API函数介绍

1.5.1 调用格式示例

​1.5.2 特点

1.5.3 总结 

二、任务调度器的挂起和恢复

2.1 概念

2.2 任务调度器的挂起和恢复API函数

2.2.1 任务调度器挂起函数

2.2.2 任务调度器恢复函数

2.3 与使用关闭中断和打开中断实现临界段代码保护的区别

2.4 挂起和恢复任务调度器适用场合

2.5 使用格式示例

2.6 任务挂起与恢复内部实现(了解)


一、临界段代码保护简介(熟悉)

1.1 什么是临界段代码

         临界段代码也叫做临界区,是指那些必须完整运行,不能被打断的代码段。

1.2 适用场合

1.3 什么可以打断当前程序的运行(打断临界段)?

1.3.1 中断

        当中断发生的时候,当前运行的程序就会被暂停,转而去执行中断。

1.3.2  任务调度

       当前任务正在运行,出现一个更高优先级的任务,那么当前运行的程序就会被阻塞住。

1.4 如何避免打断临界段?

       通过上面,我们知道打断临界段的原因有哪些,那么我们就可以对症下药了,解决问题了。

      首先,关闭中断,则中断就不会打断我的运行,其次,任务调度也打断不了我,因为:PendSV是实现任务调度的一个内核中断,并且这个中断的优先级被设置成最低15,这里的关闭中断指的是:在管理的中断优先级范围内(FreeRTOS所能管理的最高中断)的中断都不会响应,那么自然优先级最低的PendSV中断也会被关闭,那么就无法进行任务切换调度,也就无法打断当前程序的运行!

1.5 临界段代码保护API函数介绍

        通过上面的分析,我们就不谋而合了,FreeRTOS 在进入临界段代码的时候需要关闭中断(管理范围内的中断),当处理完临界段代码以后再打开中断,进行正常的中断响应和任务调度。这就是临界段代码保护的方法!

     中断级的临界区函数会返回之前中断屏蔽寄存器的值,我们需要用一个变量接收,保存下来,然后关闭中断,退出临界区时将之前的中断屏蔽寄存器的值写回去。

1.5.1 调用格式示例

1.5.2 特点

1、成对使用。

2、支持嵌套。

3、尽量保持临界段耗时短。因为:在临界区屏蔽了中断,并且也不会发生任务调度了,中断一般是紧急的事,任务调度是可以满足实时性的,如果时间太长,可想而知,会带来什么后果!因此需要注意临界区代码的执行时间不宜过长,否则可能导致系统中断响应的延迟。

1.5.3 总结 

        临界段代码保护是直接屏蔽了中断(管理范围内),系统任务调度靠中断,ISR也靠中断,因此它的影响还是非常大的!!!

二、任务调度器的挂起和恢复

2.1 概念

        挂起就是暂停的意思,任务调度器挂起就是:任务不能再进行切换/调度,恢复就是任务调度器可以重新进行任务的切换。仅仅针对的是任务调度器,就是是否可以发生任务切换!

2.2 任务调度器的挂起和恢复API函数

2.2.1 任务调度器挂起函数

  • vTaskSuspendAll() 函数用于暂停任务调度器。调用这个函数后,任务切换将不会发生,系统会保持当前正在运行的任务。挂起调度器不会关闭中断,中断服务程序(ISR)仍然可以运行。
  • 这个函数通常用于临界区代码保护,确保一段代码在执行过程中不会被任务切换打断。

2.2.2 任务调度器恢复函数

  • xTaskResumeAll() 函数用于恢复任务调度器。恢复后,如果有较高优先级的任务被挂起期间解锁,那么这些任务将会立即执行。如果没有优先级更高的任务,当前任务将继续执行。
  • 在恢复任务调度时,系统会检查任务是否需要进行上下文切换,以确保高优先级任务能及时执行。

2.3 与使用关闭中断和打开中断实现临界段代码保护的区别

FreeRTOS提供了两种主要的临界段代码保护方法:

关闭和打开中断(portENTER_CRITICAL() 和 portEXIT_CRITICAL()):

  • portENTER_CRITICAL() 函数关闭中断,确保在临界区代码执行期间不会被管理范围内的中断打断。也不会发生任务的切换。
  • portEXIT_CRITICAL() 函数重新打开中断,允许系统恢复正常中断处理。
  • 优点:
    • 提供了最高级别的临界区保护,因为它完全阻止了中断
    • 适用于非常短且时间敏感的代码段
  • 缺点:
    • 关闭中断会影响整个系统的中断响应时间,可能会导致高优先级中断的延迟处理。
    • 不适用于长时间执行的代码,因为这会严重影响系统的实时性。

挂起和恢复任务调度器(vTaskSuspendAll 和 xTaskResumeAll):

  • vTaskSuspendAll() 暂停任务调度,但不会影响中断。这意味着ISR仍然可以执行,且ISR可以发起上下文切换(如果在ISR中调用了Yield函数)
  • xTaskResumeAll() 恢复任务调度,并在必要时进行上下文切换。
  • 它仅仅是防止了任务之间的资源争夺,中断照样可以直接响应;中断可以抢占当前任务的资源,但是其他任务无法抢占当前任务资源
  • 优点:
    • 保持中断响应能力,因为中断不会被禁用
    • 适用于需要保护不被任务切换打断的长时间执行代码,但又不需要完全禁止中断的场景。
  • 缺点:
    • 不适用于极端时间敏感的代码保护,因为ISR仍然可以打断执行。
    • 任务调度恢复时需要检查并可能进行任务切换,增加了一些开销。

   总结一下就是:利用任务调度器的挂起和恢复实现的临界段代码保护只是不会发生任务的切换,但是中断并未关闭,在临界区,中断依然可以响应,然而,利用关中断和开中断实现的临界段代码,它直接关闭了管理范围内的中断,任务切换也不会发生!它的力度是非常强的!

2.4 挂起和恢复任务调度器适用场合

挂起调度器的方式,适用于临界区位于任务与任务之间;既不用去延时中断,又可以做到临界区的安全。就是说:它不会受到任务切换而被打断,同时又能保证中断的及时响应!

  1. 任务与任务之间的临界区:

    • 在多任务操作系统中,多个任务之间可能需要访问共享资源(例如全局变量、硬件设备等)。为了防止任务之间的资源竞争,需要保护这些访问操作。
    • 临界区是指访问共享资源的代码段,需要确保这个代码段在一个任务执行时不会被其他任务打断。
  2. 不延时中断:

    • 挂起任务调度器 (vTaskSuspendAll()) 的方式不会关闭中断,这意味着系统中的中断服务程序(ISR)仍然可以正常执行。ISR的执行不会受到挂起任务调度器的影响。
    • 由于中断没有被关闭,系统的中断响应时间不会受到影响。这样可以保证系统的实时性,因为时间关键的中断仍然能够及时处理。
  3. 临界区的安全:

    • 当任务调度器被挂起时,当前任务可以独占CPU资源执行临界区代码,而不必担心在这个过程中会被其他任务打断。这样可以确保任务间临界区的代码安全执行。
    • 任务调度器挂起期间,虽然ISR可以执行,但ISR不会引起任务切换。因此,任务执行临界区代码的过程中,不会被其他任务打断。

2.5 使用格式示例

2.6 任务挂起与恢复内部实现(了解)

至此,已经讲解完毕!初次学习,循序渐进,一步步掌握即可!以上就是全部内容!请务必掌握,创作不易,欢迎大家点赞加关注评论,您的支持是我前进最大的动力!下期再见!

这篇关于FreeRTOS基础(七):临界段代码保护及调度器挂起与恢复的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1024152

相关文章

利用Python调试串口的示例代码

《利用Python调试串口的示例代码》在嵌入式开发、物联网设备调试过程中,串口通信是最基础的调试手段本文将带你用Python+ttkbootstrap打造一款高颜值、多功能的串口调试助手,需要的可以了... 目录概述:为什么需要专业的串口调试工具项目架构设计1.1 技术栈选型1.2 关键类说明1.3 线程模

Python Transformers库(NLP处理库)案例代码讲解

《PythonTransformers库(NLP处理库)案例代码讲解》本文介绍transformers库的全面讲解,包含基础知识、高级用法、案例代码及学习路径,内容经过组织,适合不同阶段的学习者,对... 目录一、基础知识1. Transformers 库简介2. 安装与环境配置3. 快速上手示例二、核心模

Java的栈与队列实现代码解析

《Java的栈与队列实现代码解析》栈是常见的线性数据结构,栈的特点是以先进后出的形式,后进先出,先进后出,分为栈底和栈顶,栈应用于内存的分配,表达式求值,存储临时的数据和方法的调用等,本文给大家介绍J... 目录栈的概念(Stack)栈的实现代码队列(Queue)模拟实现队列(双链表实现)循环队列(循环数组

Linux之计划任务和调度命令at/cron详解

《Linux之计划任务和调度命令at/cron详解》:本文主要介绍Linux之计划任务和调度命令at/cron的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux计划任务和调度命令at/cron一、计划任务二、命令{at}介绍三、命令语法及功能 :at

Android Mainline基础简介

《AndroidMainline基础简介》AndroidMainline是通过模块化更新Android核心组件的框架,可能提高安全性,本文给大家介绍AndroidMainline基础简介,感兴趣的朋... 目录关键要点什么是 android Mainline?Android Mainline 的工作原理关键

使用Java将DOCX文档解析为Markdown文档的代码实现

《使用Java将DOCX文档解析为Markdown文档的代码实现》在现代文档处理中,Markdown(MD)因其简洁的语法和良好的可读性,逐渐成为开发者、技术写作者和内容创作者的首选格式,然而,许多文... 目录引言1. 工具和库介绍2. 安装依赖库3. 使用Apache POI解析DOCX文档4. 将解析

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

使用Python实现全能手机虚拟键盘的示例代码

《使用Python实现全能手机虚拟键盘的示例代码》在数字化办公时代,你是否遇到过这样的场景:会议室投影电脑突然键盘失灵、躺在沙发上想远程控制书房电脑、或者需要给长辈远程协助操作?今天我要分享的Pyth... 目录一、项目概述:不止于键盘的远程控制方案1.1 创新价值1.2 技术栈全景二、需求实现步骤一、需求

Java中Date、LocalDate、LocalDateTime、LocalTime、时间戳之间的相互转换代码

《Java中Date、LocalDate、LocalDateTime、LocalTime、时间戳之间的相互转换代码》:本文主要介绍Java中日期时间转换的多种方法,包括将Date转换为LocalD... 目录一、Date转LocalDateTime二、Date转LocalDate三、LocalDateTim

mysql的基础语句和外键查询及其语句详解(推荐)

《mysql的基础语句和外键查询及其语句详解(推荐)》:本文主要介绍mysql的基础语句和外键查询及其语句详解(推荐),本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋... 目录一、mysql 基础语句1. 数据库操作 创建数据库2. 表操作 创建表3. CRUD 操作二、外键