ImageVerifierCode 换一换
格式:DOCX , 页数:8 ,大小:19.28KB ,
资源ID:25319026      下载积分:3 金币
快捷下载
登录下载
邮箱/手机:
温馨提示:
快捷下载时,用户名和密码都是您填写的邮箱或者手机号,方便查询和重复下载(系统自动生成)。 如填写123,账号就是123,密码也是123。
特别说明:
请自助下载,系统不会自动发送文件的哦; 如果您已付费,想二次下载,请登录后访问:我的下载记录
支付方式: 支付宝    微信支付   
验证码:   换一换

加入VIP,免费下载
 

温馨提示:由于个人手机设置不同,如果发现不能下载,请复制以下地址【https://www.bdocx.com/down/25319026.html】到电脑端继续下载(重复下载不扣费)。

已注册用户请登录:
账号:
密码:
验证码:   换一换
  忘记密码?
三方登录: 微信登录   QQ登录  

下载须知

1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。
2: 试题试卷类文档,如果标题没有明确说明有答案则都视为没有答案,请知晓。
3: 文件的所有权益归上传用户所有。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 本站仅提供交流平台,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

版权提示 | 免责声明

本文(ucosii中文书邵贝贝.docx)为本站会员(b****9)主动上传,冰豆网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知冰豆网(发送邮件至service@bdocx.com或直接QQ联系客服),我们立即给予删除!

ucosii中文书邵贝贝.docx

1、ucosii中文书邵贝贝第5章 时间管理 在3.10节时钟节拍中曾提到,C/OS-(其它内核也一样)要求用户提供定时中断来实现延时与超时控制等功能。这个定时中断叫做时钟节拍,它应该每秒发生10至100次。时钟节拍的实际频率是由用户的应用程序决定的。时钟节拍的频率越高,系统的负荷就越重。 3.10节讨论了时钟的中断服务子程序和节时钟节函数OSTimeTick该函数用于通知C/OS-发生了时钟节拍中断。本章主要讲述五个与时钟节拍有关的系统服务: OSTimeDly() OSTimeDlyHMSM() OSTimeDlyResume() OSTimeGet() OSTimeSet()本章所提到的函数

2、可以在OS_TIME.C文件中找到。5.1 任务延时函数,OSTimeDly()C/OS-提供了这样一个系统服务:申请该服务的任务可以延时一段时间,这段时间的长短是用时钟节拍的数目来确定的。实现这个系统服务的函数叫做OSTimeDly()。调用该函数会使C/OS-进行一次任务调度,并且执行下一个优先级最高的就绪态任务。任务调用OSTimeDly()后,一旦规定的时间期满或者有其它的任务通过调用OSTimeDlyResume()取消了延时,它就会马上进入就绪状态。注意,只有当该任务在所有就绪任务中具有最高的优先级时,它才会立即运行。程序清单 L5.1所示的是任务延时函数OSTimeDly()的代

3、码。用户的应用程序是通过提供延时的时钟节拍数一个1 到65535之间的数,来调用该函数的。如果用户指定0值L5.1(1),则表明用户不想延时任务,函数会立即返回到调用者。非0值会使得任务延时函数OSTimeDly()将当前任务从就绪表中移除L5.1(2)。接着,这个延时节拍数会被保存在当前任务的OS_TCB中L5.1(3),并且通过OSTimeTick()每隔一个时钟节拍就减少一个延时节拍数。最后,既然任务已经不再处于就绪状态,任务调度程序会执行下一个优先级最高的就绪任务。程序清单 L 5.1 OSTimeDly().void OSTimeDly (INT16U ticks) if (tick

4、s 0) (1) OS_ENTER_CRITICAL(); if (OSRdyTblOSTCBCur-OSTCBY &= OSTCBCur-OSTCBBitX) = 0) (2) OSRdyGrp &= OSTCBCur-OSTCBBitY; OSTCBCur-OSTCBDly = ticks; (3) OS_EXIT_CRITICAL(); OSSched(); (4) 清楚地认识0到一个节拍之间的延时过程是非常重要的。换句话说,如果用户只想延时一个时钟节拍,而实际上是在0到一个节拍之间结束延时。即使用户的处理器的负荷不是很重,这种情况依然是存在的。图F5.1详细说明了整个过程。系统每隔10

5、ms发生一次时钟节拍中断F5.1(1)。假如用户没有执行其它的中断并且此时的中断是开着的,时钟节拍中断服务就会发生F5.1(2)。也许用户有好几个高优先级的任务(HPT)在等待延时期满,它们会接着执行F5.1(3)。接下来,图5.1中所示的低优先级任务(LPT)会得到执行的机会,该任务在执行完后马上调用F5.1(4)所示的OSTimeDly(1)。C/OS-会使该任务处于休眠状态直至下一个节拍的到来。当下一个节拍到来后,时钟节拍中断服务子程序会执行F5.1(5),但是这一次由于没有高优先级的任务被执行,C/OS-会立即执行申请延时一个时钟节拍的任务F5.1(6)。正如用户所看到的,该任务实际的

