xenomai 应用开发 2:单触发模式与周期性模式运行多任务--WT

一、多任务模式

当下的实时系统基于多任务和任务间通信的互补概念。多任务环境允许将实时应用程序构建为一组独立的任务,每个任务都有单独的执行线程和自己的一组系统资源。任务间通信设施允许这些任务同步和协调它们的活动。Xenomai多任务调度程序使用中断驱动的、基于优先级的抢占式任务调度。它具有快速的上下文切换和低中断延迟。
多任务处理造成了许多并发执行的任务,而实际上,内核根据调度算法交错执行这些任务。每个任务都有自己的上下文,它是任务每次计划由内核运行时看到的CPU环境和系统资源。在上下文切换时,任务的上下文保存在任务控制块(TCB)中。任务的上下文包括:

一个执行的线程,即任务的程序计数器
CPU寄存器和浮点寄存器(如果需要)
一组动态变量和函数调用的返回地址
I/O分配标准输入,输出,错误
延迟计时器
一个时间片计时器
内核控制结构
信号处理程序
调试和性能监视值

在Xenomai中,时间是由系统时钟给出的。
系统时钟可以通过函数rt_timer_read()读取,例如:
RTIME now; now = rt_timer_read();

二、单触发(one-shot)模式创建运行多任务

单触发模式如上篇文章xenomai 应用开发 1模式
1、单触发模式启用5个任务,并且为五个任务分别命名:
#define NTASKS 5
RT_TASK task[NTASKS];
可按这种方式进行创建,并给予相同的优先级。 程序与结果如下所示:

#include <stdio.h>
#include <signal.h>
#include <unistd.h>

#include <alchemy/task.h>

#define NTASKS 5
RT_TASK task[NTASKS];
//RT_TASK hello_task;


// function to be executed by task
void demo(void *arg) {
    RT_TASK_INFO curtaskinfo;
    rt_task_inquire(NULL,&curtaskinfo);
    rt_printf("Task name: %s \n", curtaskinfo.name);
}


int main(int argc, char* argv[])
{
   int i;
     char  str[20] ;
     char  name[]={'A','B','C','D','D'};//为5个任务分别命名
     
      
     for(i=0; i < NTASKS; i++) {
       printf("start task  : %d\n",i);
       sprintf(str,"task%c",name[i]);
       rt_task_create(&task[i], str, 0, 50, 0);
       rt_task_start(&task[i], &demo, 0);} 
}

在这里插入图片描述
如图示:第一个实时任务序号为0,名字叫 taskA,第二个为taskB,以此类推。
2、第一个程序中有个参数arg,它是指向void类型的指针。在调用rt_task_start()期间,可以将整数参数作为对任务的引用传递(arg是给任务函数的void类型指针参数)。例如,对于一个变量int index,使用
rt_task_start(&taskd…, &demo, &index)
修改第一个程序,使五个任务中的每一个都有唯一的参数。让任务通过强制类型转换和取消引用该编号来打印该编号,例如,
int num = * (int *)arg;
同样,每个任务具有相同的优先级。执行程序并描述其输出。

#include <stdio.h>
#include <signal.h>
#include <unistd.h>

#include <alchemy/task.h>

#define NTASKS 5
RT_TASK task[NTASKS];
//RT_TASK hello_task;


// function to be executed by task
void demo(void *arg) {
    int num = * (int *)arg;
    rt_printf("The number is %d \n", num);
    RT_TASK_INFO curtaskinfo;
    rt_task_inquire(NULL,&curtaskinfo);
    rt_printf("Task name: %s \n", curtaskinfo.name);
}


int main(int argc, char* argv[])
{
   int i;
     char  str[20] ;
     char  name[]={'A','B','C','D','E'};
     
      
     for(i=0; i < NTASKS; i++) {
       printf("start task  : %c\n",name[i]);
       sprintf(str,"task%d",i);
       rt_task_create(&task[i], str, 0, 50, 0);
       rt_task_start(&task[i], &demo, &i);}
}

在这里插入图片描述3、以one-shot模式创建不同优先级的任务,观察终端打印结果。
修改1例子中的程序,赋予五个任务以不同的优先级:

#include <stdio.h>
#include <signal.h>
#include <unistd.h>

#include <alchemy/task.h>

#define NTASKS 5
RT_TASK task[NTASKS];
//RT_TASK hello_task;


// function to be executed by task
void demo(void *arg) {
 
    RT_TASK_INFO curtaskinfo;
    rt_task_inquire(NULL,&curtaskinfo);
    rt_printf("Task name: %s \n", curtaskinfo.name);
    rt_printf("Task priority is %d \n" ,curtaskinfo.prio);
}


int main(int argc, char* argv[])
{
   int i;
     char  str[20] ;
     char  name[]={'A','B','C','D','D'};
     int   priority[]={50,51,52,53,54};
      
     for(i=0; i < NTASKS; i++) 
	{
       printf("start task  : %c\n",name[i]);
       sprintf(str,"task%d",i);
       rt_task_create(&task[i], str, 0, priority[i], 0);
       rt_task_start(&task[i], &demo, 0);
	}
}

