当前位置:首页 > 汽车电子1 > 糖果Autosar
[导读]上一节简单记录了进程task。有了进程以后,我们需要关心怎么样分配CPU资源(或者运行时间)给每个进程。那么就要引入排程(scheduling)的概念。排程一般都是OS里面非常重要的一部分,但是在深入进入排程和理解RTX排程器(scheduler)如何运作之前,不妨看看RTX提供...

上一节简单记录了进程task。有了进程以后,我们需要关心怎么样分配CPU资源(或者运行时间)给每个进程。那么就要引入排程(scheduling)的概念。排程一般都是OS里面非常重要的一部分,但是在深入进入排程和理解RTX排程器(scheduler)如何运作之前,不妨看看RTX提供的许多简单易容的时间管理相关的操作,这些操作虽然也涉及排程器的运作,但是不需要我们对排程器和相关算法有深刻的理解。


1.配置前提

  • RTX配置为不使用Round-Robin(轮转式)排程(在RTX_Conf_CM.c中 取消勾选Roudn-Robin Task Switching)


  • 创建的进程,优先度全部一样(为0)

具体的原因,具体介绍完排程器后就会一目了然。

我们可以考虑上一节笔记给的例子:


  1. OS_TID taskID1;

  2. OS_TID taskID2;


  3. __task void init (void) {

  4. //Necessary Initialization

  5. //...

  6. //Create a task

  7. taskID1 = os_tsk_create(task1, 0);

  8. taskID2 = os_tsk_create(task2, 0);

  9. os_tsk_delete_self (); // Delete the init(self) task

  10. }


  11. int main(void)

  12. {

  13. //Necessary Initialization

  14. //...

  15. os_sys_init(init);


  16. }


运行后,会有两个同等优先级的task,task1和task2。



2.简单的时间管理操作


os_time_get(void);

首先是这个操作,返回一个U32数,为当前操作系统运行的时间,以Timer Ticks Value为单位(见上面RTX配置图),预设是10ms。所以如果返回0x000000C4,那么OS走了1960ms,也就是1.96s。


然后就是三个主动放弃当前对CPU占用的操作。这也是为什么我称之为简单的时间管理操作,因为这看起来并不是排程器要求当前进程放弃其对CPU的占用,而是他们“自愿”放弃的。也就是说,这三个操作,只能在当前进程中使用,而且其目标对象就是当前进程本身。效果都是把他们从运行的状态改变到其他状态。


如果我们看上面的程序,我们会发现,其实如果没有相应的事件管理的话,task1其实是会一直运行直到结束。那么如果task1在某一时刻,执行以下任一操作:

os_tsk_pass();

进程状态从RUNNING(运行)进入READY(就绪),加入一个先进先出的队列。排程器此时会选择下一个队列中已经READY(就绪)的进程去执行,在这里,也就是task2。那么如果task2运行一段时间后也执行了相同操作,那么它就会把运行机会重新交回给task1。


os_dly_wait(delay_time);

进程状态从RUNNING(运行)进入WAIT_DLY(等待延迟)。排程器此时会选择下一个队列中已经READY(就绪)的进程去执行,在这里,也就是task2。和os_tsk_pass()不同的是,进程并不直接进入就绪等待队列,而是等delay_time×Timer Ticks Value之后才重新加入这个先进先出的队列。例如填入5,那么预设情况下,task1就会暂停,等待50ms后,重新加入就绪等待队列。


os_itv_set(interval_time); 和 os_itv_wait(void);

这个得先在进程入口设置周期时间,interval_time,然后在进程中执行该操作的话,进程状态从RUNNING(运行)进入WAIT_ITV(等待周期)。排程器此时会选择下一个队列中已经READY(就绪)的进程去执行,在这里,也就是task2。和os_tsk_pass()不同的是,进程并不直接进入就绪等待队列,而是等interval_time×Timer Ticks Value之后才重新加入这个先进先出的队列。例如填入5,那么预设情况下,task1就会暂停,等待50ms后,重新加入就绪等待队列。但是与os_delay_wait()不同的是,如果在等待周期过程中,没有别的task在占用CPU,这个在等待WAIT_ITV的task是可以进入RUNNING状态的。这个很明显是为有周期性的进程而设的。


这三个介绍完,就到一个定时调用,执行如下操作:


os_tmr_create(tcnt,para)

这个操作,会在tcnt×Timer Ticks Value时间后,会调用os_tmr_call(para);,para是这个调用传递的参数。这个并不是一个进程,它不改变当前所有进程的状态,而是直接进入该函数,执行相关内容。你可以选择在RTX_Config.c中找到它的原型。一般不会把大段代码放在里面,而且它必须要能够自己结束!(而不是像一般进程一样,无限循环。)你可以理解它为一个闹钟,提醒OS做特定简短的任务。


另外在RTX_Config.c中,还有一个类似的原型,不过这次是一个进程,void os_idle_demon(void); 如果当前没有进程运行或处在就绪状态(都在等待状态),那么RTX就会运行这个进程,预设这个进程只是空转,不干任何实际的事情。


3.小结


这里介绍的一些操作,可以单独运用,但更多情况是和后面要介绍到的排程器的具体运作相关。后面有机会再记录。


本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
关闭