6、延时少于一个节拍!在负荷很重的系统中,任务甚至有可能会在时钟中断即将发生时调用OSTimeDly(1),在这种情况下,任务几乎就没有得到任何延时,因为任务马上又被重新调度了。如果用户的应用程序至少得延时一个节拍,必须要调用OSTimeDly(2),指定延时两个节拍!Figure 5.1 Delay resolution.5.2 按时分秒延时函数 OSTimeDlyHMSM()OSTimeDly()虽然是一个非常有用的函数,但用户的应用程序需要知道延时时间对应的时钟节拍的数目。用户可以使用定义全局常数OS_TICKS_PER_SEC(参看OS_CFG.H)的方法将时间转换成时钟段,但这种方法有时

7、显得比较愚笨。笔者增加了OSTimeDlyHMSM()函数后,用户就可以按小时(H)、分(M)、秒(S)和毫秒(m)来定义时间了,这样会显得更自然些。与OSTimeDly()一样,调用OSTimeDlyHMSM()函数也会使C/OS-进行一次任务调度,并且执行下一个优先级最高的就绪态任务。任务调用OSTimeDlyHMSM()后,一旦规定的时间期满或者有其它的任务通过调用OSTimeDlyResume()取消了延时(参看5.02,恢复延时的任务OSTimeDlyResume(),它就会马上处于就绪态。同样,只有当该任务在所有就绪态任务中具有最高的优先级时,它才会立即运行。 程序清单 L5.2所

8、示的是OSTimeDlyHMSM()的代码。从中可以看出,应用程序是通过用小时、分、秒和毫秒指定延时来调用该函数的。在实际应用中,用户应避免使任务延时过长的时间,因为从任务中获得一些反馈行为(如减少计数器,清除LED等等)经常是很不错的事。但是,如果用户确实需要延时长时间的话,C/OS-可以将任务延时长达256个小时(接近11天)。 OSTimeDlyHMSM()一开始先要检验用户是否为参数定义了有效的值L5.2(1)。与OSTimeDly()一样,即使用户没有定义延时,OSTimeDlyHMSM()也是存在的L5.2(9)。因为C/OS-只知道节拍,所以节拍总数是从指定的时间中计算出来的L5

9、.2(3)。很明显,程序清单 L5.2中的程序并不是十分有效的。笔者只是用这种方法告诉大家一个公式,这样用户就可以知道怎样计算总的节拍数了。真正有意义的只是OS_TICKS_PER_SEC。L5.2(3)决定了最接近需要延迟的时间的时钟节拍总数。500/OS_TICKS_PER_SECOND的值基本上与0.5个节拍对应的毫秒数相同。例如,若将时钟频率(OS_TICKS_PER_SEC)设置成100Hz(10ms),4ms的延时不会产生任何延时!而5ms的延时就等于延时10ms。 C/OS-支持的延时最长为65,535个节拍。要想支持更长时间的延时,如L5.2(2)所示,OSTimeDlyHMS

10、M()确定了用户想延时多少次超过65,535个节拍的数目L5.2(4)和剩下的节拍数L5.2(5)。例如,若OS_TICKS_PER_SEC的值为100,用户想延时15分钟,则OSTimeDlyHMSM()会延时15x60x100=90,000个时钟。这个延时会被分割成两次32,768个节拍的延时(因为用户只能延时65,535个节拍而不是65536个节拍)和一次24,464个节拍的延时。在这种情况下,OSTimeDlyHMSM()首先考虑剩下的节拍,然后是超过65,535的节拍数L5.2(7)和(8)(即两个32,768个节拍延时)。程序清单 L 5.2 OSTimeDlyHMSM().INT

11、8U OSTimeDlyHMSM (INT8U hours, INT8U minutes, INT8U seconds, INT16U milli) INT32U ticks; INT16U loops; if (hours 0 | minutes 0 | seconds 0 | milli 0) (1) if (minutes 59) return (OS_TIME_INVALID_MINUTES); if (seconds 59) return (OS_TIME_INVALID_SECONDS); If (milli 999) return (OS_TIME_INVALID_MILLI);

12、 ticks = (INT32U)hours * 3600L * OS_TICKS_PER_SEC (2) + (INT32U)minutes * 60L * OS_TICKS_PER_SEC + (INT32U)seconds * OS_TICKS_PER_SEC + OS_TICKS_PER_SEC * (INT32U)milli + 500L/OS_TICKS_PER_SEC) / 1000L; (3) loops = ticks / 65536L; (4) ticks = ticks % 65536L; (5) OSTimeDly(ticks); (6) while (loops 0)

