μCos操作系统

所需积分/C币:12 2018-01-18 12:39:51 4MB PDF
收藏 收藏
举报

μCos操作系统详细介绍,源码注释。用Borland C/C++ 编译器(V3.1)编译过,用选择项产生Intel/AMD80186 处理器(大模式下编译)的代码。这些代码实际上是在Intel Pentium II PC (300MHz)上 运行和测试过,Intel Pentium II PC可以看成是特别快的80186。笔者选择PC 做为目标系 统是由于以下几个原因:首先也是最为重要的,以PC做为目标系统比起以其他嵌入式环境, 如评估板,仿真器等,更容易进行代码的测试,不用不断地烧写EPROM,不断地向EPROM 仿 真器中下载程序等等。用户只需要简单地编译、链接和执行。其次,使用Borla
数 程序清单LL.1可移植型数据类型。 Typedef unsigned char BOOlEAN; yredef unsigned char INT8U 'yredef signed char TNT8S Tyre igne int INT16J fypcdcf signed int INTl6S; Typedef unsigned long INT32 §1g ng INT32S Typedef float FP32; T typedef double FF64; #seine BYTE. INTS dc三 inc UBYTE ⊥NT8U #ce=ine WORD INT16S #ce=ine UWORD INT16U #ce=ine LoNg INT32S #ac=inc UlONG ⊥NT32U 以INT16U数据类型为例,它代表16位无符号整数数据类型。μC/0sII和用户的应用代 码可以定义这种类型的数据,范围从0到65,535。如果将μC0/S-I移植到32位处理器中, 那就意味着ⅠⅥT16U不雨不是一个无符号整型数据,而是一个无符号短整型数据。然而将无论 1C/(OsI用到哪那里,都会当作INT16L处理。表1.1是以 Borland c/CH编译器为例,为80x86 提供的定义语句。为了和pC0S兼容,还定义了BYTE,WORD,LONG以及相应的无符号变量。这 使得用户可以不作仟何修改就能将C/OS的代码移植到μC/0S-II中。之所以这样做是因为笔 者觉得这种新的数据类型定义有更多的灵活性,也更加易读易懂。对·些人来说,WOR意味 着32位数,而此处却意味着16位数。这些新的数据类型应该能够消除此类含混不请 1.03全局变量 以下是如何定义全局变量。众所周知,全局变量应该是得到内存分配且可以被其他模块 通过C语言中 extern关键字调用的变量。因此,必须在.C和.H文件中定义。这种重复的 定义很容易导致错误。以下讨论的方法只需用在头文件中定义一次。虽然有点不易懂,但用户 旦掌握,使用起来却很灵活。表1.2中的定义出现在定义所有全局变量的.H头文件中。 程序清单L1.2定义全局宏。 #i fdef xxX GLOBALS #ac三 inc XXX EX #else #Ce1nexx× FiXT extern .H文件中每个全局变量都加上了ⅹxEXT的前缀。ⅹx代表模块的名字。该模块的.C文件中 有以下定义 # Cene XX× GTOBA S #includc includes. h" 当编译器处理.C文件时,它强制 XXX EX(在相应.H文件中可以找到)为空,(因为 XXX GLOBALS 已经定义)。所以编译器给每个全局变量分配内存空间,而当编译器处理其他.C文件时, XXX GLOBAL没有定义, XXX EXT被定义为 extern,这样用户就可以调用外部全局变量。为了 说明这个機念,可以参见uC/0SII.H,其中包括以下定义: #i fdcf CS GLOBALS #e三 ine OS EXT #∈1se #ce=ine OS EXr extern t cdif OEXT工NT32U osidlectr OS EXT INT32U OSIalectrrun; OS EXT INT32U ○SId1 ectrMax; 同时, uCOS I.H有中以下定义: #e三 ine OS GLOBALS #include includes h/ 当编译器处理 uCOS II.C时,它使得头文件变成如下所示,因为 OS EXT被设置为空。 INT32U oSIdlectr INT32U OSIdlectrruni INT32U osIdlectrMay 这样编译器就会将这些全局变量分配在內存中。当编译器处理其他.C文件时,以文件变成了 如下的样了,因为0 S GLOBAL没有定义,所以0sEXT被定义为 extern。 exern int32U ostdlectr extern Int32U oSIdlectrRun: extern INT32U osIdlectrMax 在这种情况下,不产生内存分配,而任何C文件都可以使用这些变量。这样的就只需在.H 文件中定义一次就可以了。 1.04 OS ENTER CRITICAL ( fl OS EXIT CRITICAL O 用户会看到,调用 OS ENTER CRITICAL O和 OS EXIT CRITICAL O两个宏,贯穿本书的所 有源代码。 OS ENTER CRITICAL关中断;而0 S EXIT CRITICAL0开中断。关中断和开中断 是为了保护临界段代码。这些代码很显然与处理器有关。关于宏的定义可以在 OS CPU.H中找 到。9.03.02节详细讨论定义这些宏的两种方法。 程序情单L1.3进入正确部分的宏。 Hcesine OS CRITICAL METEOD tif CS CRITICAL METHOD - 1 #ce三 ine OS ENTER CR- TICAL() asm CL工 Define Os EXIT CRITICAL( asm ST #endif OS CRITICAL ME THOD = 2 #ce=ine os ENTER CRTTTCATo asm I PUSHF; CT.T J #ce=ine os EXIT CRITICAL( asm POPF t cdif 用户的应用代码可以使用这两个宏米开中断和关中断。很明显,关中断会影响中断延迟, 所以要特别小心。用户还可以用信号量来保护林阶段代码。 1.0基于PC的服务 PC.C文件和PC.H文件(在\ SOFTWARE BL0CKS\PC\ SOURCE日录下)是笔者在范例中使 用到的一些基于PC的服务程序。与1C0s-I以前的版(即pC/0s)不同,笔者希望集 中这些函数以避免在各个例了中都重复定义,也更容易适应不同的编译器。PC.C包括字符显 示,时间度量和其他各种服务。所有的函数都以PC为前缀。 1.05.01字符显示 为了性能更好,显示数直接向显示内存区中写数据。在VGA显示器中,显示内存从绝 对地址0x000B8000开始(或用段、偏移量表示则为B800:000℃)。在单色显示器中,用户 以把井 define constant disp base从0xB800改为0xB000。 PC.C中的显示函数用x和y坐标来直接向显示内存中写ASCI字符。PC的显示可以达到 25行80列共2,000个字符。每个字符需要两个宇节来显示。第·个字节是用户想要显示 的字符,第二个字节用来确定前景色和背景色。前景色用低凹位来表小,背景色用第4位到 6位来表示。最高位表示这个字符是否闪烁,(1)表示闪烁,(0)表示不闪烁。用PC.H中 # defen constants定义前景和背景色,PC.C包括以下四个函数: lear the screen PC DispClrLine o Clear a single row (or line PC DispChar o) Display a single ascii character any where on the screen PC DispStro Display an ascii string anywhere on the screen 1.05.02花费时间的测量 时间测量函数主要用于测试一个函数的运行花了多少时间。测量时间是用PC的82C54 定时器2。被测的程序代码是放在函数 PC ElapsedStart(和PC_E1 apsedStop o之间来测量 的。在用这两个函数之前,应该调用 PC ElapsedInit(来初始化,它主要是计算运行这两个 函数本身所附加的的时间。这样, PC Elapsedstop(函数中返回的数佰就是准确的测量结果 了。注意,这两个函数都不具各可重入性,所以,必须小心,不要有多个任务同时调用这两 个函数。表1.4说明了如何测量 PC Displaychar(的执行时问。注意,时间是以uS为单位 的 程序清单L1.4测量代码行的间。 INFlu time PC FlapsedTnit( PC Elapsedstart(i PC Di spChar(40,24, A, DTSP FGND NETTE) timc- PC Elapscastop()i 1.05.03其他函数 COsI的应用程序和其他D0s应用程序是一样的,换句话说,用户可以像在DOS下编 译其他单线程的程序一样编译和链接用户程序。所生成的.EXE程序可以在DOS卜装载和运行, 当然应用程序应该从main()函数开始。因为μC/0s-I是多任务,而且为每个任务开辟 个堆栈,所以单线程的⑩Os环境应该保存,在退出μC/OsII程序时返回到D0s。调用 PCD0 SSaveReturn(可以保存当前DOS环境,而调用 PC DOSReturn(可以返叫到DoS。PC.C 中使用ANSⅠC的 set]mp(, long jmp函数来分别保存和恢复DOS环境。 Borland c/C-+编译 库提供这些函数,多数其它的编译程序也应有这类函数。 应该注意到无论是应用程序的错误还是只调用exit(0)而没有调用 PC DOSReturn(函数 都会使D0S环境被破坏,从而导致DOS或 WINDOWS95下的D0S窗口崩溃。 调用 PC GetDatelime(函数可得到PC中的口期和时间,并且以SACl⊥字符串形式返回 格式是MM-DD-YYHH:M:SS,用户需要19个字符来存放这些数据。该函数使用了 Borland c/C++ 的 gettime(和 getdate o函数,其它DS环境下的C编译应该也有类似函数。 PC GetKey(函数检査是否有按键被按下。如果有按键被按下,函数返回其值。这个函 数使用了 Borland o/C++的 khit()和 getch(函数,其它D0S环境下的C编译应该也有类似 函数。 哟数 PC SettickRate(允许用户为uC0S-I定义频率,以改变钟节拍的速率。在D0S 下,每秒产生18.20618次时钟节拍,或每隔54.925ms一次。这是因为82C51定时器芯片没 有初始化,而使用默认值65,535的结果。如果初始化为58,659,那么时钟节拍的速率就会 精确地为20.000H。笔者决定将时钟节拍设得史快一些,用的是200Hz(实际是上是 199.9966Hz)。注意OS_CPUA.ASM中的 STickISr(函数将会每11个时钟节拍调用一次DoS 中的时钟节拍处理,这是为了保证在DOS下时钟的准确性。如果用户希望将吋钟节拍的速度 殳置为20HZ,就必须这样做。在返回DOS以前,要调用 PC SetTickRate(,并设置18为目 标频率, PC Settickrate(就会知道用户要设置为18.2Hz,并且会正确设置82C54 PC.C中最后两个函数是得到和设置中断向量,笔者是用 Borland c/C++中的库函数来完 成的,但是 PC VectGet(和 PC VectSet(很容易改写,以适用于其它编译器。 1.06应用pC/0s-I的范例 木章中的例子都用 Borland c/C+-编译器编译通过,是在 Windows95的D0S窗口下编译 的。可执行代码可以在每个范例的OBJ子目录下找到。实际上这些代码是在 Borland ide ( Integrated Development Environment)下编译的,编译时的选项如表1.1所小: 表T1.1ⅠDE中编译选项。 Code generation Model Large Options Treat enums as ints Assume ss Equals ds Default for memory model vanced code generation Floating point Emulation Instruction set 80186 Options Generate underbars Debug info in OB IS Fast floating point Optimizations Optimizations Global register allocation Invariant code motion Induction variables Loop optimization Suppress redun dant load Copy propagation Dead code climination ump optimization In-line intrinsic functions Register variables Automatic Common subexpressions Optimize globally Optimize for S peed 笔者的 Borland c/C+-编译器安装在C:\CPP目录下,如果用户的编译器是在不同的目录 下,可以在 Options/ Directories的提示下改变IDE的路径 μC/OS-II是一个可裁剪的操作系统,这意味着用户可以去掉不需要的服务。代码的削减 可以通过设置0SCFG.H中的# defines os??FN为0米实现。用户不需要的服务代码就不 生成。本章的范例就用这种功能,所以每个例子都定义了不同的0S??2EN。 1.07例1 第一个范例可以在\ SOFTWARE{ uCOS II\EX1x86L日录下找到,它有13个任务(包括 1C/(0s-TI的空闲任务)。1COs-增加了两个内部任务:空闲任务和一个计算CP利用率 的任务。例1建立了11个其它任务。1 askstart(任务是在函数main0中建立的;它的功能 是建立其它任务并且在屏幕上显示如下统计信息: 每秒钟任务切换次数; ●CPU利用百分率 ●寄存器切换次数; 目前日期和时间; ●C/OS-I的版本号 TaskStart(还检查是否按下ESC键,以决定是否返回到D0S。 其余10个任务基于相同的代码—Task();每个任务在屏幕上随机的位置显示一个0到 9的数字 1.07.01main0 例1基本上和最初uCOS中的第·个例了做样的事,但是笔者整理了其中的代码,并 且在屏幕上加了彩色显示。同时笔者使用原来的数据类犁( UBYTE,JRD等)来说明 1C/(0s-I向下兼容 main(程序从清整个屏幕廾始,为的是保证屏幕上不留有以前的DOS下的显示 L1.5(1)|。注意,笔者定义了白色的字符和黑色的背景色。既然要请屏幕,所以可以只定义 背景色而不定义前景色,但是这样在退回DOS之后,用户就什么也看不见了。这也是为什么 总要定义一个可见的前景色。 1COS-II要用户在使用仟何服务之前先调用 OSInit()[L.5(2)]。它会建立两个仟务 空闲任务和统计任务,前者在没有其它任务处于就绪态时运行;后者计算CPU的利用率。 程序单LL.5main() d main (void) PC DispClrScr(DISP FGND WHITE DISP EGND BLACK)i OSInit( PC DOSSavereturnoi C vcctsct(ucos, CSCt Rar.domSem OSSemCreate(l (5) osTaskCreae(Tasksta 6 (void *)0, (void *)aTaskstartstk[TASK STK SIZE-l1 aSSart 当前D0S环境是通过调用 PC DOSSaveReturno[L1.5(3)来保存的。这使得用户可以返 冋到没有运行μC/0s-I以前的DS环境。跟随清单11.6中的程序可以看到 DOSSaveReturn O做了很多事情。 PC DOSSavereturn(首先设置 PC Exitrlag为 FALSE L1.6(1),说明用户不是要返回DOS,然后初始化0 STickdosctr为1L1.6(2),因为 这个变量将在0 STickISR O中递减,而0将使得这个变量在 OSTickisro中减1后变为25。 然后, PC DOSSaveReturn(将DOs的时钟节拍处理( tick handler)存入个自由向量表入 口中1.6(3)-(4)1,以便为COS-I的时钟节拍处理所调用。接着 PC DOSSavereturn(调 用jmp()[L1.6(5)],它将处理器状态〔即所有寄存器的值)存入被称为 PC JumpBuf的结 构之中。俣存处理器的全部寄存器使得程序返回到 PC DOSSaveReturn O并且在调用 set Jmp ()之后立即执行。因为 PC ExitFlag被初始化为 FALSELL1.6(1)]。PC_ DOSSavcReturn O跳 过i状态语句工1.6(6)-(9)]回到main()函数。如果用户想要返回到DOs,可以调用 PC DOSReturn((程序清单L1.7),它设置 PC ExitFlag为TRUE,并且执行 long jmp()语 句[1.7(2)],这时处理器将跳 PC DOSSaveReturn O[在调用 set]mp()之后][1.6(5)], 此时 PC Exitrlag为TRUE,故if语句以后的代码将得以执行。 PC DOSSavereturn(将时钟 节拍改为18.2Hz[Ll.6(6)],恢复PC时钟节拍中饧服务[L1.6(7)],清屏幕[L1.6(8)],通 过exit(0)返同DS[L1.6(9)] 程序清单Ll.6保存DOS环境。 void Pc DoSSaveReturn (void) PC ExitFlag FALSE OSTiCkDOS(-≌ (2) PC TiCKISR PC VectGe(VECT TICK); (3) OS ENTER CRITICAL (i PC Vcctsct(vECI DCS CHAiN, PC TickIS3)i OS EXIT CRITICAL ( Setjmp(PC JumpBuf)i (5) if (PC Ex a==TRU三) OS ENTER CRIT工CaL() PC SetT=ckRate(18) (6) PC VcctSct(VECT TICK, PC TickISR)i (7) OS EXIT CRITICAL ( PC Di spClrScr(DISP FGND WHITE DISP BGND BLACK) exit(o)i 9) 程序情单L17设置返应DOS。 ⅴ id pc DosReturn(void PC ExIt上1ag=IRUE lorgjmmp(PC JunpBufr 1)i (2) 现在回到main()这个函数,在程序清单L1.5中,main()调用 PC VectSet来设 置μCOs中的CU寄存器切换。任务级的CPU寄存器切换由80x86IN指令米分配向量地 址。笔者使用向量0x80(即128),因为它未被DOS和BI0S使用。 这里用了一个信号量来保护 Borland c/C++库中的产牛随机数的函数L1.5(5)],之所以 使用信号量保护·下,是因为笔者不知道这个函数是否具备可重入性,笔者假设其不具备, 初始化将信号量设置为1,意思是在某一时刻只有一个任务可以调用随机数产生函数。 在开始多任务之前,笔者建立了一个叫做 TaskStart(的任务[1.5(6)],在启动多任务 OS5tart(之前用户至少要先建立个任务,这·点非常重要L1.5(7)]。不这样做用户的应用 程序将会崩溃。实际上,如果用户要计算CPU的利用率时,也需要先建立一个任务。1COS-II 的统计任务要求在整个一秒钟内没有任何其它任务运行。如果用户在启动多任务之前要建立 其它任务,必须保证用户的任务代码监控全局变量 OSStatRdy和延时程序[即调用 STimeDly()1的执行,直到这个变量变成TRUE。这表明pC0s-I的CPU利用率统计函数已 经采集到了数据。

...展开详情
试读 127P μCos操作系统
立即下载 低至0.43元/次 身份认证VIP会员低至7折
    抢沙发
    一个资源只可评论一次,评论内容不能少于5个字
    关注 私信 TA的资源
    上传资源赚积分,得勋章
    最新推荐
    μCos操作系统 12积分/C币 立即下载
    1/127
    μCos操作系统第1页
    μCos操作系统第2页
    μCos操作系统第3页
    μCos操作系统第4页
    μCos操作系统第5页
    μCos操作系统第6页
    μCos操作系统第7页
    μCos操作系统第8页
    μCos操作系统第9页
    μCos操作系统第10页
    μCos操作系统第11页
    μCos操作系统第12页
    μCos操作系统第13页
    μCos操作系统第14页
    μCos操作系统第15页
    μCos操作系统第16页
    μCos操作系统第17页
    μCos操作系统第18页
    μCos操作系统第19页
    μCos操作系统第20页

    试读已结束,剩余107页未读...

    12积分/C币 立即下载 >