资源描述
内容介绍
1、 介绍uC/OS-II嵌入式操作系统
2、 基于uC/OS-II的用电管理终端软件的设计
书籍:《嵌入式实时操作系统uC/OS-II》
作者:Jean Labrosse
uC/OS-II V2.52 通过了美国航空航天管理局(FAA)的安全认证;
安全性、可靠性是得到认证的。
我们为什么会选择uC/OS-II嵌入式操作系统?
1、 与终端硬件平台相适应
全部源代码5500行,可裁减定制,生成的可执行代码占15~20k,
可以移植到多种系列单片机上,包括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/OS-II以及绝大多数商业实时内核都是可剥夺型内核。最高优先级的任务一旦就绪,就抢占运行着的低优先级的任务,得到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.C
OS_FLAG.C OS_TIME.C
OS_MBOX.C uCOS-II.C
OS_MEM.C uCOS-II.H
OS_MUTEX.C OS_SEM.C
OS_Q.C
uC/OS-II 配置文件
(与应用程序有关)
OS_CFG.H
INCLUDES.H
移植uC/OS-II
(与处理器类型有关的代码)
OS_CPU.H OS_CPU_C.C
OS_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_TaskStk[TASK_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 ……. ISRn
main1()
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()
内存管理
任务堆栈
定时器
CPU
RAM
任务间通信
任务管理
ISR
ISR
任务创建
任务控制块
任务删除
信号量
互斥信号量
事件标志组
事件控制块
消息邮箱
消息队列
串行I/O
ISR
时间管理
时钟节拍
挂起任务
恢复任务
uC/OS-II启动OSStart
任务延时
任务延时恢复
ROM
uC/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()完成;中断级的任务调度由OSIntExit()完成。这两种任务调度情况调用的任务切换函数不同:任务级的任务调度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寄存器;
执行中断返回指令;OSIntExit()与中断返回指令什么关系?
6、时钟节拍
uC/OS-II要求用户提供一个周期性的时钟源,来实现时间的延迟和超时功能,时钟节拍应该每秒发生10~100次/秒。时钟节拍率越高,系统的额外负荷就越重。
应该在多任务系统启动后,也就是调用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因为优先级是唯一的,故可以通过oldprio来唯一标识一个任务……
) 可以动态改变任务的优先级。
6、挂起任务,OSTaskSuspend()
调用OSTaskSuspend(INT8U prio)因为优先级是唯一的,故可以通过oldprio来唯一标识一个任务……
可以挂起一个任务,被挂起的任务只有通过调用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
任务2
W
展开阅读全文