当前位置:首页 > 嵌入式 > 嵌入式软件
[导读]先把入口函数main给贴出来,就从这里开始,来自文件main.c/******************************************************************************** Function Name : main* De

先把入口函数main给贴出来,就从这里开始,来自文件main.c

/*******************************************************************************

* Function Name : main

* Description : 主函数,对系统以及硬件初始化,建立主函数并开启系统

* Input : None

* Output : None

* Return : None

*******************************************************************************/

int main(void)

{

CPU_IntDis(); // 禁止CPU中断 连接到汇编

OSInit(); // uCOS系统初始化

BSP_Init(); // 硬件初始化

OSTaskCreate //建立主任务, 优先级最高 建立这个任务另外一个用途是为了以后使用统计任务

(

(void (*) (void *)) App_TaskStart, //指向任务代码的指针

(void *) 0, //任务开始执行时,传递给任务的参数的指针

(OS_STK *) &App_TaskStartStk[APP_TASK_START_STK_SIZE - 1], //分配给任务的堆栈的栈顶指针,从 (INT8U) APP_TASK_START_PRIO //分配给任务的优先级

);

OSTimeSet(0);

OSStart();

return(0);

}

在开始 uC/OS_II 的调度之前,我们需要调用函数OSInit(),他负责建立任务控制块链表,就绪任务表等数据结构,然后初始化全局变量。然后把需要用的外部设备进行初始化,主要是时钟初始化,中断嵌套初始化,端口初始化,调用函数BSP_Init(),uC/OS_II规定在任务调度开始前至少有一个任务已经建立,所以我们建立一个任务APP_TaskStart,并且给这个任务分配优先级以及堆栈等资源这是必须的啦,然后我们用OSTimeSet(0)函数初始化系统的时钟节拍数后,就调用OSStart()函数开始任务调度,任务就会从所有建立的任务里最高优先级开始执行。

大家还记得刚才建立了一个APP_TaskStart任务,在系统开始任务调度的时候,系统里除了默认的优先级最低的空闲任务外只有这一个任务被注册了,自然就会运行这个任务,我们先来看下他的相关源代码来自文件task.c:

/*******************************************************************************

* Function Name : App_TaskStart

* Description : 主任务

* Input : None

* Output : None

* Return : None

*******************************************************************************/

void App_TaskStart(void* p_arg)

{

(void) p_arg;

OS_CPU_SysTickInit(); // 初始化系统心跳

#if (OS_TASK_STAT_EN > 0)

OSStatInit(); // 统计任务初始化函数

#endif

App_TaskCreate(); // 创建新的用户任务

while(1)

{

LED4_HIGH;

OSTimeDlyHMSM(0,0,1,0);

LED4_LOW;

OSTimeDlyHMSM(0,0,1,0);

}

}

/*******************************************************************************

* Function Name : App_TaskCreate

* Description : 建立用户任务

* Input : None

* Output : None

* Return : None

*******************************************************************************/

void App_TaskCreate(void)

{

//===================================================================// 测试任务1

OSTaskCreateExt(

Task_Test1, // 指向任务代码的指针,也就是任务函数名

(void *)0, // 任务开始执行时传递给任务的参数

(OS_STK *)&Task_Test1Stk[Task_Test1_STK_SIZE-1],//分配给任务堆栈的栈顶指针,自顶向下

Task_Test1_PRIO, // 分配给任务的优先级

Task_Test1_PRIO, // 预备给以后版本的标识符,现在同任务优先级

(OS_STK *)&Task_Test1Stk[0], // 指向任务堆栈的栈底指针,用于堆栈的检验

Task_Test1_STK_SIZE, // 指定堆栈的容量,用于堆栈检验

(void *)0, // 指向用户附加数据域的指针,用来扩展任务控制块

OS_TASK_OPT_STK_CHK|OS_TASK_OPT_STK_CLR // 任务选项:使能堆栈检测 和 创建任务时清空堆栈

);

//===================================================================// 测试任务2

OSTaskCreateExt(

Task_Test2,

(void *)0,

(OS_STK *)&Task_Test2Stk[Task_Test2_STK_SIZE-1],

Task_Test2_PRIO,

Task_Test2_PRIO,

(OS_STK *)&Task_Test2Stk[0],

Task_Test2_STK_SIZE,

(void *)0,

OS_TASK_OPT_STK_CHK|OS_TASK_OPT_STK_CLR

);

//===================================================================// 测试任务3

OSTaskCreateExt(

Task_Test3,

(void *)0,

(OS_STK *)&Task_Test3Stk[Task_Test3_STK_SIZE-1],

Task_Test3_PRIO,

Task_Test3_PRIO,

(OS_STK *)&Task_Test3Stk[0],

Task_Test3_STK_SIZE,

(void *)0,

OS_TASK_OPT_STK_CHK|OS_TASK_OPT_STK_CLR

);

}

