【小黑嵌入式系统第十二课】μC/OS-III程序设计基础(二)——系统函数使用场合、时间管理、临界区管理、使用规则、互斥信号量

本文主要是介绍【小黑嵌入式系统第十二课】μC/OS-III程序设计基础(二)——系统函数使用场合、时间管理、临界区管理、使用规则、互斥信号量,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

上一课:
【小黑嵌入式系统第十一课】μC/OS-III程序设计基础(一)——任务设计、任务管理(创建&基本状态&内部任务)、任务调度、系统函数

文章目录

    • 一、系统函数使用场合
      • 1.1 时间管理
        • 1.1.1 控制任务的执行周期
        • 1.1.2 控制任务的运行节奏
        • 1.1.3 状态查询
      • 1.2 资源同步
        • 1.2.1 “资源同步”图解
        • 1.2.2 “资源同步” 实现方式
      • 1.3 行为同步
        • 1.3.1 行为同步
        • 1.3.2 数据通信
    • 二、时间管理
      • 2.1 概述
      • 2.2 `OSTimeDly()`
      • 2.3 `OSTimeDlyHMSM()`
      • 2.4 `OSTimeDlyResume()`
      • 2.5 `OSTimeGet()`、`OSTimeSet()`
    • 三、临界区管理
      • 3.1 进入然后退出临界区
      • 3.2 禁止然后允许调度
    • 四、事件的一般使用规则
      • 4.1 相似性
      • 4.2 先创建后使用
      • 4.3 配对使用
      • 4.4 在ISR中使用
    • 五、互斥信号量
      • 5.1 简介
      • 5.2 资源同步


一、系统函数使用场合

1.1 时间管理

1.1.1 控制任务的执行周期

在任务函数的代码中可以通过插入延时函数来控制任务周期性运行,定时闲置CPU一段时间,供其它任务使用。

在这里插入图片描述

注意:延时函数OSTimeDly()是以系统节拍数为参数,而延时函数OSTimeDlyHMSM()是以实际时间值为参数,但在执行过程中仍然转换为系统节拍数。如果实际时间不是系统节拍的整数倍,将进行四舍五入处理。设系统节拍为50毫秒,调用OSTimeDly(20)的效果是延时1秒钟,调用OSTimeDlyHMSM(0,1,27,620)的实际时间是延时1分27秒600毫秒。


1.1.2 控制任务的运行节奏

在任务函数的代码中也可以通过插入延时函数来控制任务的运行节奏。

在这里插入图片描述


1.1.3 状态查询

查询过程是一个无限循环过程,只有当希望的状态出现以后才能退出这个无限循环,这种情况在实时操作系统管理下是不允许的,它将剥夺低优先级任务的运行机会。解决这个问题的办法是“用定时查询代替连续查询” 。

在这里插入图片描述


1.2 资源同步

1.2.1 “资源同步”图解

在这里插入图片描述


1.2.2 “资源同步” 实现方式
  1. 使用关中断:通过调用禁止中断函数OS_CRITICAL_ENTER()和允许中断函数OS_CRITICAL_EXIT()实现的。
  2. 使用关调度:通过调用禁止任务调度函数OSSchedLock()和允许任务调度函数OSSchedUnlock()实现的,因为禁止调度违背了多任务的初衷,所以不建议用户使用。
  3. 使用信号量与互斥信号量:通过等待信号量和发送信号量实现共享资源的独享。

1.3 行为同步

1.3.1 行为同步

一个任务的运行过程需要和其它任务的运行配合,才能达到预定的效果,任务之间的这种动作配合和协调关系称为“行为同步”。

在这里插入图片描述


1.3.2 数据通信

在这里插入图片描述
注意:尽管指针可能是局部变量,但只要指针指向的变量是全局变量,操作指针指向的变量时也需要当作全局变量来处理。


二、时间管理

2.1 概述

μC/OS-III提供了若干个时间管理服务函数,可以满足任务在运行过程中对时间管理的需求。在使用时间管理服务函数时,必须十分清楚一个事实:时间管理服务函数是以系统节拍为处理单位的,实际的时间与希望的时间是有误差的,最坏的情况下误差接近一个系统节拍。因此时间管理服务函数只能用在对时间精度要求不高的场合,或者时间间隔较长的场合。


2.2 OSTimeDly()

系统延时函数OSTimeDly()调用图解:

在这里插入图片描述

下面我们设计一个任务,让一个LED以50个时钟节拍为单位闪烁,说明OSTimeDly()函数的用途。由于篇幅关系,只给出任务主要处理代码。

在这里插入图片描述

注意:上面的设计是OSTimeDly() 控制任务的周期性执行,还可以用它来控制任务的运行节拍。


2.3 OSTimeDlyHMSM()

μC/OS-III提供了OSTimeDlyHMSM()系统延时函数,这个函数是以小时(H)、分(M)、秒(S)和毫秒(m)四个参数来定义延时时间的,函数在内部把这些参数转换为时钟节拍,再通过单次或多次调用OSTimeDly()进行延时和任务调度,所以延时原理和调用延时函数OSTimeDly()是一样的。

为了说明OSTimeDlyHMSM()函数的使用方法,下面我们设计一个任务,让一个LED以2Hz的频率闪烁。下面给出任务主要处理代码。

在这里插入图片描述

注意:上面的设计是**OSTimeDlyHMSM()**控制任务的周期性执行,还可以用它来控制任务的运行节拍。


2.4 OSTimeDlyResume()

µC/OS-III允许用户结束任务正处于的延时期,延时的任务可以不等待延时期满,而是通过其它任务取消其延时来使其处于就绪态,通过调用OSTimeDlyResume()和指定要恢复的任务来完成。

在这里插入图片描述

为了说明OSTimeDlyResume()函数的使用方法,我们设计一个系统。TaskLED任务让一个LED以0.5Hz的频率闪烁;按键任务处理按键事件,每按键一次,LED状态立即翻转一次。下面是两个任务的处理流程。

在这里插入图片描述
TaskLED任务代码如下:

在这里插入图片描述
TaskKEY任务的代码如下:

在这里插入图片描述


2.5 OSTimeGet()OSTimeSet()

无论时钟节拍何时发生,µC/OS-III都会将一个32位的计数器加1,这个计数器在用户调用OSStart()初始化多任务时和4,294,967,295个节拍执行完一遍的时候从0开始计数。

在这里插入图片描述

用户可以通过调用OSTimeGet()来获得该计数器的当前值, OSTimeGet()的详细信息见下表。

在这里插入图片描述
在这里插入图片描述

用户可以通过调用OSTimeSet()来改变计数器的值,OSTimeSet()的详细信息见下表。

在这里插入图片描述
在这里插入图片描述

为了说明OSTimeGet()函数的使用方法,我们设计一个任务,计算两次按键的时间间隔放在全局变量ktime中。下面是任务的处理流程。

在这里插入图片描述
TaskKEY任务代码如下:

在这里插入图片描述


三、临界区管理

3.1 进入然后退出临界区

进入然后退出临界段是“资源同步”的方法之一,能够在访问共享资源时保障信息的可靠性和完整性。

在这里插入图片描述

为了说明它在”资源同步”时的使用,我们设计一个系统,假设有两个任务,它们都对全局变量sum1和sum2操作。低优先级任务让这两个变量始终相等,并不断在计数;高优先级任务不断的判断这两个变量是否相等,不相等则点亮LED,下面是两个任务的处理流程。

在这里插入图片描述
TaskLED任务代码如下:

在这里插入图片描述

TaskAdd任务代码如下:

在这里插入图片描述


3.2 禁止然后允许调度

给调度器上锁OSSchedlock()函数用于禁止任务调度,直到任务完成后调用给调度器开锁OSSchedUnlock()函数为止。使用它有3点需要注意。

  1. OSSchedlock()OSSchedUnlock()必须成对使用,也可以嵌套使用;
  2. OSSchedlock()只是禁止了任务的调度,而没有禁止中断,此时如果允许中断,中断到来时还是会执行对应的中断服务程序;
  3. 调用OSSchedLock()以后,用户的应用程序不得使用任何能将现行任务挂起的系统调用,直到配对的OSSchedUnlock()调用为止。

注意:对于用户来说,极少使用禁止然后允许调度的方法。不过,很多操作系统内部和驱动程序使用它来减少中断响应时间。


四、事件的一般使用规则

4.1 相似性

事件管理函数是μC/OS-III中最多的系统函数,而且每种事件具有的管理函数数目不同。但是所有的事件都有类似的4个函数,它们是所有事件的基本功能,其函数名类似,使用方法也类似,详细函数见下表。

