VxWorks下实时多任务程序的实现
扫描二维码
随时随地手机看文章
1 引言
随着信息技术的快速发展和应用,计算机的发展正显示出微型化和专业化的趋势。目前,嵌入式系统已得到广泛应用,实时系统在各高精尖技术领域得到迅猛发展,具有实时操作系统的计算机系统也得到了广泛的应用。实时操作系统是嵌入式应用软件的基础和开发平台,它不同于分时操作系统,它的主要任务是对事件进行实时的处理。虽然事件可能在无法预知的时刻到达,但是软件必须在事件发生时在严格的时限内做出响应。在实时操作系统下设计实时多任务程序是实现实时系统的重要一环,本文介绍了VxWorks操作系统的技术特点,给出了一种在VxWorks操作系统下实时多任务程序的设计方法,经过仿真,验证了这种设计方法的可行性和有效性。
2 VxWorks操作系统简介
Vxworks就是一个由WRS公司推出的嵌入式实时操作系统,具有多任务、可裁减、可靠性好、实时性高等特点, 是业界公认的性能出色的一种实时操作系统。VxWorks实时操作系统由400多个相对独立的、短小精炼的目标模块组成,用户可根据需要选择适当模块来裁剪和配置系统,这有效地保证了系统的安全性和可靠性。系统的链接器可按应用的需要自动链接一些目标模块。这样,通过目标模块之间的按需组合,可得到许多满足功能需求的应用。VxWorks操作系统的基本构成模块主要包括以下部分:高效的实时内核Wind、兼容实时系统标准POSIX、I/O系统、本机文件系统、文件I/O系统、网络特性、虚拟内存(可选单元VxVMI)、共享内存(可选单元VxMP)、驻留目标工具、Wind基类、工具库、性能优化、目标代理、板级支持包、VxWorks仿真器(VxSim)。
VxWorks中与构建实时多任务系统有关的内容如下:
1)wind内核任务调度策略
Wind内核采用基于优先级的抢占式调度法作为它的缺省策略,同时它也提供了轮转调度法。
基于优先级的抢占式调度,具有很多优点。这种调度方法为每个任务指定不同的优先级。没有处于阻塞或悬置态的最高优先级任务将一直运行下去。当更高优先级的任务由就绪态进入运行时,系统内核立即保存当前任务的上下文,切换到更高优先级的任务。而轮转调度法分配给处于就绪态的每个同优先级的任务一个相同的执行时间片。
2)信号量
VxWorks信号量是提供任务间通信、同步和互斥的最优选择,也是提供任务间同步和互斥的主要手段。VxWorks 提供3 种信号量来解决不同的问题。
a. 二进制信号量:二进制信号量(Binary)能够满足任务间的互斥和同步,需要的系统开销最小。当一个任务调用semTake()请求一个信号量时,如果此时信号量可用,信号量会被清零,并且任务立即继续执行;如果信号量不可用,任务会被阻塞来等待信号量。
b. 互斥:互斥信号量(Mutex)用来实现对共享资源的保护,当两个以上的任务共享同一块内存缓冲区或同一个I/O设备之类的资源时,可能会发生竞争状态。互斥信号量可以通过对共享资源上锁,实现高效的互斥访问。
c. 计数器信号量:计数器信号量(Count)是实现任务同步和互斥的另一种手段。计数器信号量除了像二进制信号量那样工作外,还保持对信号量释放次数的跟踪。
3) 中断服务程序
VxWorks的所有中断服务程序使用同一中断堆栈,它在系统启动时就已根据具体的配置参数进行了分配和初始化,因此,必须保证它的大小,以使它能满足最坏的多中断情况。为了快速响应中断,中断服务程序ISR运行在特定的空间,不同于其它任何任务,中断处理没有任务的上下文切换。中断也有缺陷:ISR不运行在常规的任务上下文,它没有任务控制块。对于ISR的基本约束就是它们不能激活那些可能使调用程序阻塞的函数。
3 实时多任务程序设计
在一个实时系统中,中断处理是至关重要的,系统通过中断机制响应外部事件,并对外部事件作出处理,系统对中断的响应速度和中断服务程序的处理速度直接反映了实时系统的性能。VxWorks提供函数intconnect(),它允许将指定的C函数与任何中断相联系。这个指定的C函数就是这个中断的中断服务程序, 中断服务程序必须尽快结束,以免阻塞低优先级的中断,它不同于普通的任务模块,必须满足特殊的要求就是不许阻塞。为了既保证中断服务程序尽快结束而且又避免ISR在调用函数时受到约束,这时可以使用信号量来将中断与任务相联系,用任务来完成IRS的功能,建立实时多任务系统。
具体的做法为,先利用函数taskSpamn()来创建多个任务,建立多任务环境,与中断相联系的任务处于高优先级,利用semTake(semID)使该任务等待信号量而处于阻塞状态,只有当相应的中断到来时,中断服务程序执行semGive(semID)给出信号量,任务得到相应的信号量而执行。任务间通信利用互斥信号量实现内存共享。
下面以一个数据采集的程序实例来说明这种实时多任务程序的设计方法。
对于一个需要定周期采样并且还要等待外部数据准备好后,读入数据并与采样数据融合的系统,我们进行了如下设计,系统划分为三个任务,按优先级高低排列的任务为:
a. 采样与数字滤波(tsample);
b. 读入数据与数据处理(tdataprocess);
c. 系统监控(tmonitor)。
其中,tsample和tdataprocess通过信号量分别与定时中断和外部数据准备好中断相联系,它们之间的通信利用互斥信号量实现内存共享。当没有中断到来时,任务得不到信号量而处于阻塞状态,执行tmonitor;当中断到来时,中断服务程序产生信号量,相应的高优先级任务得到信号量而抢先执行,当高优先级的任务执行完后再执行低优先级任务。程序如下:
1)主程序创建三个任务和两个中断连接。
STATUS main (void)
{…
tidsample=taskSpawn("tsample",200,0,STACK_SIZE,(FUNCPTR)sample,0,0,0,0,0,0,0,0,0,0); tiddataprocess=taskSpawn("tDataprocess",210,0,STACK_SIZE,(FUNCPTR)dataprocess,0,0,0,0,0,0,0,0,0,0);
…}
2) 中断服务程序只产生相应的信号量。
void intSample(void){semGive(sampleSemID);}
void intDataprocess(void){semGive(dataprocessSemID);}
3)高优先级任务等待信号量而处于阻塞状态,要取得相应的中断服务程序发出的信号量才能运行。
void sample (void)
{semTake(sampleSemID,WAIT_FOREVER);
…}
void dataprocess (void)
{semTake(DataprocessSemID,WAIT_FOREVER);
…}
该程序在WxSim下运行,在软件逻辑分析仪WindView下显示的运行结果如图1—4。
图 3 图 4
其中,图3显示的运行过程是,任务tmonitor执行à中断intrupptà中断服务程序给出信号量semGive(sampleSemID)à中断tintà中断服务程序给出信号量semGive(dataprocessSemID)à任务tsample得到信号量semTake(sampleSemID)à任务tsample执行完后退出à任务dateprocess得到信号量semTake(dataprocessSemID)à任务tdataprocess执行完后退出à任务tmonitor继续执行。
图4显示的运行过程是:任务tmonitor执行à中断tintà中断服务程序给出信号量semGive(dataprocessSemID)à任务dataprocess得到信号量semTake(dataprocessSemID)à任务tdataprocess执行à中断intrupptà中断服务程序给出信号量semGive(sampleSemID)à任务tsample得到信号量 semTake(sampleSemID)à任务tsample执行完后退出à任务tdataprocess继续执行,执行完后退出à 任务tmonitor继续执行。
由运行结果可以看出,与中断对应的任务在得到信号量后没有立即执行,当中断服务程序返回中断点后,有几微秒的延迟(Pentium ,100M),这是由于任务切换上下文造成的,这一延迟量对于系统的实时性影响不大,是可以接受的,程序的运行结果达到了预期要求。
4 结束语
本文作者创新点:在VxWorks实时操作系统中,使用信号量将中断与任务相联系,中断服务程序只产生信号量,保证了中断服务程序尽快结束,以免阻塞低优先级的中断;用任务来完成中断服务程序所需要完成的功能,避免了VxWorks对中断服务程序的约束。在软件逻辑分析仪WindView下看到的程序运行结果表明,该方法建立实时多任务系统是可行的。