同时给出任务的优先级及堆栈大小等信息来自文件app_cfg.h

//任务优先级

#define APP_TASK_START_PRIO 10

#define Task_Test1_PRIO 7

#define Task_Test2_PRIO 8

#define Task_Test3_PRIO 9

//任务堆栈大小

#define APP_TASK_START_STK_SIZE 64

#define Task_Test1_STK_SIZE 128

#define Task_Test2_STK_SIZE 128

#define Task_Test3_STK_SIZE 128

可以看到,任务APP_TaskStart的优先级最低,所以在这个任务里创建其他的任务的时候他就会被更高优先级的任务把CPU的占有权抢去,在uC/OS_II里每建立一个任务后都会产生一次任务的调度,如果这个建立的任务优先级更高,则系统就会去执行这个刚创立的任务,如果低就只能等着了。所以在建立Task_Test1任务后,就会跳转执行此任务,现在我们来看下这三个测试任务的源代码来自文件app.c[!--empirenews.page--]

/*******************************************************************************

* Function Name : Task_Test1

* Description : 任务1

* Input : None

* Output : None

* Return : None

*******************************************************************************/

void Task_Test1(void* p_arg)

{

(void) p_arg ;

while(1)

{

if(KEY_WKUP == 0)

{

LED1_HIGH;

}

else

{

LED1_LOW;

}

OSTimeDlyHMSM(0,0,0,100); // 延时,为其他低优先级的任务执行留有空间

}

}

/*******************************************************************************

* Function Name : Task_Test2

* Description : 任务2

* Input : None

* Output : None

* Return : None

*******************************************************************************/

void Task_Test2(void* p_arg)

{

(void) p_arg ;

while(1)

{

LED2_HIGH;

OSTimeDlyHMSM(0,0,0,500);

LED2_LOW;

OSTimeDlyHMSM(0,0,0,500);

}

}

/*******************************************************************************

* Function Name : Task_Test3

* Description : 任务3

* Input : None

* Output : None

* Return : None

*******************************************************************************/

void Task_Test3(void* p_arg)

{

(void) p_arg ;

while(1)

{

LED3_HIGH;

OSTimeDlyHMSM(0,0,0,500);

LED3_LOW;

OSTimeDlyHMSM(0,0,0,500);

}

}

刚才说到哪里了?对,现在开始执行Task_Test1的任务了,很显然,这个任务是对一个按键的检测,检测完成后进入一个100ms的延时函数,在uC/OS_II里延时函数的作用要比以前的大得多,在uC/OS_II里任务中调用了延时函数时,该任务就被退出了任务就绪表,然后调用一次系统调度OS_Sched(),重新寻找最高优先级的任务去执行,这个时候咱们的系统只注册了两个任务,这样就只能继续执行任务APP_TaskStart了,刚才在这个函数里建立第一个测试任务就被抢去了CPU的占用权,现在回来继续创建Task_Test2第二个测试任务,这第二个测试任务优先级也比开始任务高,所以自然的它又被人抢去了CPU的占用权,在第二个测试任务里大家又会看到有延时函数,功能肯定是相同的,以此类推第三个测试任务也就清晰的多了。这时还有一个问题就是延时结束后系统操作,刚才测试任务1进入延时后,不会被系统调用,他的延时变量OSTCBDly是随着系统的心跳进行递减的,如果有两个任务同时在延时中只要他们的任务控制块里的延时变量不是0就会在心跳中断服务函数里减1,等到他被减为0,系统会把这个任务重新放到任务就绪表里,并且运行一次系统调度函数OS_Sched(),通过这种机制来保证系统始终在运行着最高优先级的任务。这样系统的调度问题就解释完了,在后面关于中断和信号量使用时,他们对系统的执行顺序也是有影响的,他们是如何参与到系统的调度里,就看后面的讲解了。