在这里插入图片描述


4.2 先创建后使用

任何一个事件,必须先创建后使用。创建事件是通过调用函数OS???Create()实现的,其中???为事件的类型。创建事件可以在main()函数中,但更多的是在任务初始化部分。使用方法如下。

在这里插入图片描述

一般来说,在嵌入式系统中,事件是静态使用的,即创建后永远不删除。但有时候需要动态使用,即根据需要创建和删除事件,此时创建事件就是在任务的事件执行代码中,使用方法如下。

在这里插入图片描述


4.3 配对使用

下面给出一个示例,假设Task0为高优先级任务,Task1为低优先级任务。 Task0代码如下。

在这里插入图片描述
Task1代码如下:

在这里插入图片描述


4.4 在ISR中使用

要掌握事件函数在中断服务程序中的调用规则,我们必须清楚中断服务有哪些特点。

  • 中断与所有的任务异步
  • 中断服务程序总体是顺序结构
  • 中断服务程序需要尽快退出
  • 中断服务程序不能等待

下面给出事件在中断服务程序中使用方法,假设Task0任务接收ISR发送的消息,任务代码如下。

在这里插入图片描述
ISR中的代码如下:

在这里插入图片描述

注意:

  1. 中断服务程序一般不会调用建立和删除事件函数,否则要么没有起到事件的作用,要么程序很复杂;
  2. 中断服务程序不能调用等待事件的函数,否则可能造成程序崩溃。

五、互斥信号量

5.1 简介

在日常生活中,出租车是一种常用的共享资源,当出租车载客时,从外面可以看到标识为载客;当空闲时,标识为空车。这样等车的人就可以根据标识知道出租车的当前状态,判断是否能够座上这辆车。这个标识牌就是一个二值信号量。由于这种二值信号量可以实现对共享资源的独占式处理,所以叫做互斥信号量。

在这里插入图片描述

互斥信号量也称为mutex,专用于资源同步。互斥信号量具有一些特性:占用一个空闲优先级,以便解决优先级反转问题。

假设任务1和任务3共享一个资源,任务2为优先级介于任务1和任务3之间的一个与该共享资源无关任务,分析优先级反转问题。

在这里插入图片描述

此时,只要任务3没有释放共享资源,即使任务1比任务2优先级更高,但却在任务2之后运行,这种现象就是优先级反转。
并且在任务1等待共享资源期间,任务2就绪时可剥夺任务3的运行,导致额外增加了任务1的等待时间,这被称为无界优先级反转,因任何一个(介于任务1、3之间的)中等优先级的任务都可以延长任务1的等待时间。

假设任务1和任务3共享一个资源,使用互斥信号量进行资源同步,任务2为优先级介于任务1和任务3之间的一个与该共享资源无关任务,通过互斥信号量解决优先级反转问题。

在这里插入图片描述

此时,任务2无法在任务1之前得到运行,不发生优先级反转

综上所述,可以说能防止优先级反转现象的信号就是互斥信号量。

使用互斥信号量有以下3点需要注意:

  1. 在嵌入式系统中,经常使用互斥信号量访问共享资源来实现资源同步。而用来实现资源同步的互斥信号量在创建时初始化,这是由OSMutexCreate ()函数来实现的;

  2. OSMutexPost ()发送互斥信号量函数与OSMutexPend ()等待互斥信号量函数必须成对出现在同一个任务调用的函数中,因此我们需要编写一个公共的库函数,因为有多个任务可能调用这个函数 ;

  3. 信号量最好在系统初始化的时候创建,不要在系统运行的过程中动态地创建和删除。在确保成功地创建信号量之后,才可对信号量进行接收和发送操作。


5.2 资源同步

为了说明使用互斥信号量访问共享资源实现资源同步,设计两个任务,它们在不同的音符按键按下期间让DAC输出不同的音符声音,要求这两个任务不会互相干扰。下面是两个任务的处理流程。

在这里插入图片描述

为了实现资源同步,我们需要保证OSMutexPost()OSMutexPend()成对出现在同一个任务函数中,所以我们编写一个库函数DAC()供两个任务调用,代码如下。

在这里插入图片描述

下面给出两个DAC任务的主要处理代码:

在这里插入图片描述

使用互斥信号量做共享资源的管理,可参考示例程序:“Micrium_CY8CKIT-050B_uCOS-III-LCDMutex_GNU(PSoC Creator 4.0).rar”

