UCOS-II培训材料.doc_第1页
UCOS-II培训材料.doc_第2页
UCOS-II培训材料.doc_第3页
UCOS-II培训材料.doc_第4页
UCOS-II培训材料.doc_第5页
已阅读5页,还剩20页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

21ic中国电子网:/ 21ic电子技术论坛:/ 内容介绍1、 介绍uC/OS-II嵌入式操作系统2、 基于uC/OS-II的用电管理终端软件的设计书籍:嵌入式实时操作系统uC/OS-II作者:Jean LabrosseuC/OS-II V2.52 通过了美国航空航天管理局(FAA)的安全认证;安全性、可靠性是得到认证的。我们为什么会选择uC/OS-II嵌入式操作系统?1、 与终端硬件平台相适应全部源代码5500行,可裁减定制,生成的可执行代码占1520k,可以移植到多种系列单片机上,包括ARM;2、 考虑成本,免费的源代码公开;3、 uC/OS-II代码简单,容易掌握和使用;具有多任务调度的基本功能;uC/OS-II嵌入式操作系统的缺点:1、缺少技术支持,相关的支持软件少;2、和商业软件比,功能较弱(如不支持时间片轮转,最大任务数为64等);对应用开发的支持不够;uC/OS内核介绍和基于RTOS的设计介绍一、概述l 使用嵌入式RTOS的优点1 将复杂的系统分解为多个相对独立的任务,采用“分而治之”的方法降低系统的复杂度。通过将应用程序分割成若干独立的任务,RTOS使得应用程序的设计过程大为简化;2 使得应用程序的设计和扩展变得容易,无需较大的改动就可以增加新的功能;3 用户给系统增加一些低优先级的任务,则用户系统对高优先级的任务的响应时间几乎不受影响;4 实时性能得到提高。使用可剥夺型内核,所有时间要求苛刻的事件都得到了尽可能快捷有效的处理;5 通过有效的服务,如信号量、邮箱、队列、延时及超时等,RTOS使资源得到更好的利用;l 使用嵌入式RTOS的缺点1 使用RTOS增加了系统的内存和CPU等使用开销,例如任务之间的通讯、RTOS的调度程序等;2 需要采用一些新的软件设计方法,对系统设计人员的要求高一些。例如驱动程序的设计要考虑到共享资源的互斥问题;3 系统任务的划分是比较复杂的过程,需要设计人员对业务和RTOS操作系统都很熟悉。l uC/OS操作系统的特点uC/OS是一个完成的,可移植、可固化、可裁减的抢占式实时多任务操作系统内核。主要用ANSI的C语言编写,少部分代码是汇编语言。uC/OS主要有以下特点:1、可移植性 可以移植到多个CPU上,包括三菱单片机。2、可固化 可以固化到嵌入式系统中3、可裁减可以定制uC/OS,使用少量的系统服务4、可剥夺性 uC/OS是完全可剥夺的实时内核,uC/OS总是运行优先级最高的就绪任务。5、多任务运行 uC/OS可以管理最多64个任务。不支持时间片轮转调度法,所以要求每个任务的优先级不一样。6、可确定性 uC/OS的函数调用和系统服务的执行时间可以确定。7、任务栈 每个任务都有自己的单独的栈,而且每个任务栈空间的大小可以不一样。8、系统服务 uC/OS有很多系统服务,如信号量、时间标志、消息邮箱、消息队列、时间管理等等。二、uC/OS内核介绍l 基本概念1、 前后台系统 也称为超循环系统。应用程序是一个无限的循环,循环中实现相应的操作,这部分看成后台行为。用中断服务程序处理异步事件,处理实时性要求很强的操作,这部分可以看成前台行为。2、 共享资源 可以被一个以上任务使用的资源叫做共享资源。3、 任务:一个任务是一个线程,一般是一个无限的循环程序。一个任务可以认为CPU资源完全只属于自己。任务可以是以下五种状态之一:休眠态,就绪态,运行态,挂起态和被中断态。uC/OS-II提供的系统服务可以使任务从一种状态变为另一种状态。4、 任务切换:任务切换就是上下文切换,也是CPU寄存器内容切换。当内核决定运行另外的任务时,它保存正在运行任务的当前状态(CPU寄存器的内容)到任务自己的栈区。入栈完成后,就把下一个将要运行的任务状态从该任务的栈中重新装入CPU寄存器,并开始下一个任务的运行,这个过程叫做任务切换。5、 内核 多任务系统中内核负责管理和调度各个任务,为每个任务分配CPU时间,并负责任务间的通信。内核总是调度就绪态的优先级最高的任务。内核本身增加了系统的额外负荷,因为内核提供的服务需要一定的执行时间。6、 可剥夺型内核 uC/OSII以及绝大多数商业实时内核都是可剥夺型内核。最高优先级的任务一旦就绪,就抢占运行着的低优先级的任务,得到CPU的使用权。7、 可重入函数:可以被多个任务调用,并且不用担心数据会被破坏的函数。8、 优先级反转 优先级反转问题是使用实时内核系统中出现最多的问题。描述如下:假设当前系统有任务3在运行,并且低优先级的任务3占用了共享资源,而高优先级任务1就绪得到CPU使用权后,也要使用任务3占用的共享资源,任务1只能挂起等待任务3使用完共享资源。任务3继续运行时,优先级在任务1和任务3之间的任务2就绪并抢占了任务3的CPU使用权,直到运行完后才把CPU使用权还给任务3。任务3继续运行,在释放了共享资源后任务1才得以运行。这样,任务1实际上降到了任务3优先级的水平。这种情况就是优先级反转问题。uC/OS-II中,可以利用互斥信号量来这个解决。9、 互斥方法 使用共享数据结构进行任务间通信时,要求对其进行互斥。保证互斥的方法有:关中断、使用测试变量、禁止任务切换和利用信号量。10、同步 可以利用信号量使任务与任务,任务与ISR之间同步。任务之间没有数据交换。11、事件标志:当任务要与多个事件同步时,需要使用事件标志(event flag)。事件标志同步分为独立型同步(逻辑“或”关系)和关联型同步(逻辑“与”关系)。12、任务间通信:任务间信息的传递有两个途径,通过全局变量或者通过内核发消息给另一个任务。通过内核服务发送的消息包括:消息邮箱、消息队列。任务或者ISR可以把一个指针放到消息邮箱中,让另一个任务接收。消息队列实际上是邮箱阵列。13、时钟节拍:是特定的周期性的定时器中断。时钟节拍是系统的心脏脉动,提供周期性的信号源,是系统进行任务调度的频率依据和任务延时依据。时钟节拍越快,系统开销就越大。我们移植过程中采用的方法:初始化定时器TA0,周期是20ms,作为操作系统时钟节拍。l uC/OS-II内核结构1、uC/OS-II是以源代码形式提供的实时操作系统内核,其包含的文件结构如下:应用软件(用户代码)uC/OS-II内核文件(与处理器类型无关的代码)OS_CORE.C OS_TASK.COS_FLAG.C OS_TIME.COS_MBOX.C uCOS-II.COS_MEM.C uCOS-II.HOS_MUTEX.C OS_SEM.COS_Q.CuC/OS-II 配置文件(与应用程序有关)OS_CFG.HINCLUDES.H移植uC/OS-II(与处理器类型有关的代码)OS_CPU.H OS_CPU_C.COS_CPU_A.ASM CPU 定时器图1、uC/OS-II 的文件结构软件硬件说明:基于uC/OS-II操作系统进行应用系统时,设计任务的主要任务是将系统合理划分成多个任务,并由RTOS进行调度,任务之间使用uC/OS-II提供的系统服务进行通信,以配合实现应用系统的功能。上图中应用代码部分主要是设计人员设计的业务代码。与前后台系统一样,基于uC/OS-II的多任务系统也有一个main主函数,main函数由编译器所带的C启动程序调用。在main主函数中主要实现uC/OS-II的初始化OSInit()、任务创建、一些任务通信方法的创建、uC/OS-II的多任务启动OSStart()等常规操作。另外,还有一些应用程序相关的初始化操作,例如:硬件初始化、数据结构初始化等。 在使用uC/OS-II提供的任何功能之前,必须先调用OSInit()函数进行初始化。在main主函数中调用OSStart()启动多任务之前,至少要先建立一个任务。否则应用程序会崩溃。OSInit()初始化uC/OS-II所有的变量和数据结构,并建立空闲任务OS_TaskIdle(),这个任务总是处于就绪态。例子:一个典型的应用程序main主函数如下void main(void)/*-硬件初始化,等用户代码初始化-*/ init_mcu(); init_lcd(); init_hdtimer();OSInit();/* 初始化uC/OS-II */*通过调用OSTaskCreate ( ) 或OSTaskCreateExt ( )创建至少一个任务;*/OSTaskCreate(sample_Task, (void*)0, &sample_TaskStkTASK_STK_SIZE - 1, 2 );/*通过调用OSSemCreate() 创建信号量等任务通信方式;*/CalcSem = OSSemCreate(0);OSStart();/* 开始多任务调度!OSStart()永远不会返回 */调用OSStart()后,uC/OS-II就运行main函数所创建任务中优先级最高的一个就绪任务。用户应该在uC/OS-II启动运行后的第1个任务中调用时钟节拍启动函数。在uC/OS-II移植到M16C62的过程中实现了函数init_timer_ta0(),来初始化时钟TA0。本文后面有关部分讨论了时钟节拍的问题。上例中如果创建了多个(n个)任务,在main函数调用OSStart()后,操作系统就启动了多任务调度,接管了CPU和其他资源的使用权,负责为每个任务分配CPU使用权和使用时间,同时对共享资源进行管理。从宏观上看,整个系统就象有多个执行的程序并行运行,每个程序都是无限循环的main函数。如下图所示:.ISR1 ISR2 . ISRnmain1()Main2()Main3()Main5()Main4()Main_n( )图2 由多个任务和多个中断组成的实时多任务系统文件OS_CFG.H是与应用程序有关的配置文件,主要是对操作系统进行设置。包括:设置系统的最多任务数OS_MAX_TASKS;最多事件控制块设置 OS_MAX_EVENTS;堆栈方向的设置OS_STK_GROWTH(1为递减、0为递增);是否支持堆栈检验OS_TASK_CREATE_EXT;是否支持任务统计OS_ASK_STAT_EN;是否支持事件标志组OS_FLG_EN等等。文件INCLUDES.H是主控头文件,包含了整个系统需要的所有头文件。包括操作系统的头文件和用户设计的应用系统的头文件。OS_CPU.H、OS_CPU_A.ASM等文件是与移植uC/OS-II有关的文件,包含了与处理器类型有关的代码。这几个文件的介绍参见何博士的uC/OS-II移植文档。2、uC/OS-II内核体系结构图uC/OS-II内核主要对用户任务进行调度和管理,并为任务间共享资源提供服务。包含的模块有任务管理、任务调度、任务间通信、时间管理、内核初始化等。uC/OS-II内核体系结构如下所示:多任务应用程序任务1任务2任务3任务n任务调度任务切换OS_TASK_SW()OSIntCtxSw()内存管理任务堆栈定时器CPURAM任务间通信任务管理ISRISR任务创建任务控制块任务删除信号量互斥信号量事件标志组事件控制块消息邮箱消息队列串行I/OISR时间管理时钟节拍挂起任务恢复任务uC/OS-II启动OSStart任务延时任务延时恢复ROMuC/OS-II初始化OSInit中断服务硬件操作系统应用程序图 3、 uC/OS-II内核体系结构3、任务状态及其转换关系在多任务系统中,任务是设计者实现应用系统的基本形式,也是uC/OS-II系统进行调度的基本单元。任务可以是一个无限的循环,也可以在一次执行后被操作系统删除。任务函数和任何C函数一样,具有一个返回类型和一个参数,但是它决不返回。任务必须是以下2种结构之一:void YourTask(void pdata) for ( ; ; ) /*用户代码*/ 或void YourTask(void *pdata)/*用户代码*/OSTaskDel(OS_PRIO_SELF);在任一给定的时刻,uC/OS-II的任务状态只能是以下5种之一:l 睡眠态:指任务驻留在程序空间(ROM或RAM),还没有交给uC/OS-II来管理。通过创建任务将任务交给uC/OS-II。任务被删除后就进入睡眠态。l 就绪态:任务创建后就进入就绪态。任务的建立可以在多任务运行之前,也可以动态的由一个运行的任务建立。l 运行态:占用CPU资源运行的任务,该任务为进入就绪态的优先级最高的任务。任何时刻只能有一个任务处于运行态。l 等待状态:由于某种原因处于等待状态的任务。例如,任务自身延时一段时间,或者等待某一事件的发生。l 中断服务态:任务运行时被中断打断,进入中断服务态。正在执行的任务被挂起,中断服务子程序控制了CPU的使用权。uC/OS-II控制下的任务状态转换图如下:图4、uC/OS-II的任务状态转换图3、任务控制块(OS_TCB)任务控制块(TCB)是一个数据结构OS_TCB,一旦一个任务创建,就有一个和它关联的TCB被赋值。当任务的CPU使用权被剥夺时,它用来保存该任务的状态。这样,当任务重新获得CPU使用权时,可以从TCB中获取任务切换前的信息,准确的继续运行。任务控制块包含了许多任务信息,主要有:.OSTCBStkPtr 指向当前任务堆栈栈顶的指针。uC/OS-II允许每个任务有自己的堆栈,每个任务堆栈的大小可以不一样。.OSTCBNext 和.OSTCBPrev 指向OS_TCB双向链表的前、后连接。.OSTCBEventPtr 指向事件控制块的指针;.OSTCBDly 保存任务的延时节拍数,或允许等待事件发生的最多节拍数。.OSTCBPrio 任务的优先级;文件OS_CFG.H中定义的最多任务数OS_MAX_TASKS决定了分配给用户程序的任务控制块的数目。所有的任务控制块都放在任务控制块数组OSTCBTbl 中。uC/OS-II初始化时,所有OS_TCB都被链接成单向空任务链表。任务一旦建立,就将链表开头的OS_TCB赋给该任务。一旦任务被删除,OS_TCB就还给空任务链表。任务建立时,函数OS_TCBInit()初始化任务控制块。4、任务调度器 uC/OS-II总是运行进入就绪态的优先级最高的任务。任务调度器的功能是:在就绪表中查找最高优先级的任务,然后进行必要的任务切换,运行该任务。uC/OS-II的任务调度有两种情况:任务级的任务调度由OS_Sched()完成;中断级的任务调度由OSIntExt()完成。这两种任务调度情况调用的任务切换函数不同:任务级的任务调度OS_Sched()调用了任务切换函数 OS_TASK_SW(),而中断级的调度OSIntExt()调用了任务切换函数OSIntCtxSw()。任务级的任务调度是由于有更高优先级的任务进入就绪态,当前的任务的CPU使用权被剥夺,发生了任务到任务的切换;中断级的调度是指当前运行的任务被中断打断,由于ISR运行过程中有更高优先级的任务被激活进入就绪态。而中断返回前ISR调用OSIntExt()函数,该函数查找就绪表发现有必要进行任务切换,从而被中断的任务进入等待状态,运行被激活的高优先级的任务。任务切换任务切换有两种:OS_TASK_SW()和OSIntCtxSw()。任务级的任务切换OS_TASK_SW()是宏调用,通过软中断指令来实现CPU寄存器内容切换。例如:#define OS_TASK_SW() asm(“int #32”),具体实现参见移植文档。任务级的任务切换过程:1) 保存当前运行的任务的CPU寄存器值到该任务的堆栈。如:堆栈指针,程序计数器,状态寄存器等。2) 将要运行的高优先级的任务的寄存器值从堆栈恢复到CPU寄存器。3) 进行TCB的切换,并运行任务。中断级的任务切换OSIntCtxSw()是在OSIntExt()中调用的,我们一般在用户ISR中调用OSIntExt()以实现中断返回前的任务调度。由于ISR已经将CPU寄存器的值存入被中断的任务的堆栈中,所以OSIntCtxSw()的实现和OS_TASK_SW()不一样,具体参见移植文档。就绪表每个就绪的任务都放在就绪表中,就绪表有两个变量:OSRdyGrp和OSRdyTbl。OSRdyGrp中,将任务按优先级分组,八个为一组。OSRdyGrp的每一位代表每组任务是否有进入就绪态的任务。在就绪表中查找优先级最高的任务不需要扫描整个OSRdyTbl,只要查优先级判定表OSUnMapTbl。OSUnMapTbl是常量表,所以查找优先级最高的任务的执行时间为常量,和就绪表的任务数无关。5、中断服务在用户的ISR中可以调用OSIntEnter()和OSIntExit()通知uC/OS-II发生了中断,这样可以实现ISR返回前的任务调度。uC/OS-II中的中断服务子程序示例:用户中断服务子程序: 保存CPU寄存器; 调用OSIntEnter(); if( OSIntNesting = 1) OSTCBCur - OSTCBStkPtr = SP; 清中断源; 重新开中断;执行用户ISR代码;调用OSIntExit();恢复CPU寄存器;执行中断返回指令;6、时钟节拍uC/OS-II要求用户提供一个周期性的时钟源,来实现时间的延迟和超时功能,时钟节拍应该每秒发生10100次/秒。时钟节拍率越高,系统的额外负荷就越重。应该在多任务系统启动后,也就是调用OSStart()后再开启时钟节拍器。系统设计者可以在第1个开始运行的任务中调用时钟节拍启动函数。假设用定时器TA0作为时钟中断源,那么,在移植过程中实现了函数init_timer_ta0(),此函数用来初始化定时器TA0,并将其打开。uC/OS-II中的时钟节拍服务是在ISR中调用OSTimeTick()实现的。OSTimeTick()跟踪所有任务的定时器以及超时时限。7、uC/OS-II的初始化和启动调用uC/OS-II的服务之前要先调用系统初始化函数OSInit()。OSInit()初始化uC/OS-II所有的变量和数据结构,并建立空闲任务。uC/OS-II初始化任务控制块、事件控制块、消息队列缓冲、标志控制块等数据结构的空缓冲区。多任务的启动是通过调用OSStart()实现的。启动之前要至少创建一个任务。OSStart()调用就绪任务启动函数OSStartHighRdy(),其功能是将任务栈的值恢复到CPU寄存器,并执行中断返回指令,强制执行该任务代码。l 任务管理 uC/OS-II可以管理最多64个任务。任务管理包括创建任务、删除任务、改变任务的优先级及挂起和恢复任务等。1、建立任务,OSTaskCreate ( )、OSTaskCreateExt ( )OSTaskCreate ( )需要四个参数:void (*task)(void *pd),void *pdata,OS_STK *ptos,INT8U prio。task是指向任务函数的指针;pdata是任务开始执行时,传递给任务的参数指针;ptos是分配给任务的堆栈的栈顶指针;prio是分配给任务的优先级。OSTaskCreateExt ( )是OSTaskCreate ( )的扩展,需要的参数比OSTaskCreate ( )更多,共九个,前4个和OSTaskCreate ( )的参数一样。其余的参数是为了系统进行堆栈检验和任务统计等扩展服务功能提供的参数。2、任务堆栈 每个任务都有自己的堆栈空间,为OS_STK类型,并且由连续的内存空间组成。可以静态分配堆栈空间,也可以动态分配。uC/OS-II支持的处理器的堆栈既可以是递减的,也可以是递增的。在创建任务时必须知道堆栈是递减还是递增的,因为必须把堆栈的栈顶传递给OSTaskCreate ( )和OSTaskCreateExt ( )。可以在文件OS_CPU.H中的OS_STK_GROWTH进行堆栈方向的设置。3、删除任务,OSTaskDel()删除任务是指任务处于休眠状态,不再被RTOS调用。4、请求删除任务,OSTaskDelReq()为了避免删除任务时,任务占用的资源因为没有被释放而丢失,可以调用OSTaskDelReq(),让拥有这些资源的任务使用完资源后,先释放资源,再删除自己。5、改变任务优先级,OSTaskChangePrio()调用OSTaskChangePrio(INT8U oldprio, INT8U newprio) 可以动态改变任务的优先级。6、挂起任务,OSTaskSuspend()调用OSTaskSuspend(INT8U prio)可以挂起一个任务,被挂起的任务只有通过调用OSTaskResume()函数来恢复。7、恢复任务,OSTaskResume()恢复因为调用OSTaskSuspend(INT8U prio)挂起的任务。三、uC/OS-II任务间通信方式1、 信号量信号量由两部分组成:一部分是16位的无符号整型信号量的计数值;另一部分是由等待该信号量的任务组成的等待任务表;信号量用于对共享资源的访问,用钥匙符号,符号旁数字代表可用资源数,对于二值信号量该值为1;信号量还可用于表示某事件的发生,用旗帜符号表示,符号旁数字代表事件已经发生的次数;提供服务:OSSemCreate(),建立一个信号量,对信号量赋予初始计数值。如信号量是用于表示一个或多个事件的发生的,其初始值通常为0;如信号量用于对共享资源的访问,则该值赋为1;如信号量用于表示允许任务访问n个相同的资源,则该值赋为n,并把该信号量作为一个可计数的信号量使用;OSSemDel(),删除一个信号量,在删除信号量前必须首先删除操作该信号量的所有任务;OSSemPend(),等待一个信号量;OSSemPost(),发出一个信号量;OSSemAccept(),无等待地请求一个信号量;OSSemQuery(),查询一个信号量的当前状态;不推荐任务和中断服务子程序共享信号量,因为信号量一般用于任务级。如果确实要在任务和中断服务子程序中传递信号量,则中断服务子程序只能发送信号量;OSSemDel()和OSSemPend()服务不能被中断服务子程序调用;2、 互斥型信号量(mutex)互斥型信号量用于处理共享资源;由于终端硬件平台的某些实现特性,例如单片机管脚的复用,多个任务需要对硬件资源进行独占式访问。所谓独占式访问,指在任意时刻只能有一个任务访问和控制某个资源,而且必须等到该任务访问完成后释放该资源,其他任务才能对此资源进行访问。操作系统进行任务切换时,可能被切换的低优先级任务正在对某个共享资源进行独占式访问,而任务切换后运行的高优先级任务需要使用此共享资源,此时会出现优先级反转的问题。即,高优先级的任务需要等待低优先级的任务继续运行直到释放该共享资源,高优先级的任务才可以获得共享资源继续运行。可以在应用程序中利用互斥型信号量(mutex)解决优先级反转问题。互斥型信号量是二值信号量。由于uC/OS-II 不支持多任务处于同一优先级,可以把占有mutex的低优先级任务的优先级提高到略高于等待mutex的高优先级任务的优先级。等到低优先级任务使用完共享资源后,调用OSMutexPost() ,将低优先级任务的优先级恢复到原来的水平。互斥型信号量(mutex)的操作:建立一个互斥型信号量OSMutexCreate (INT8U prio, INT8U *err)等待一个互斥型信号量(挂起)OSMutexPend (OS_EVENT * pevent, INT16U timeout, INT8U * err )释放一个互斥型信号量 OSMutexPost (OS_EVENT * pevent)优先级反转问题。优先级反转问题发生于高优先级的任务需要使用某共享资源,而该资源已被一个低优先级的任务占用的情况。为了降解优先级反转,内核可以将低优先级任务的优先级提升到高于高优先级任务的优先级,直到低优先级的任务使用完占用的共享资源。优先级继承优先级PIP,略高于最高优先级任务的优先级;例:OSMutexCreate(9,&err);建立互斥型信号量,9为PIP;其它服务:OSMutexDel(),删除互斥型信号量,在删除信号量之前应先删除可能用到该信号量的所有任务;OSMutexPend(),等待一个互斥型信号量(挂起),定义超时值为0时则无限期等待;OSMutexPost(),释放一个互斥型信号量;OSMutexAccept(),无等待地获取互斥型信号量(不挂起);OSMutexQuery(),获取互斥型信号量的当前状态;所有服务只能用于任务与任务之间,不能用于任务与中断服务子程序之间;3、 事件标志组( event flag)事件标志组由2部分组成:一是保存各事件状态的标志位;二是等待这些标志位置位或清除的任务列表。可以用8位、16位或32位的序列表示事件标志组,每一位表示一个事件的发生。要使系统支持事件标志组功能,需要在OS_CFG.H 文件中打开OS_FLAG_EN选项。应用场合: 如果一个任务需要等待多个事件的同时发生或者多个事件的中的某个事件发生才能转为就绪态,就可以考虑事件标志组进行任务的同步通信。操作集合:OSFlagCreate(),建立一个事件标志组;事件标志组数据结构包括:指针类型,等待事件标志组的任务列表,以及表明当前事件标志状态的位;OSFlagDel(),删除一个事件标志组,在删除事件标志组前必须首先删除操作该事件标志组的所有任务;OSFlagPend(),等待事件标志组的事件标志位;OSFlagPost(),置位或清0事件标志组中的事件标志;OSFlagAccept(),无等待地获得事件标志组中的事件标志,可以被ISR调用;OSFlagQuery(),查询事件标志组的状态;其中OSFlagCreate()、OSFlagDel()、OSFlagPend()只能被任务调用,不能被中断调用;其它服务任务和中断都可调用;4、消息邮箱一种通信机制,可以使一个任务或者中断服务子程序向另一个任务发送一个指针型的变量,通常该指针指向一个包含了消息的特定数据结构。提供的服务:OS_EVENT * OSMboxCreate(void* msg),建立一个邮箱,并对邮箱须定义指针的初始值。一般情况下该值为NULL,但也可以初始化一个邮箱,使其在最开始就包含一条消息。如使用邮箱的目的是通知一个事件的发生(发送一个消息),则初始化该邮箱为空,即null,因为在开始时很有可能事件没有发生;如用邮箱共享某些资源,则要初始化该邮箱为一个非空的值,此时邮箱被当做一个二值信号量使用;OSMboxDel(),删除一个邮箱,在删除邮箱前必须首先删除操作该邮箱的所有任务;OSMboxPend(),等待邮箱中的消息;OSMboxPost(),向邮箱发送一则消息;OSMboxPostOpt(),向邮箱发送一则消息,增强功能,可以广播;OSMboxAccept(),无等待地从邮箱得到一则消息;OSMboxQuery(),查询一个邮箱的状态;其中:OSMboxDel()、OSMboxPend()、OSMboxQuery()只有任务可以调用,中断不能调用;其它服务两者均可调用;邮箱的使用:可使用的最大邮箱数由配置文件决定;用邮箱作为二值信号量,可在其与信号量之间选择其一;用邮箱实现延时,可取代OSTimeDly()服务;5、消息队列另一种通信机制,允许一个任务或者中断服务子程序向另一个任务发送以指针方式定义的变量或其它任务,因具体应用不同,每个指针指向的包含了消息的数据结构的变量类型也有所不同。提供服务:OSQCreate(),建立一个消息队列,并给它赋给两个参数:指向消息数组的指针和数组的大小。OSQDel(),删除一个消息队列,在删除一个消息队列前必须首先删除所有可能用到这个消息队列的任务;OSQPend(),等待消息队列中的消息,返回消息指针;OSQPost(),向消息队列发送一则消息(基于FIFO,先进先出);OSQPostFront(),向消息队列发送一则消息(基于LIFO,后进先出);OSQPostOpt(),向消息队列发送一则消息(FIFO或LIFO);OSQAccept(),无等待地从消息队列获得消息;OSQFlush(),清空消息队列,清空队列中的所有消息以重新使用;OSQQuery(),查询消息队列的状态;其中:OSQDel()、OSQPend()、OSQQuery()只有任务可以调用,中断不能调用;其它服务两者均可调用;消息队列的使用:使用消息队列读取模拟量的值;消息队列用作计数型信号量;使用消息队列的方式消耗的是消息队列指针指向的数据类型的变量,原来可以用信号量来管理的每个共享资源都需要占用一个队列控制块,因此消息队列与信号量相比,节省了代码空间,而牺牲了RAM空间;此外,当用计数型信号量管理的共享资源很多时,消息队列的方式效率非常低;四、多任务系统中驱动程序的实现方法系统在运行多任务时,由于任务切换的存在,驱动程序库的实现与以前基于前后台系统的实现方法有区别。因为一个硬件驱动在对硬件接口进行操作时可能会发生任务切换,正在执行的硬件接口操作会被打断,而任务切换后运行的任务可能对同一个硬件接口进行操作,从而破坏任务切换前的接口操作状态。为了防止出现这种情况,驱动程序必须考虑到对硬件操作的原子性,即一个底层的操作序列完成前不能被另一个操作打断。例如,由于xRam读写和LCD显示数据线的复用,向LCD显示一个字节数据的操作(lcd_draw_byte)不能被xRam读写操作wr_abyte和rd_abyte打断。同样,在向外部RAM读写一个字节数据时也不能被LCD显示操作打断。另外,两个调用xRam读写操作的任务在进行xRam读写字节操作时也不能相互打断。在uC/OS-II中,保护一个原子操作的手段有:关闭中断、禁止任务切换和利用互斥信号量。例如,在函数void lcd_draw_byte(int ix,int iy,char data)中使用互斥信号量保证此函数执行完成才能进行任务切换。例:在xRam读写函数中使用互斥信号量/*功能:向外部RAM写一个字节,入口参数: addr 外部RAM地址, data写入的字节*/void wr_abyte(UBYTE data,addr_t addr)INT8U err;OSMutexPend(xRamMutex,0,&err); /等待一个互斥信号量写W24100 xram过程OSMutexPost(xRamMutex,);/释放一个互斥信号量多任务系统中,不仅硬件复用的驱动程序之间需要互斥信号量保证操作的正确,而且相同的驱动被不同的任务调用时也会出现独占访问的问题。除了LCD显示和xRam,系统中其他需要互斥信号量保护的驱动程序,包括EEPROM读写子程序(模拟IIC总线)、dataflash读写操作驱动(模拟spi总线)。对于uart通讯驱动和rtc4553时钟芯片读写驱动可以考虑在一个任务中调用,因此不存在互斥访问的问题。无论是不同驱动程序之间的硬件资源共享问题,还是同一个驱动被不同任务调用时的共享问题,都可以采用隐含的信号量才实现,即将信号量放在驱动内部。任务在与某一资源打交道的时候并不知道相应的驱动内有一个互斥信号量。任务在调用一个驱动函数时,实际在申请得到一个信号量。如下图所示:IIC总线WrbytII()图5、用互斥信号量实现一个EEPROM驱动任务1任务2WrbytII()EEPROM驱动子程序表示互斥信号量将互斥信号量隐含在驱动内五、键盘驱动和LCD显示的实现LCD显示驱动作为功能子程序实现。LCD显示驱动程序必须考虑显示驱动对硬件资源进行操作时有可能进行任务切换,进行xRam读写而出现硬件访问错误。可以考虑利用互斥信号量将LCD显示驱动和xRam驱动程序实现为可重入函数。键盘处理程序作为任务来实现,任务通常是一个无限的循环,相当于一个线程,当一个任务运行时,该任务单独占用CPU资源,直到被其他高优先级的任务剥夺运行权利而转为就绪态任务。键盘用两个任务来实现,一个是键盘扫描任务KeyboardScanTask,定时对按键状态进行扫描,并将健值放到缓冲中;另一个是键盘处理任务KeyboardProcTask,定期从键盘缓冲获取健值,并进行相应处理。键盘处理任的执行周期比扫描任务长,但是比扫描任务优先级高。键盘扫描任务是定时任务。以前的前后台系统实现时是将键盘扫描驱动keyProcess放到20ms定时器中断服务队列中。20ms定时器中断发生时,进行键盘处理keyProcess。KeyProcess调用scanfkey函数,scanfkey函数实现键盘按键状态的扫描,并返回当前按键值。KeyProcess同时处理键盘抖动的因素,采取的方法是连续两次20ms定时器任务时判断按键的值是否一样,如果是,则说明操作员在进行按键操作。所以设置定时器的间隔时间不能太长,否则键盘需要按下很长时间才能对其响应,键盘的响应就会很慢。将键盘keyProcess加入20ms定时器中断任务队列中的方法如下:void init_keyboard(void) insert_task(keyProcess);现在,在多任务调度系统中为了提高系统的实时性,应该尽量减少中断服务子程序的操作内容,借鉴以前系统将键盘驱动加入20ms定时器中断任务队列的实现,我们可以采取两种方法来实现键盘扫描任务。一个是采用操作系统的任务延迟,一个是利用定时中断服务子程序发送信号量。第一种方法的实现比较简单,也比较节省内存资源。但是由于操作系统的调度等原因,延迟时间不够精确。第二种方法实现一个20ms的定时中断服务子程序,在中断服务子程序中发出一个信号量(post)给键盘扫描任务。如果键盘扫描任务处于就绪状态,在得到信号量时,键盘驱动就继续运行,运行结束后挂起,继续等待下一次信号量。这种方法延时比较精确,但实现复杂些,消耗内存多些。键盘处理任务的实现相对简单,就是定时的从键盘缓冲读取按键值。如果有按键值,就根据相应的值进行逻辑处理。键盘处理任务比键盘扫描任务的优先级高。原系统中的20ms定时任务有:1、 键盘任务 insert_task(keyProcess); 2、 LED显示 insert_task(lmp_proc);3、 延时和计数任务 unsigned char insert_timer(int *cnt)系统中的5ms定时任务有:1. 电表通讯任务 insert_ctask(cch_mt_ctask);2. 电台通信通讯任务insert_ctask(cch_rd_ctask);六、交流采集中断和计算任务的通信交流采样模块采用中断驱动模式,分别用定时器TA2作为采样定时器,定时器TB0、TB1、TB2为三相测频定时器,定时器TA2工作于定时方式(用于采样),定时器TB0、TB1、TB2工作于脉冲周期测量方式(用于A、B、C三相工频测频)。采样周期是每16个电网频率周期,采集一个周期的数据,即隔15个周期采一个周期。采集频率是一个周期内采64个点。由于电网频率是在一定范围内波动的,所以采样模块对每个工频进行频率测量,第16个周期(采样周期)的工频是根据前16个周期的工频进行的估算,以便确定出将要进行采样的时间间隔。估算公式如下:number_next = (nub_acc/16+ number)/2;number:表示当前工频周期f8计数器的计数值。nub_acc:表示前16个工频周期的f8计数器累计值。number_next:估算的将要进行采样的工频周期的频率。交采表的精度也是整个测试项目主要考虑的问题。为了满足表计精度要求,交采表部分的交流采样要求实时性强,响应快,所以应该用中断服务子程序(ISR)来实现,并且将中断优先级设置为最高(7级)。交采表计算部分作为一个任务来实现,优先级主要实现电流电压和电网频率的信号采集,然后根据三相电表的计算公式进行有效电流、电压、功率等数据相的计算。整个系统中,除了交流采样的实时性要求最高外,为了能够达到1级电能表的要求,交采表的电表数据计算任务的优先级高于系统其他任务的优先级。因此设置交采计算任务的优先级为0级(最高级)。这样可以满足15个工频周期内将交流采样的64个电流、电压值计算完成。交流采样中断服务子程序(ISR)与电表数据计算任务之间的同步采用信号量的通信方法,交流采样ISR完成以后发出一个信号给等待的数据计算任务,计算任务完成以后挂起,等待下一个交流采集信号量。交流采集ISR计算任务发出 POST接受 PEND图6、用信号量使任务与中断服务同步表示信号量表示消息流向 七、多任务系统中任务划分原则和任务优先级设定l 任务划分的原则1、I/O 依赖性:如果变换依赖于I/O,那么它运行的速度常常受限于与它互操作的I/O设备的速度,这种情况下变换应成为一个独立的任务;在系统中创建多个与I/O设备相当数目的I/O任务;I/O任务只实现与设备相关的代码;I/O任务的执行只受限于I/O设备的速度,而不是处理器;在任务中分离设备相关性;如键盘显示任务、数据存储任务等;2、时间关键性的功能:将有时间关键性(deadline)的功能分离出来,组成独立运行的任务;赋予这些任务高的优先级,以满足对时间的需要;如交采任务;3、计算功能;计算功能占用CPU的时间多;捆绑计算功能成任务,赋予它们较低优先级运行,能被高优先级的任务抢占,消耗CPU的剩余时间;保持高优先级的任务是轻量级的(负荷较轻);如统计计算任务;4、功能内聚:各紧密相关的功能,不能分别对应不同的任务;将这些紧密相关的功能组,组成一个任务,使各功能共享资源或相同事件的驱动;组成一个任务会减小通信的开销,而且

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论