在这里插入图片描述设置了优先级过后,第一个任务获得最低优先级,最后一个任务获得最大的优先级,但是打印每个任务的顺序并没变,因为根据for循环,几个任务并不是同时发生的,当第一个任务创建并且执行完自己的function时,第二个任务才会被创建,受到任务调度的影响。

三、以周期性模式运行多任务

实现多任务运行,并且占用较少的CPU资源,xenomai 提供了一些常用的API以供实时任务使用:

1、设定实时任务周期

int rt_task_set_periodic(RT_TASK *task, RTIME start_time,RTIME period);

通过在处理器时间线中编程任务的第一个发布点及其周期,使任务具有周期性。然后,任务应该调rt_task_wait_period()来休眠,直到到达处理器时间线中的下一个定期释放点。
*task 任务描述符,如果任务为空,则将当前任务设为定期任务。
start_time 第一个释放点的初始(绝对)日期,以时钟节拍表示。如果start_time等于TM_NOW,则使用当前系统日期。
period 任务的周期。如果已启用一个周期计时器任务,可以通过传递 TM_INFINITE给period 以停止周期性任务。
2、当周期性任务完成其功能时,它释放处理器并开始使用以下命令等待下一个周期的到来:
void rt_task_wait_period(NULL);
3、程序及结果:周期性运行3个实时任务,分别以一秒,两秒,三秒为周期进行运行任务。

#include <stdio.h>
#include <signal.h>
#include <unistd.h>
#include <alchemy/task.h>
#include <sys/mman.h>
#include <alchemy/timer.h>
#include <math.h>

#define NTASKS 3
RT_TASK task[NTASKS];
// function to be executed by task

void task_function(void *arg)
{   RTIME now; 
    int i=1;
    
   int num = * (int *)arg;
   float period[]={1e9,2e9,3e9};
   //rt_printf("The arg is %d \n", num);
   
    rt_task_set_periodic(NULL, TM_INFINITE, period[num]);
   
   //开始任务循环
  while (1) {
    /*
      Do your thing here
     */
    
    RT_TASK_INFO curtaskinfo;
    rt_task_inquire(NULL,&curtaskinfo);
    rt_printf(" 第 %d 次循环 %s 任务 -- 循环时间: %.5f s\n",i,curtaskinfo.name,(rt_timer_read()-now)/1000000000.0);
    i=i+1;
   now = rt_timer_read();
   rt_task_wait_period(NULL);
   
  }
  return;
}

int main(int argc, char* argv[])
{
     int i;
     char  str[20] ;
     char  name[]={'A','B','C','D','E'};
     mlockall(MCL_CURRENT|MCL_FUTURE);//锁定内存以避免此程序的内存交换
      
     for(i=0; i < NTASKS; i++) 
      {
       //printf("start task  : %c\n",name[i]);
       sprintf(str,"task%d",i);
       rt_task_create(&task[i], str, 0, 50, 0);
       rt_task_start(&task[i], &task_function, &i);
       }
       
       pause();
}
1 次循环 task0 任务 -- 循环时间: 24339.75194 s
 第 1 次循环 task1 任务 -- 循环时间: 24313.98238 s
 第 1 次循环 task2 任务 -- 循环时间: 24288.21280 s
 第 2 次循环 task0 任务 -- 循环时间: 0.99999 s
 第 3 次循环 task0 任务 -- 循环时间: 0.99998 s
 第 2 次循环 task1 任务 -- 循环时间: 1.99999 s
 第 4 次循环 task0 任务 -- 循环时间: 0.99998 s
 第 2 次循环 task2 任务 -- 循环时间: 3.00000 s
 第 5 次循环 task0 任务 -- 循环时间: 0.99998 s
 第 3 次循环 task1 任务 -- 循环时间: 2.00000 s
 第 6 次循环 task0 任务 -- 循环时间: 0.99998 s
 第 7 次循环 task0 任务 -- 循环时间: 0.99998 s
 第 4 次循环 task1 任务 -- 循环时间: 1.99999 s
 第 3 次循环 task2 任务 -- 循环时间: 2.99999 s
 第 8 次循环 task0 任务 -- 循环时间: 0.99998 s
 第 9 次循环 task0 任务 -- 循环时间: 0.99998 s
 第 5 次循环 task1 任务 -- 循环时间: 2.00000 s
 第 10 次循环 task0 任务 -- 循环时间: 0.99998 s
 第 4 次循环 task2 任务 -- 循环时间: 3.00000 s
 第 11 次循环 task0 任务 -- 循环时间: 0.99998 s
 第 6 次循环 task1 任务 -- 循环时间: 1.99999 s
 第 12 次循环 task0 任务 -- 循环时间: 0.99998 s

前面3条打印信息为任务在系统中的起始时间.后面的均为该任务运行的头一次与后一次在系统时钟中的差值,我们从结果可以看出,task0以1秒为周期,task1以2为周期,task2以3秒为周期。但周期有少量的出入,是系统延时所导致的。


版权声明:本文为robot1701原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。