刚才我们说了延时函数的作用,如果我把上面的代码里黄色高亮的部分注视掉会有什么样的效果,结果就是其他的3个任务都无法运行了,系统将一直处理任务Task_Test1,再加入我们如果把黄色高亮部分的延时参数调大一些,调整到1s,结果也不令人满意,虽然我给了提起任务充足的运行时间,但是由于每检测一次我都会等待很长时间,导致我的检测精度大大降低,按键变的极为难用。这时候我们就应该思考,在uC/OS_II这样一个实时的嵌入式操作系统里面,最可怕的就是无目的的等待,所以我们尽量使用中断这样方式去处理接口信息,中断和实时系统是相得益彰的。

 

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

在六个服务领域表现出显著提升;从2021年的第7位跃居第2位 印度班加罗尔2022年4月11日 /美通社/ -- 下一代数字服务和咨询领域的全球领军者Infosys(印度国家证券交易所、孟买证券交易所和纽约证券交易所代...

关键字: Group os

(全球TMT2022年4月11日讯)下一代数字服务和咨询领域的全球领军者Infosys近期宣布,在Everest Group PEAK Matrix®2022年度 IT服务提供商奖中排名第二。在第六届PEAK Matr...

关键字: Group os

香港2022年3月29日 /美通社/ -- 在传统与创新的十字路口,一些人不仅记录了历史,而且潜心打造未来。回顾Toshiba TV第一次屏幕闪烁的那一刻,创新便已深植于公司的“血液”,致力于真实...

关键字: os

北京2022年3月21日 /美通社/ -- 微美全息软件有限公司(纳斯达克: WIMI)(以下简称为“微美全息”或“公司”),一家全球领先的增强现实(“AR”)服务提供商,今天宣布宣布被纳入Proshares Metav...

关键字: os

Infosys 元宇宙工坊(Infosys metaverse foundry)是 Infosys Living Labs 不可或缺的一部分,它正在推动企业的数字创新议程,以适应新兴的优先事项和市场趋势。 Infosys...

关键字: os

Chronos芯片将搭载于Ouster面向量产车推出的DF系列固态数字激光雷达上,提高其性能与功效...

关键字: 芯片 激光雷达 os

(全球TMT2022年3月11日讯)2022年3月11日,高分辨率数字激光雷达供应商Ouster发布最新Chronos 芯片,这是一款车规级、完全定制化的数字激光雷达接收芯片,将搭载于其DF系列固态激光雷达上。Chro...

关键字: 芯片 激光雷达 os

上海2022年3月10日 /美通社/ -- 中国上海,2022年3月,柏中(Bossard)宣布推出Smart Vending Cloud帮助客户解决MRO耗材管理问题,进一步升级“智能工厂物流”服务。此举也意味着柏中在...

关键字: 全自动 os

(全球TMT2022年2月22日讯)孚昇电子(Fusion Worldwide)收购了位于新加坡的大型电子元件测试公司Prosemi Mfg Pte Ltd。由于供应链经历了历史性的限制,导致周转时间过长,此次收购将进...

关键字: os se 电子元件

苏黎世2022年2月14日 /美通社/ -- 通过向HeiQ AeoniQ LLC进行500万美元的投资股权,并根据业绩里程碑计划额外投资400万美元,HUGO B...

关键字: os ni
关闭
关闭