当前位置:首页 > 单片机 > 单片机
[导读]这段时间已成功把uC/OS-II 2.52移植到51单片机上,现总结移植步骤:1.在main函数只包括void main(void){OSInit();OSStart();}看是否能编译通过,可以通过后继续下一步。2.验证OSTaskStkInit()和OSStartHighRdy()函数这

这段时间已成功把uC/OS-II 2.52移植到51单片机上,现总结移植步骤:

1.在main函数只包括
void main(void)
{
OSInit();

OSStart();
}
看是否能编译通过,可以通过后继续下一步。
2.验证OSTaskStkInit()和OSStartHighRdy()函数
这 里首先要修改OS_CFG.H文件,设置OS_TASK_STAT_EN 为0,以禁止统计任务。只让空闲任务工作,并单步执行,直到uC/OS-II切换到OS_TaskIdle()。单步时跳过OSInit()函数,单步进 入OSStart()函数。 一直单步运行到调用OSStartHighRdy()(这是OSStart()函数的最后一句),然后单步进入OSStartHighRdy()。这时, 编译器应该切换到汇编语言模式下,因为OSStartHighRdy()是用汇编语句实现的。OSStartRdy()会开始运行第1个任务;而此时并没 有任何应用任务.只有OS_TaskIdle()可以运行。如果调试器在OS_TaskIdle()循环中运行,且在无限循环中已经执行几次,那么就已经 验证了OSTaskStkInit()和OSStartHighRdy()函数是成功的。

3.验证OSCtxSw()函数 (注这是中断是关着的,见OS_CORE.C的OS_Sched())
如果上一步测试成功,这一步代码验 证就比较容易了.因为已经知道由OSTaskStkInit()初始化的堆栈结构是正确的。在这一步测试中,添加了一个应用程序.并不断切换到空闲任务。 在这之前,应该保证已经正确设置了软中断向量或指令陷阱TRAP向量,使之指向OSCtxSw()函数。这里uC/OS-II应该将TestTask() (假设刚建立的任务名为它)作为第1个任务开始执行,而不再是空闲任务。可以单步执行,一直到TestTask()函数的开始.注这时并没有允许中断,也 没有打开时钟节拍,所以OSTimeDly(1)不会返回到TestTask(). 并经调度后由OSCtxSw()返回到OS_TaskIdle()任务中。这样的说明OSCtxSw()函数移植成功。

#i nclude "includes.h"
OS_STK TestTaskStk[100];


void TestTask(void *ptrdata)
{
ptrdata=ptrdata;
while(1)
{
OSTimeDly(1);
}
}

void main(void)
{
OSInit();

OSTaskCreate(TestTask,(void *)0,&TestTaskStk[99],0);//51仿真堆栈?C_XBP递增
OSStart();
}

3.验证OSIntCtxSw()和OSTickISR()函数
OSIntCtxSw()很像OSCtxSw(),且比OSCtxSw()简单。
在此测试之前,应该保证时钟中断向量指向了时钟节拍中断服务子程序,然后,初始化时钟节拍并开中断。我这里测试使用100Hz的节拍,节拍的速度应该与在OS_CFG.H中设置的OS_TICKS_PER_SEC吻合。
测试代码如下:
#i nclude "includes.h"
sbit LedBlink=P1^0;

OS_STK TestTaskStk[100];


void TestTask(void *ptrdata)
{
ptrdata=ptrdata;

OS_ENTER_CRITICAL();

TMOD=0x01;
TH0=0xdc;//10ms,100hz
TL0=0x00;
TR0=1;
ET0=1;

OS_EXIT_CRITICAL();

while(1)
{
OSTimeDly(1);
if (LedBlink==FALSE)
{
LedBlink=TRUE;

}else
{
LedBlink=FALSE;

}
}
}

void main(void)
{
OSInit();

OSTaskCreate(TestTask,(void *)0,&TestTaskStk[99],0);//51仿真堆栈?C_XBP递增
OSStart();
}