13、 (7) OSTimeDly(32768); (8) OSTimeDly(32768); loops-; return (OS_NO_ERR); else return (OS_TIME_ZERO_DLY); (9) 由于OSTimeDlyHMSM()的具体实现方法,用户不能结束延时调用OSTimeDlyHMSM()要求延时超过65535个节拍的任务。换句话说,如果时钟节拍的频率是100Hz,用户不能让调用OSTimeDlyHMSM(0,10,55,350)或更长延迟时间的任务结束延时。5.3 让处在延时期的任务结束延时,OSTimeDlyResume() C/OS-允许用户结束延时正处于延时

14、期的任务。延时的任务可以不等待延时期满,而是通过其它任务取消延时来使自己处于就绪态。这可以通过调用OSTimeDlyResume()和指定要恢复的任务的优先级来完成。实际上,OSTimeDlyResume()也可以唤醒正在等待事件(参看第六章任务间的通讯和同步)的任务,虽然这一点并没有提到过。在这种情况下,等待事件发生的任务会考虑是否终止等待事件。 OSTimeDlyResume()的代码如程序清单 L5.3所示,它首先要确保指定的任务优先级有效 L5.3(1)。接着,OSTimeDlyResume()要确认要结束延时的任务是确实存在的L5.3(2)。如果任务存在,OSTimeDlyResum

15、e()会检验任务是否在等待延时期满L5.3(3)。只要OS_TCB域中的OSTCBDly包含非0值就表明任务正在等待延时期满,因为任务调用了OSTimeDly(),OSTimeDlyHMSM()或其它在第六章中所描述的PEND函数。然后延时就可以通过强制命令OSTCBDly为0来取消L5.3(4)。延时的任务有可能已被挂起了,这样的话,任务只有在没有被挂起的情况下才能处于就绪状态L5.3(5)。当上面的条件都满足后,任务就会被放在就绪表中L5.3(6)。这时,OSTimeDlyResume()会调用任务调度程序来看被恢复的任务是否拥有比当前任务更高的优先级L5.3(7)。这会导致任务的切换。程

16、序清单 L 5.3 恢复正在延时的任务INT8U OSTimeDlyResume (INT8U prio) OS_TCB *ptcb; if (prio = OS_LOWEST_PRIO) (1) return (OS_PRIO_INVALID); OS_ENTER_CRITICAL(); ptcb = (OS_TCB *)OSTCBPrioTblprio; if (ptcb != (OS_TCB *)0) (2) if (ptcb-OSTCBDly != 0) (3) ptcb-OSTCBDly = 0; (4) if (!(ptcb-OSTCBStat & OS_STAT_SUSPEND)

17、 (5) OSRdyGrp |= ptcb-OSTCBBitY; (6) OSRdyTblptcb-OSTCBY |= ptcb-OSTCBBitX; OS_EXIT_CRITICAL(); OSSched(); (7) else OS_EXIT_CRITICAL(); return (OS_NO_ERR); else OS_EXIT_CRITICAL(); return (OS_TIME_NOT_DLY); else OS_EXIT_CRITICAL(); return (OS_TASK_NOT_EXIST); 注意,用户的任务有可能是通过暂时等待信号量、邮箱或消息队列来延时自己的(参看第六

18、章)。可以简单地通过控制信号量、邮箱或消息队列来恢复这样的任务。这种情况存在的唯一问题是它要求用户分配事件控制块(参看6.00),因此用户的应用程序会多占用一些RAM。5.4 系统时间,OSTimeGet()和OSTimeSet() 无论时钟节拍何时发生,C/OS-都会将一个32位的计数器加1。这个计数器在用户调用OSStart()初始化多任务和4,294,967,295个节拍执行完一遍的时候从0开始计数。在时钟节拍的频率等于100Hz的时候,这个32位的计数器每隔497天就重新开始计数。用户可以通过调用OSTimeGet()来获得该计数器的当前值。也可以通过调用OSTimeSet()来改变该

19、计数器的值。OSTimeGet()和OSTimeSet()两个函数的代码如程序清单 L5.4所示。注意,在访问OSTime的时候中断是关掉的。这是因为在大多数8位处理器上增加和拷贝一个32位的数都需要数条指令,这些指令一般都需要一次执行完毕,而不能被中断等因素打断。程序清单 L 5.4 得到和改变系统时间INT32U OSTimeGet (void) INT32U ticks; OS_ENTER_CRITICAL(); ticks = OSTime; OS_EXIT_CRITICAL(); return (ticks);void OSTimeSet (INT32U ticks) OS_ENTER_CRITICAL(); OSTime = ticks; OS_EXIT_CRITICAL();

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1