μC/OS-II的任务管理
μC/OS-Ⅱ支持的处理器的堆栈既可以从上(高地址)往下(低地址)递减也可以从下往上递增(参看4.02,任务堆栈)。用户在调用OSTaskCreateExt()的时候必须知道堆栈是递增的还是递减的(参看用户所用处理器的OS_CPU.H中的OS_STACK_GROWTH),因为用户必须得把堆栈的栈顶传递给OSTaskCreateExt(),而栈顶可能是堆栈的最低地址(当OS_STK_GROWTH 为0时),也可能是最高地址(当OS_STK_GROWTH为1时)。 一旦OSTaskStkInit()函数完成了建立堆栈的任务,OSTaskCreateExt()就调用 OSTCBInit()[L4.3(7)], 从空闲的OS_TCB缓冲池中获得并初始化一个OS_TCB。 OSTCBInit() 的代码在OSTaskCreate()中曾描述过(参看4.00节),从OSTCBInit()返回后, OSTaskCreateExt()要检验返回代码[L4.3(8)],如果成功,就增加OSTaskCtr[L4.3(9)], OSTaskCtr用于保存产生的任务数目。 如果OSTCBInit()返回失败, 就置OSTCBPrioTbl[prio] 的入口为0[L4.3(13)]以放弃对该任务优先级的占用。然后,OSTaskCreateExt()调用 OSTaskCreateHook()[L4.3(10)],OSTaskCreateHook()是用户自己定义的函数,用来扩展 OSTaskCreateExt()的功能。OSTaskCreateHook()可以在OS_CPU_C.C中定义(如果 OS_CPU_HOOKS_EN置1),也可以在其它地方定义(如果OS_CPU_HOOKS_EN置0)。注意, OSTaskCreateExt()在调用OSTaskCreateHook()时,中断是关掉的,所以用户应该使 OSTaskCreateHook()函数中的代码尽量简化,因为这将直接影响中断的响应时间。 OSTaskCreateHook()被调用时会收到指向任务被建立时的OS_TCB的指针。这意味着该函数可以访问OS_TCB数据结构中的所有成员。 如果OSTaskCreateExt()函数是在某个任务的执行过程中被调用的(即OSRunning置为 True[L4.3(11)]),以任务调度函数会被调用[L4.3(12)]来判断是否新建立的任务比原来的任务有更高的优先级。如果新任务的优先级更高,内核会进行一次从旧任务到新任务的任务切换。如果在多任务调度开始之前(即用户还没有调用OSStart()),新任务就已经建立了,则任务调度函数不会被调用。 4.2任务堆栈 每个任务都有自己的堆栈空间。堆栈必须声明为OS_STK类型,并且由连续的内存空间组成。用户可以静态分配堆栈空间(在编译的时候分配)也可以动态地分配堆栈空间(在运行的时候分配)。静态堆栈声明如程序清单L4.4和4.5所示,这两种声明应放置在函数的外面。 程序清单 L4.4 静态堆栈 staticOS_STKMyTaskStack[stack_size]; 或 程序清单 L4.5 静态堆栈 OS_STKMyTaskStack[stack_size]; 用户可以用C编译器提供的malloc()函数来动态地分配堆栈空间,如程序清单L4.6所示。在动态分配中,用户要时刻注意内存碎片问题。特别是当用户反复地建立和删除任务时,内存堆中可能会出现大量的内存碎片,导致没有足够大的一块连续内存区域可用作任务堆栈,这时malloc()便无法成功地为任务分配堆栈空间。 程序清单 LL4.6 用malloc()为任务分配堆栈空间 OS_STK*pstk; pstk=(OS_STK*)malloc(stack_size); if(pstk!=(OS_STK*)0){/* 确认malloc()能得到足够地内存空间 */ Createthetask; } 图4.1表示了一块能被malloc()动态分配的3K字节的内存堆[F4.1(1)]。为了讨论问题方便,假定用户要建立三个任务(任务A,B和C),每个任务需要1K字节的空间。设第一个1K字节给任务A,第二个1K字节给任务B,第三个1K字节给任务C[F4.1(2)]。然后,用户的应用程序删除任务A和任务C,用free()函数释放内存到内存堆中[F4.1(3)]。现在,用户的内存堆虽有2K字节的自由内存空间,但它是不连续的,所以用户不能建立另一个需要2K字节内存的任务(即任务D)。如果用户并不会去删除任务,使用malloc()是非常可行的。 图 F4.1 内存碎片 μC/OS-Ⅱ支持的处理器的堆栈既可以从上(高地址)往下(低地址)长也可以从下往上长(参看4.02,任务堆栈)。用户在调用OSTaskCreate()或OSTaskCreateExt()的时候必须知道堆栈是怎样长的,因为用户必须得把堆栈的栈顶传递给以上两个函数,当OS_CPU.H文件中的OS_STK_GROWTH置为0时,用户需要将堆栈的最低内存地址传递给任务创建函数,如程序清单4.7所示。 程序清单 L4.7 堆栈从下往上递增 OS_STKTaskStack[TASK_STACK_SIZE]; OSTaskCreate(task,pdata,TaskStack[0],prio); 当OS_CPU.H文件中的OS_STK_GROWTH置为1时,用户需要将堆栈的最高内存地址传递给任务创建函数,如程序清单4.8所示。 程序清单 L4.8 堆栈从上往下递减 OS_STKTaskStack[TASK_STACK_SIZE]; OSTaskCreate(task,pdata,TaskStack[TASK_STACK_SIZE-1],prio); 这个问题会影响代码
- Windows CE 进程、线程和内存管理(11-09)
- RedHatLinux新手入门教程(5)(11-12)
- uClinux介绍(11-09)
- openwebmailV1.60安装教学(11-12)
- Linux嵌入式系统开发平台选型探讨(11-09)
- Windows CE 进程、线程和内存管理(二)(11-09)