时钟节拍中断调用OSTickISR(),继而调用OSTimeTick。
说明:OSTickISR中一定要有中断调度程序,否则的话,空闲任务无法再切换到其它应用任务,因为空闲任务没有任务延时函数.

编译成功后,点Run,然后在TestTask任务OSTimeDly(1);处设置断点,并单步进入,看能否正确任务调度到 OSTaskIdle()即空闲任务,空闲任务一直到运行,直到接收到时钟节拍中断(即Timer0),时钟节拍中断调用OSTickISR(),继而调 用OSTimerTick().OSTimerTick()将TestTask()的.OSTCBDly计数器递减到0,使该任务进入就绪态;而当 OSTickISR()完成并调用OSIntExit()函数时,OSIntExit()将会注意到有更重要的任务,TestTask()已处于就绪态等 待运行。这时ISR就不会再返回到空闲任务,而是做任务切换,回到TestTask()任务。当然,以上是假设OSIntCtxSw()和 OSTickISR()这2个函数都能正常工作.如果OSIntCtxSw()正常工作,而且已经将时钟节拍频率设置为100Hz(10ms),这里可以 看到LED即P1.0在来回闪烁.

4.然后再写其它中断服务函数的入口处程序,可以参考OSTickISR()。

5.至此,uC/OS-II的移植工作就OK了,可以添加更多的应用程序了,真正的项目工作可以开始了。


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

单片机内部有很多的特殊功能寄存器,每个寄存器在单片机内部都分配有唯一的地址,一般我们会根据寄存器功能的不同给寄存器赋予各自的名称,当我们需要在程序中操作这些特殊功能寄存器时,必须要在程序的最前面将这些名称加以声明,声明的...

关键字: C51 数据类型 扩充定义

数据元(Data Element),也称为数据元素,是用一组属性描述其定义、标识、表示和允许值的数据单元,在一定语境下,通常用于构建一个语义正确、独立且无歧义的特定概念语义的信息单元。数据元可以理解为数据的基本单元,将若...

关键字: C51 数据类型

▼点击下方名片,关注公众号▼欢迎关注【玩转单片机与嵌入式】公众号,回复关键字获取更多免费资料。回复【加群】,限时免费进入知识共享群;回复【3D封装库】,常用元器件的3D封装库;回复【电容】,获取电容、元器件选型相关的内容...

关键字: C51 MDK RealView

在Keil C51软件中51单片机的中断服务和外设驱动程序的开发

关键字: keil5 编译 C51

Intel公司1980年推出了MCS-51系列单片机:集成 8位CPU、4K字节ROM、128字节RAM、4个8位并口、1个全双工串行口、2个16位定时/计数器。寻址范围64K,并有控制功能较强的布尔处理器。 80C5...

关键字: C51 KEIL 编程

c上标3下标5怎么算用计算机,c上标3下标5怎么算

关键字: C51 KEIL

▼点击下方名片,关注公众号▼大家好,很高兴和各位一起分享我的第16篇原创文章,喜欢和支持我的工程师,一定记得给我点赞、收藏、分享。加微信[xyzn3333]与作者沟通交流,免费获取更多单片机与嵌入式的海量电子资料。很多初...

关键字: 51单片机 C51

常看见初学者要求使用_at_,这是一种谬误,把C当作ASM看待了。在C中变量的定位是编译器的事情,初学者只要定义变量和变量的作 用域,编译器就把一个固定地址给这个变量。

关键字: C51 单片机 误区 注意事项

简介:编程首要是要考虑程序的可行性,然后是可读性、可移植性、健壮性以及可测试性。这是总则。但是很多人忽略了可读性、可移植性和健壮性(可调试的方法可能歌不相同),这是不对的。

关键字: C51 编程规范 文件配置

如果你用 Keil C51 进行编译,记住一点:它不区分大小写!!!卧槽,今天编程序那个调错啊,就因为一个数组名和一个变量名完全一样,只是大小写不一样罢了,标准 C 我怎么记得这样可以啊……上网一查,卧槽,Keil C5...

关键字: C51 单片机 编程要点
关闭
关闭