《嵌入式实时系统FreeRTOS.ppt》由会员分享,可在线阅读,更多相关《嵌入式实时系统FreeRTOS.ppt(152页珍藏版)》请在taowenge.com淘文阁网|工程机械CAD图纸|机械工程制图|CAD装配图下载|SolidWorks_CaTia_CAD_UG_PROE_设计图分享下载上搜索。
1、嵌 入 式 实 时 系 统 -F r e e R T O S1、FreeRTOS简介 FreeRTOS是一个嵌入式系统使用的开源实时系统。是一个嵌入式系统使用的开源实时系统。FreeRTOS小巧,简单,易用。能支持许多不同硬件小巧,简单,易用。能支持许多不同硬件 架构以及交叉编译器。架构以及交叉编译器。此系统可以免费进行商业应用,被大量公司与科研公司此系统可以免费进行商业应用,被大量公司与科研公司用于嵌入式产品的开发。用于嵌入式产品的开发。支持的处理器架构:支持的处理器架构:ARM7,ARM9,COLTEX-m3、AVR、PIC等。等。1、FreeRTOS简介n嵌入式系统:嵌入式系统:n一个专
2、门设计用来做一些简单事情的计算机系统,一个专门设计用来做一些简单事情的计算机系统,如电视遥控器,车载如电视遥控器,车载GPS,电子手表,或者起搏,电子手表,或者起搏器这类。嵌入式系统比通用计算机系统更小更慢,器这类。嵌入式系统比通用计算机系统更小更慢,通常也更便宜。通常也更便宜。n如:低端:一个运行速度为如:低端:一个运行速度为25MHz的的8位位CPU,几几KB的内存。的内存。n 高端:一个运行速度为高端:一个运行速度为750MHz的的32位位 CPU,一个,一个GB左右的内存,和几个左右的内存,和几个GB的闪存。的闪存。1、FreeRTOS简介n实时(实时(RT)n软实时:按照任务的优先级
3、,尽可能快地完成操软实时:按照任务的优先级,尽可能快地完成操作即可作即可。n硬实时:硬实时要求在规定的时间内必须完成操硬实时:硬实时要求在规定的时间内必须完成操作作。1、FreeRTOS简介系统基本架构系统基本架构其代码可以分解为三个主要区块:其代码可以分解为三个主要区块:任务任务通讯通讯硬件接口硬件接口任务:大约任务:大约50%的的FreeRTOS的核心代码的核心代码1、FreeRTOS简介通讯:大约通讯:大约40%的的FreeRTOS核心代码是用来处理核心代码是用来处理通讯的。任务和中断使用队列互相发送数据,并通讯的。任务和中断使用队列互相发送数据,并且使用信号灯和互斥来发送临界资源的使用
4、情况。且使用信号灯和互斥来发送临界资源的使用情况。硬件接口:大约有硬件接口:大约有6%的的FreeRTOS的核心代码,的核心代码,在硬件无关的在硬件无关的FreeRTOS内核与硬件相关的代码内核与硬件相关的代码间扮演着垫片的角色。间扮演着垫片的角色。2、任务简介2.1任务函数任务函数 任务是由任务是由C语言函数实现的。任务函数其必须返语言函数实现的。任务函数其必须返回回void,而且带有一个而且带有一个void指针参数。其函数原指针参数。其函数原型参见程序清单型参见程序清单1。void ATaskFunction(void*pvParameters);程序清单程序清单1 任务函数原型任务函数原
5、型2.1任务函数任务函数注意事项:注意事项:FreeRTOS 任务不允许以任何方式从实现函数中返回任务不允许以任何方式从实现函数中返回它们绝不能有一条它们绝不能有一条”return”语句,也不能执行到函数语句,也不能执行到函数末尾。如果一个任务不再需要,可以显式地将其删除末尾。如果一个任务不再需要,可以显式地将其删除(void vTaskDelete(xTaskHandlepxTaskToDelete);)。)。2.1任务函数任务函数创建任务创建任务:创建任务使用创建任务使用FreeRTOS的的API函数函数xTaskCreate()。程序清单程序清单22.1任务函数任务函数n参数介绍参数介绍
6、:npvTaskCode 一个指向任务的实现函数的指针一个指向任务的实现函数的指针(效果上仅仅是函数效果上仅仅是函数n名名)。npcName 具有描述性的任务名。这个参数不会被具有描述性的任务名。这个参数不会被FreeRTOS使用。其使用。其只是单只是单n纯地用于辅助调试。纯地用于辅助调试。usStackDepth 当任务创建时,内核会分为每个当任务创建时,内核会分为每个任务分配属于任务自己的唯一状态。任务分配属于任务自己的唯一状态。nusStackDepth值用于告诉内核为它分配多大的栈空间。值用于告诉内核为它分配多大的栈空间。n这个值指定的是栈空间可以保存多少个字这个值指定的是栈空间可以保
7、存多少个字(word),而不是多少个字,而不是多少个字n节节(byte)。比如说,如果是。比如说,如果是32位宽的栈空间,传入的位宽的栈空间,传入的usStackDepthn值为值为100,则将会分配,则将会分配400字节的栈空间字节的栈空间(100*4bytes)。n。n参数:参数:npvParameters 任务函数接受一个指向任务函数接受一个指向void的指针的指针(void*)。pvParameters的值即是传递到任务中的值。的值即是传递到任务中的值。nuxPriority 指定任务执行的优先级。指定任务执行的优先级。优先级的取值范围可以从最低优先级优先级的取值范围可以从最低优先级0
8、到最高优先级到最高优先级(configMAX_PRIORITIES 1)。nconfigMAX_PRIORITIES 是一个由用户定义的常量是一个由用户定义的常量。npxCreatedTask 用于传出任务的句柄。这个句柄将在用于传出任务的句柄。这个句柄将在API调用中对该创建出调用中对该创建出来的任务进行引用,比如改变任务优先级,或者删除任务。如果应用程序中来的任务进行引用,比如改变任务优先级,或者删除任务。如果应用程序中不会用到这个任务的句柄,则不会用到这个任务的句柄,则pxCreatedTask可以可以n被设为被设为NULLn返回值返回值 n有两个可能的返回值:有两个可能的返回值:n1.
9、pdTRUE n表明任务创建成功。表明任务创建成功。n2.errCOULD_NOT_ALLOCATE_REQUIRED_MEMORY n由于内存堆空间不足,由于内存堆空间不足,FreeRTOS无法分配足够的空间来无法分配足够的空间来保存任务结构数据和任务栈,因此无法创建任务。保存任务结构数据和任务栈,因此无法创建任务。n第五章将提供更多有关内存管理方面的信息。第五章将提供更多有关内存管理方面的信息。2.1任务函数任务函数例例1.创建任务创建任务本例演示了创建并启动两个任务的必要步骤。本例演示了创建并启动两个任务的必要步骤。两个任务内容:两个任务内容:周期性地打印输出字符串。周期性地打印输出字符
10、串。两者在创建时指定了相同的优先级。两者在创建时指定了相同的优先级。任务一:周期性输出任务一:周期性输出:Task 1 is runningrn任务二:周期性输出任务二:周期性输出:Task 2 is runningrnmain()函数:简单地创建这两个任务,然后启动调函数:简单地创建这两个任务,然后启动调度器度器运行结果:运行结果:图一图一 2.1任务函数任务函数任务分配分析:任务分配分析:图图2中看到两个任务在同时运行,事实上这两个任务中看到两个任务在同时运行,事实上这两个任务由于优先级相同,而且在一个处理器上运行,其由于优先级相同,而且在一个处理器上运行,其实是在交替运行。实是在交替运行
11、。真实的执行流程所图真实的执行流程所图3所示所示 2.1任务函数任务函数 图二2.1任务函数任务函数 上例中上例中main()函数在启动调度器之前先完成两个任务函数在启动调度器之前先完成两个任务 的创建。当然也可以从一个任务中创建另一个任务。的创建。当然也可以从一个任务中创建另一个任务。我们可以先在我们可以先在main()中创建任务中创建任务1,然后在任务,然后在任务1中创中创建任务建任务2这样需要在任务一中添加以下语句:这样需要在任务一中添加以下语句:xTaskCreate(vTask2,Task 2,1000,NULL,1,NULL);2.1任务函数任务函数n关于使用关于使用xTaskCr
12、eate()时任务函数的参数时任务函数的参数问题问题n例例1中的两个任务几乎完全相同,唯一的区中的两个任务几乎完全相同,唯一的区别就是打印输出的字符串。这种重复性可别就是打印输出的字符串。这种重复性可以通过创建同一个任务代码的两个实例来以通过创建同一个任务代码的两个实例来去除。去除。n这时任务参数就可以用来传递各自打印输这时任务参数就可以用来传递各自打印输出的字符串。出的字符串。我们仍然调用两次xTaskCreate(),但其,但其pvParameters(第四个)(第四个)参数变为了两个不同的指针,两个指针分别指向各自需要打印输出的文本。2.2任务调度任务的调度方法:任务的调度方法:通过对任
13、务设置优先级进行优先级抢占式调度。通过对任务设置优先级进行优先级抢占式调度。每个任务都赋予了一个优先级。每个任务都赋予了一个优先级。每个任务都可以存在于一个或多个状态。每个任务都可以存在于一个或多个状态。在任何时候都只有一个任务可以处于运行状态。在任何时候都只有一个任务可以处于运行状态。调度器总是在所有处于就绪态的任务中选择具调度器总是在所有处于就绪态的任务中选择具有最高优先级的任务来执行。有最高优先级的任务来执行。2.2任务调度优先级:优先级:xTaskCreate()API函数的参数函数的参数uxPriority(即第五个(即第五个参数)为创建的任务赋予了一个初始优先级。参数)为创建的任务
14、赋予了一个初始优先级。常量常量configMAX_PRIORITIES在在(FreeRTOSConfig.h文件中)的值,即是系统最多可具文件中)的值,即是系统最多可具有的优先级数目。有的优先级数目。0到到(configMAX_PRIORITES 1)函数优先级可以调用函数优先级可以调用vTaskPrioritySet()API函数进行函数进行修改。修改。2.2任务调度n关于优先级为零的任务介绍:关于优先级为零的任务介绍:n 当创建的任务都处于阻塞态时(很多时候都会有这种当创建的任务都处于阻塞态时(很多时候都会有这种情况出现),这种状态下所有的任务都不可运行,但处理情况出现),这种状态下所有的
15、任务都不可运行,但处理器总是需要代码来执行器总是需要代码来执行所以至少要有一个任务处于运所以至少要有一个任务处于运行态。行态。n 为了保证这一点,当调用为了保证这一点,当调用vTaskStartScheduler()时,时,调度器会自动创建一个空闲任务(非常短小的循环)。调度器会自动创建一个空闲任务(非常短小的循环)。n空闲任务拥有最低优先级空闲任务拥有最低优先级(优先级优先级0),这样不会妨碍具有,这样不会妨碍具有更高优先级的应用任务进入运行态。更高优先级的应用任务进入运行态。2.2任务调度n图中任务一(图中任务一(Task1)任务二()任务二(Task2)都设置有定时阻)都设置有定时阻塞,
16、任务二的优先级高于任务一的优先级,在两个任务都塞,任务二的优先级高于任务一的优先级,在两个任务都被阻塞的时候,空闲任务被阻塞的时候,空闲任务(idle)开始执行。开始执行。n 图三图三2.2任务调度n空闲任务钩子函数:空闲任务钩子函数:n 通过空闲任务钩子函数通过空闲任务钩子函数(或称回调,或称回调,hook,or call-back),可以直接在空闲任务中添加应用程序,可以直接在空闲任务中添加应用程序相关的功能。空闲任务钩子函数会被空闲任务每相关的功能。空闲任务钩子函数会被空闲任务每循环一次就自动调用一次。循环一次就自动调用一次。n 空闲任务钩子函数有很多用途,如在没有任何空闲任务钩子函数有
17、很多用途,如在没有任何应用功能需要处理的时候,将系统配置到省电模应用功能需要处理的时候,将系统配置到省电模式。式。2.2任务调度n定义一个空闲任务钩子函数:定义一个空闲任务钩子函数:nvoid vApplicationIdleHook(void);FreeRTOSConfig.h中的配置常中的配置常configUSE_IDLE_HOOK必须定义为必须定义为1,这样空闲任务钩子函数才会被调用。,这样空闲任务钩子函数才会被调用。2.2任务调度n空闲任务钩子函数必须遵从以下规则:空闲任务钩子函数必须遵从以下规则:n1.绝不能阻塞或挂起。空闲任务的目的就是希望绝不能阻塞或挂起。空闲任务的目的就是希望在
18、所有任务阻塞时仍能有任务运行,一旦钩子函在所有任务阻塞时仍能有任务运行,一旦钩子函数阻塞或挂起,将不再有任务运行。数阻塞或挂起,将不再有任务运行。n2.如果应用程序用到了如果应用程序用到了vTaskDelete()函数,则函数,则空闲钩子函数必须能够尽快返回。空闲钩子函数必须能够尽快返回。n因为在任务被删除后(因为在任务被删除后(vTaskDelete()),空闲任),空闲任务负责回收内核资源。如果空闲任务一直运行在务负责回收内核资源。如果空闲任务一直运行在钩子函数中,则无法进行回收工作。钩子函数中,则无法进行回收工作。2.2任务调度n任务优先级的改变任务优先级的改变n API函数函数vTas
19、kPriofitySet()可以用于在调度器启动后可以用于在调度器启动后改变任何任务的优先级。改变任何任务的优先级。n void vTaskPrioritySet(xTaskHandle pxTask,unsigned portBASE_TYPE uxNewPriority);n 参数:参数:n pxTask 被修改优先级的任务句柄被修改优先级的任务句柄(即目标任务即目标任务)n uxNewPriority 目标任务将被设置到哪个优先级上。目标任务将被设置到哪个优先级上。如果设置的值超过了最大可用优先级如果设置的值超过了最大可用优先级(configMAX_PRIORITIES 1),则会被自动
20、封顶为最大值。,则会被自动封顶为最大值。2.2任务调度n根据优先级的具体调度方法:根据优先级的具体调度方法:n调度器保证总是在所有可运行的任务中选调度器保证总是在所有可运行的任务中选择具有最高优先级的任务,并使其进入运择具有最高优先级的任务,并使其进入运n行态。行态。n如果被选中的优先级上具有不止一个任务,如果被选中的优先级上具有不止一个任务,调度器会让这些任务轮流执行调度器会让这些任务轮流执行(如图二如图二)。2.2任务调度n时间片的概念时间片的概念n 图二中的两个任务优先级相同,所以每个任务图二中的两个任务优先级相同,所以每个任务都执行一个都执行一个”时间片时间片”,任务在时间片起始时刻,
21、任务在时间片起始时刻进入运行态,在时间片结束时刻又退出运行态。进入运行态,在时间片结束时刻又退出运行态。图图2中中t1与与t2之间的时段就等于一个时间片。之间的时段就等于一个时间片。n 要能够选择下一个运行的任务,调度器需要在要能够选择下一个运行的任务,调度器需要在每个时间片的结束时刻运行自己本身。一个称为每个时间片的结束时刻运行自己本身。一个称为心跳心跳(tick)中断的周期性中断用于此目的。时间中断的周期性中断用于此目的。时间片的长度是可以设定的。片的长度是可以设定的。2.2任务调度n此图中的任务一与任务二与图此图中的任务一与任务二与图2中的是一样的,此图中标出了调度器中的是一样的,此图中
22、标出了调度器在时间片之间的调度过程。在时间片之间的调度过程。n 图图42.2任务调度n任务状态任务状态n在简要介绍了优先级调度后,我们需要了解与调度相关的各种任务的在简要介绍了优先级调度后,我们需要了解与调度相关的各种任务的状态。状态。n1、运行态、运行态.n运行中的任务状态。运行中的任务状态。n2、阻塞状态、阻塞状态n如果一个任务正在等待某个事件,则称这个任务处于如果一个任务正在等待某个事件,则称这个任务处于”阻塞态阻塞态(blocked)”。n挂起状态。挂起状态。n3、“挂起挂起(suspended)”处于挂起状态的任务对调度器而言是不可见处于挂起状态的任务对调度器而言是不可见的。让一个任
23、务进入挂起状态的唯一办法就是调用的。让一个任务进入挂起状态的唯一办法就是调用vTaskSuspend()API函数。函数。n4、就绪状态、就绪状态n如果任务处于非运行状态,但既没有阻塞也没有挂起,则这个任务处如果任务处于非运行状态,但既没有阻塞也没有挂起,则这个任务处于就绪于就绪(ready,准备或就绪,准备或就绪)状态。状态。2.2任务调度n完整的状态转移图完整的状态转移图 图53、队列n 独立的任务之间很可能会通过相互通信以提供有用的系独立的任务之间很可能会通过相互通信以提供有用的系统功能。统功能。n FreeRTOS中所有的通信与同步机制都是基于队列实现中所有的通信与同步机制都是基于队列
24、实现的。的。n特性:特性:n1、数据存储、数据存储n2、队列可以保存有限个具有确定长度的数据单元、队列可以保存有限个具有确定长度的数据单元n3、可被多任务存取、可被多任务存取n4、队列是具有自己独立权限的内核对象,并不属于或赋、队列是具有自己独立权限的内核对象,并不属于或赋予任何任务。所有任务都可以向同一队列写入和读出。予任何任务。所有任务都可以向同一队列写入和读出。3、队列n5、读队列时阻塞读队列时阻塞n 当某个任务试图读一个队列时,其可以指定一个阻塞超当某个任务试图读一个队列时,其可以指定一个阻塞超时时间。在这段时间中,如时时间。在这段时间中,如n果队列为空,该任务将保持阻塞状态以等待队列
25、数据有效。果队列为空,该任务将保持阻塞状态以等待队列数据有效。n6、写队列时阻塞、写队列时阻塞n 同读队列一样,任务也可以在写队列时指定一个阻塞超同读队列一样,任务也可以在写队列时指定一个阻塞超时时间。这个时间是当被写时时间。这个时间是当被写n队列已满时,任务进入阻塞态以等待队列空间有效的最长队列已满时,任务进入阻塞态以等待队列空间有效的最长时间。时间。3、队列n创建队列:创建队列:nxQueueHandle xQueueCreate(unsigned portBASE_TYPE uxQueueLength,unsigned portBASE_TYPE uxItemSize);n参数:参数:n
26、uxQueueLength 队列能够存储的最大单元数目,即队列深度。队列能够存储的最大单元数目,即队列深度。nuxItemSize 队列中数据单元的长度,以字节为单位。队列中数据单元的长度,以字节为单位。n返回值:返回值:NULL表示没有足够的堆空间分配给队列而导致创建失败。表示没有足够的堆空间分配给队列而导致创建失败。n非非NULL值表示队列创建成功。此返回值应当保存下来,以作为值表示队列创建成功。此返回值应当保存下来,以作为n操作此队列的句柄。操作此队列的句柄。3、队列n向列中写数据向列中写数据nxQueueSendToBack()与与xQueueSendToFront()API函数函数n
27、portBASE_TYPE xQueueSendToFront(xQueueHandle xQueue,const void*pvItemToQueue,portTickType xTicksToWait);/将数据发送到队将数据发送到队列尾部列尾部3、队列n参数说明参数说明nxQueue 目标队列的句柄。这个句柄即是调用目标队列的句柄。这个句柄即是调用xQueueCreate()创建该队列时的返回值。创建该队列时的返回值。npvItemToQueue 发送数据的指针。其指向将要发送数据的指针。其指向将要复制到目标队列中的数据单元。复制到目标队列中的数据单元。nxTicksToWait 阻塞超
28、时时间。如果在发送时队阻塞超时时间。如果在发送时队列已满,这个时间即是任务处于阻塞态等待队列列已满,这个时间即是任务处于阻塞态等待队列空间有效的最长等待时间。空间有效的最长等待时间。3、队列n返回值返回值 n有两个可能的返回值有两个可能的返回值:n1.pdPASS n 数据被成功发送到队列中。数据被成功发送到队列中。n2.errQUEUE_FULL n 如果由于队列已满而无法将数据写入,则将返回。如果由于队列已满而无法将数据写入,则将返回。3、队列nortBASE _TYPE xQueueSendToBack(xQueueHandle xQueue,const void*pvItemToQue
29、ue,portTickType xTicksToWait);/将数据发送到队列首部将数据发送到队列首部n其参数和返回值与前者相似,这里不再介其参数和返回值与前者相似,这里不再介绍。绍。3、队列nxQueueReceive()与与xQueuePeek()API函数函数nxQueueReceive()用于从队列中接收用于从队列中接收(读取)数据读取)数据单元。接收到的单元同时会从队列单元。接收到的单元同时会从队列n中删除。中删除。nxQueuePeek()也是从从队列中接收数据单元,不也是从从队列中接收数据单元,不同的是并不从队列中删出接收到同的是并不从队列中删出接收到n的单元。的单元。3、队列n
30、portBASE_TYPE xQueueReceive(xQueueHandle xQueue,nconst void*pvBuffer,nportTickType xTicksToWait);n参数:参数:nxQueue 被读队列的句柄。这个句柄即是调用xQueueCreate()创建该队列n时的返回值。npvBuffer 接收缓存指针。其指向一段内存区域,用于接收从队列中拷贝来n的数据。nxTicksToWait 阻塞超时时间。如果在接收时队列为空,则这个时间是任务处于n阻塞状态以等待队列数据有效的最长等待时间。把它设为0,并且队列为空,则xQueueRecieve()与xQueuePee
31、k()均会立即返回。n 如果把xTicksToWait 设置为portMAX_DELAY,并且在nFreeRTOSConig.h中设定INCLUDE_vTaskSuspend 为1,那么n阻塞等待将没有超时限制。3、队列n返回值返回值n有两个可能的返回值有两个可能的返回值:n1.pdPASS n只有一种情况会返回只有一种情况会返回pdPASS,那就是成功地从队列中读到数据。,那就是成功地从队列中读到数据。n2.errQUEUE_FULL n如果在读取时由于队列已空而没有读到任何数据,则将返回如果在读取时由于队列已空而没有读到任何数据,则将返回nerrQUEUE_FULL。n如果设定了阻塞超时时
32、间(如果设定了阻塞超时时间(xTicksToWait非非0),在函数返回之前),在函数返回之前n任务将被转移到阻塞态以等待队列数据有效。但直到超时也没有任务将被转移到阻塞态以等待队列数据有效。但直到超时也没有n其它任务或是中断服务例程往队列中写入数据,函数则会返回其它任务或是中断服务例程往队列中写入数据,函数则会返回nerrQUEUE_FULL。nxQueuePeek()的参数与返回值与的参数与返回值与xQueueReceive()类似,这里不再重复。类似,这里不再重复。3、队列nuxQueueMessagesWaiting()用于查询队列中当前有效数用于查询队列中当前有效数据单元个数据单元个
33、数n函数原型:函数原型:nunsigned portBASE_TYPE uxQueueMessagesWaiting(xQueueHandle xQueue);n参数参数 nxQueue 被查询队列的句柄。这个句柄即是调用被查询队列的句柄。这个句柄即是调用xQueueCreate()创建该队列时创建该队列时n的返回值。的返回值。n返回值返回值 n当前队列中保存的数据单元个数。返回当前队列中保存的数据单元个数。返回0表明队列为空。表明队列为空。3、队列n 以上五个函数均不能在中断服务中使用,系统会提供以以上五个函数均不能在中断服务中使用,系统会提供以上函数专门的中断版本(后面将会讲到)。上函数专
34、门的中断版本(后面将会讲到)。3、队列n队列使用例程队列使用例程n例例2:n本例示范创建一个队列,由多个任务往队列中写本例示范创建一个队列,由多个任务往队列中写数据,以及从队列中把数据读出。数据,以及从队列中把数据读出。n其中往队列中写数据的任务没有设定阻塞超时时其中往队列中写数据的任务没有设定阻塞超时时间,而读队列的任务设定了超时时间。间,而读队列的任务设定了超时时间。3、队列、队列n写队列任务函数。主函数将调用此函数分别写入写队列任务函数。主函数将调用此函数分别写入100、200.3、队列、队列n写队列任务在每次循环中都调用写队列任务在每次循环中都调用taskYIELD()。taskYIE
35、LD()通知调度通知调度器立即进行任务切换,而不必等到当前任务的时间片耗尽。器立即进行任务切换,而不必等到当前任务的时间片耗尽。3、队列、队列n读队列任务函数,函数中设置了延迟时间。读队列任务函数,函数中设置了延迟时间。3、队列、队列3、队列、队列nmain函数:其在启动调度器之前创建了一个队列和三个任务。尽管对任务的函数:其在启动调度器之前创建了一个队列和三个任务。尽管对任务的优先级的设计使得队列实际上在任何时候都不可能多于一个数据单元,本例优先级的设计使得队列实际上在任何时候都不可能多于一个数据单元,本例代码还是创建了一个可以保存最多代码还是创建了一个可以保存最多5个个long 型值的队列
36、。型值的队列。3、队列、队列3、队列、队列n 写队列任务在每次循环中都调用写队列任务在每次循环中都调用taskYIELD()。taskYIELD()通知调度器立通知调度器立即进行任务切换,而不必等到当前任务的时间片耗尽。本例中两个写队列任即进行任务切换,而不必等到当前任务的时间片耗尽。本例中两个写队列任务(写入务(写入100和和200)是具有相同的任务优先级(均为)是具有相同的任务优先级(均为1),所以一旦其中一),所以一旦其中一个任务调用了个任务调用了taskYIELD(),另一个任务将会得到执行。反映到输出上就是两,另一个任务将会得到执行。反映到输出上就是两个人物交替输出。个人物交替输出。
37、图图63、队列、队列n调用过程如下调用过程如下 图图7n 3、队列、队列n使用队列传递复合数据类型使用队列传递复合数据类型 3、队列、队列n对大型数据单元的传递对大型数据单元的传递 4、中断、中断 n4.1延迟中断处理延迟中断处理n 当中断发生时,程序将进入中断服务程序中行,如果当中断发生时,程序将进入中断服务程序中行,如果中断服务程序运行时间很长,甚至包括迟将大大降低系统中断服务程序运行时间很长,甚至包括迟将大大降低系统的实时性。这时,我们可以将中断服务程序作为触发者,的实时性。这时,我们可以将中断服务程序作为触发者,触发其他函数来执行需要执行的任务。这样中断服务程序触发其他函数来执行需要执
38、行的任务。这样中断服务程序可以很简短,大大增强了实时性。而真正需要做的事情则可以很简短,大大增强了实时性。而真正需要做的事情则放在被触发的函数中执行。放在被触发的函数中执行。n 我们可以使用二值信号量完成中断服务程序与被触发函我们可以使用二值信号量完成中断服务程序与被触发函数的这种触发关系。数的这种触发关系。4.1延迟中断处理延迟中断处理n延迟中断处理如下图所示延迟中断处理如下图所示:图图84.1延迟中断处理延迟中断处理n延迟处理任务(即被中断服务程序触发的函数)对一个信延迟处理任务(即被中断服务程序触发的函数)对一个信号量进行带阻塞性质的号量进行带阻塞性质的”take”调用,意思是进入阻塞态
39、调用,意思是进入阻塞态以等待事件发生。当事件发生后,以等待事件发生。当事件发生后,ISR 对同一个信号量进对同一个信号量进行行”give”操作,使得延迟处理任务解除阻塞,从而事件操作,使得延迟处理任务解除阻塞,从而事件在延迟处理任务中得到相应的处理。在延迟处理任务中得到相应的处理。4.1延迟中断处理延迟中断处理n 在此情形下,信号量可以看作是一个深度为在此情形下,信号量可以看作是一个深度为1的队列。这个队列由的队列。这个队列由于最多只能保存一个数据单元,所以其不为空则为满于最多只能保存一个数据单元,所以其不为空则为满(所谓所谓”二值二值”)。n 延迟处理任务调用延迟处理任务调用xSemapho
40、reTake()时,等效于带阻塞时间地读时,等效于带阻塞时间地读取队列,如果队列为空的话任务则进入阻塞态。当事件发生后,取队列,如果队列为空的话任务则进入阻塞态。当事件发生后,ISR(中断服务例程)(中断服务例程)简单地通过调用简单地通过调用xSemaphoreGiveFromISR()放置一个令牌放置一个令牌(信号量信号量)到队列中,使得队列成为满状态。这使得延迟到队列中,使得队列成为满状态。这使得延迟处理任务切出阻塞态,进入运行态。处理任务切出阻塞态,进入运行态。n 当任务完成处理后,再次读取队列,发现队列为空,又进入阻塞态,当任务完成处理后,再次读取队列,发现队列为空,又进入阻塞态,等待
41、下一次事件发生。等待下一次事件发生。4.1延迟中断处理延迟中断处理n以上过程如下图所示以上过程如下图所示:图图94.1延迟中断处理延迟中断处理4.1延迟中断处理延迟中断处理nvSemaphoreCreateBinary()API函数函数n 在使用信号量之前,必须先创建它。创建二值信号量使用在使用信号量之前,必须先创建它。创建二值信号量使用nvSemaphoreCreateBinary()API函数。函数。nvoid vSemaphoreCreateBinary(xSemaphoreHandle xSemaphore);n参数:参数:nxSemaphore 创建的信号量创建的信号量n需要说明的是
42、需要说明的是vSemaphoreCreateBinary()在实现上是一个宏,所以在实现上是一个宏,所以n信号量变量应当直接传入,而不是传址。本章中包含本函数调用的示信号量变量应当直接传入,而不是传址。本章中包含本函数调用的示n例可用于参考进行复制例可用于参考进行复制4.1延迟中断处理延迟中断处理nxSemaphoreTake()API 函数函数n延迟任务处理函数,调用此函数获取信号量。延迟任务处理函数,调用此函数获取信号量。n只有当信号量有效的时候才可以被获取。除互斥信号量只有当信号量有效的时候才可以被获取。除互斥信号量(Recursive Semaphore)外,所有类型的信号量都可以调外
43、,所有类型的信号量都可以调用函数用函数xSemaphoreTake()来获取。来获取。n但但xSemaphoreTake()不能在中断服务例程中调用。不能在中断服务例程中调用。4.1延迟中断处理延迟中断处理nportBASE_TYPE xSemaphoreTake(xSemaphoreHandle xSemaphore,portTickType xTicksToWait);n参数:参数:nxSemaphore 需要获取的信号量需要获取的信号量nxTicksToWait 阻塞超时时间。任务进入阻塞态以等待信阻塞超时时间。任务进入阻塞态以等待信号量有效的最长时间。号量有效的最长时间。n如果如果xT
44、icksToWait为为0,则,则xSemaphoreTake()在信号量在信号量无效时会无效时会n立即返回。立即返回。4.1延迟中断处理延迟中断处理n返回值返回值 有两个可能的返回值有两个可能的返回值:n1.pdPASS n只有一种情况会返回只有一种情况会返回pdPASS,那就是成功获得信号量。,那就是成功获得信号量。n2.pdFALSE n未能获得信号量。未能获得信号量。4.1延迟中断处理延迟中断处理nxSemaphoreGiveFromISR()API 函数函数n中断服务程序调用此函数给出信号量。中断服务程序调用此函数给出信号量。n n 所有信号量(除互斥信号量)都可以通过调用所有信号量
45、(除互斥信号量)都可以通过调用xSemaphoreGiveFromISR()给出。给出。n xSemaphoreGiveFromISR()是是xSemaphoreGive()的的特殊形式,专门用于中断服务例程中。特殊形式,专门用于中断服务例程中。4.1延迟中断处理延迟中断处理nportBASE_TYPE xSemaphoreGiveFromISR(xSemaphoreHandle xSemaphore,nportBASE_TYPE*pxHigherPriorityTaskWoken);n参数:参数:nxSemaphore 给出的信号量给出的信号量npxHigherPriorityTaskWok
46、en 对某个信号量而言,可能对某个信号量而言,可能有不止一个任务处于阻塞态在等待其有效。调用有不止一个任务处于阻塞态在等待其有效。调用xSemaphoreGiveFromISR()会让信号量变为有效,所以会让其中一个等待任务切出会让信号量变为有效,所以会让其中一个等待任务切出阻塞态。阻塞态。4.1延迟中断处理延迟中断处理n返回值返回值 有两个可能的返回值有两个可能的返回值:n1.pdPASS nxSemaphoreGiveFromISR()调用成功。调用成功。n2.pdFAIL n如果信号量已经有效,无法给出,则返回如果信号量已经有效,无法给出,则返回pdFAIL。4.1延迟中断处理延迟中断处
47、理n例三例三n本例在中断服务例程中使用一个二值信号量让延迟处理任本例在中断服务例程中使用一个二值信号量让延迟处理任务从阻塞态中切换出来。务从阻塞态中切换出来。n下面的一个简单的周期性任务用于每隔下面的一个简单的周期性任务用于每隔500毫秒产生一个毫秒产生一个软件中断。此任务在产生中断之前和之后都会打印输出一软件中断。此任务在产生中断之前和之后都会打印输出一个字符串。个字符串。4.1延迟中断处理延迟中断处理4.1延迟中断处理延迟中断处理n 下面的任务展现的是延迟处理任务的具体实现下面的任务展现的是延迟处理任务的具体实现此任此任务通过使用二值信号量与软件中断进行同步。务通过使用二值信号量与软件中断
48、进行同步。n 这个任务也在每次循环中打印输出一个信息,这样做这个任务也在每次循环中打印输出一个信息,这样做的目的同样是可以在程序的执行输出结果中直观地看出任的目的同样是可以在程序的执行输出结果中直观地看出任务与中断的执行流程。务与中断的执行流程。4.1延迟中断处理延迟中断处理4.1延迟中断处理延迟中断处理n 下个任务展现的是中断服务例程,这才是真正的中断处下个任务展现的是中断服务例程,这才是真正的中断处理程序。理程序。n 这段代码做的事情非常少,仅仅是给出一个信号量,这段代码做的事情非常少,仅仅是给出一个信号量,以让延迟处理任务解除阻塞。以让延迟处理任务解除阻塞。4.1延迟中断处理延迟中断处理
49、4.1延迟中断处理n main()函数很简单,创建二值信号量及任务,安装中函数很简单,创建二值信号量及任务,安装中断服务例程,然后启动调度器。断服务例程,然后启动调度器。4.1延迟中断处理延迟中断处理n例例3整个执行流程可以描述如下整个执行流程可以描述如下:n1.中断产生。中断产生。n2.中断服务例程启动,给出信号量以使延迟处理中断服务例程启动,给出信号量以使延迟处理任务解除阻塞。任务解除阻塞。n3.当中断服务例程退出时,延迟处理任务得到执当中断服务例程退出时,延迟处理任务得到执行。延迟处理任务做的第一件事便是获取信号量。行。延迟处理任务做的第一件事便是获取信号量。n4.延迟处理任务完成中断事
50、件处理后,试图再次延迟处理任务完成中断事件处理后,试图再次获取信号量获取信号量如果此时信号量无效,任务将切如果此时信号量无效,任务将切入阻塞待等待事件发生。入阻塞待等待事件发生。4.1延迟中断处理延迟中断处理n输出结果参见下图。和期望的一样,延迟处理任务在中断产生后立即输出结果参见下图。和期望的一样,延迟处理任务在中断产生后立即执行。所以延迟处理任务的输出信息将周期任务的两条输出信息分开。执行。所以延迟处理任务的输出信息将周期任务的两条输出信息分开。n 图图104.1延迟中断处理延迟中断处理n执行流程执行流程:图图114.2计数信号量计数信号量n 在中断以相对较慢的频率发生的情况下,二值信号量