这篇关于【小黑嵌入式系统第十二课】μC/OS-III程序设计基础(二)——系统函数使用场合、时间管理、临界区管理、使用规则、互斥信号量的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

详解Vue如何使用xlsx库导出Excel文件

《详解Vue如何使用xlsx库导出Excel文件》第三方库xlsx提供了强大的功能来处理Excel文件,它可以简化导出Excel文件这个过程,本文将为大家详细介绍一下它的具体使用,需要的小伙伴可以了解... 目录1. 安装依赖2. 创建vue组件3. 解释代码在Vue.js项目中导出Excel文件,使用第三

Linux alias的三种使用场景方式

《Linuxalias的三种使用场景方式》文章介绍了Linux中`alias`命令的三种使用场景:临时别名、用户级别别名和系统级别别名,临时别名仅在当前终端有效,用户级别别名在当前用户下所有终端有效... 目录linux alias三种使用场景一次性适用于当前用户全局生效,所有用户都可调用删除总结Linux

java图像识别工具类(ImageRecognitionUtils)使用实例详解

《java图像识别工具类(ImageRecognitionUtils)使用实例详解》:本文主要介绍如何在Java中使用OpenCV进行图像识别,包括图像加载、预处理、分类、人脸检测和特征提取等步骤... 目录前言1. 图像识别的背景与作用2. 设计目标3. 项目依赖4. 设计与实现 ImageRecogni

python管理工具之conda安装部署及使用详解

《python管理工具之conda安装部署及使用详解》这篇文章详细介绍了如何安装和使用conda来管理Python环境,它涵盖了从安装部署、镜像源配置到具体的conda使用方法,包括创建、激活、安装包... 目录pytpshheraerUhon管理工具:conda部署+使用一、安装部署1、 下载2、 安装3

Mysql虚拟列的使用场景

《Mysql虚拟列的使用场景》MySQL虚拟列是一种在查询时动态生成的特殊列,它不占用存储空间,可以提高查询效率和数据处理便利性,本文给大家介绍Mysql虚拟列的相关知识,感兴趣的朋友一起看看吧... 目录1. 介绍mysql虚拟列1.1 定义和作用1.2 虚拟列与普通列的区别2. MySQL虚拟列的类型2

使用MongoDB进行数据存储的操作流程

《使用MongoDB进行数据存储的操作流程》在现代应用开发中,数据存储是一个至关重要的部分,随着数据量的增大和复杂性的增加,传统的关系型数据库有时难以应对高并发和大数据量的处理需求,MongoDB作为... 目录什么是MongoDB?MongoDB的优势使用MongoDB进行数据存储1. 安装MongoDB

在C#中获取端口号与系统信息的高效实践

《在C#中获取端口号与系统信息的高效实践》在现代软件开发中,尤其是系统管理、运维、监控和性能优化等场景中,了解计算机硬件和网络的状态至关重要,C#作为一种广泛应用的编程语言,提供了丰富的API来帮助开... 目录引言1. 获取端口号信息1.1 获取活动的 TCP 和 UDP 连接说明:应用场景:2. 获取硬

关于@MapperScan和@ComponentScan的使用问题

《关于@MapperScan和@ComponentScan的使用问题》文章介绍了在使用`@MapperScan`和`@ComponentScan`时可能会遇到的包扫描冲突问题,并提供了解决方法,同时,... 目录@MapperScan和@ComponentScan的使用问题报错如下原因解决办法课外拓展总结@

mysql数据库分区的使用

《mysql数据库分区的使用》MySQL分区技术通过将大表分割成多个较小片段,提高查询性能、管理效率和数据存储效率,本文就来介绍一下mysql数据库分区的使用,感兴趣的可以了解一下... 目录【一】分区的基本概念【1】物理存储与逻辑分割【2】查询性能提升【3】数据管理与维护【4】扩展性与并行处理【二】分区的

使用Python实现在Word中添加或删除超链接

《使用Python实现在Word中添加或删除超链接》在Word文档中,超链接是一种将文本或图像连接到其他文档、网页或同一文档中不同部分的功能,本文将为大家介绍一下Python如何实现在Word中添加或... 在Word文档中,超链接是一种将文本或图像连接到其他文档、网页或同一文档中不同部分的功能。通过添加超