第7章 任务间的同步与通讯之消息邮箱_第1页
第7章 任务间的同步与通讯之消息邮箱_第2页
第7章 任务间的同步与通讯之消息邮箱_第3页
第7章 任务间的同步与通讯之消息邮箱_第4页
第7章 任务间的同步与通讯之消息邮箱_第5页
已阅读5页,还剩48页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

第七章任务间的同步与通讯消息邮箱什么是消息邮箱消息邮箱的数据结构消息邮箱相关函数消息邮箱指针消息邮箱任务1任务2发送消息请求消息消息缓冲区(发送消息缓冲区指针)(读取消息缓冲区指针)消息邮箱是一种通信机制,它能使任务或中断服务向另一个任务发送一个指针型的变量,这个指针指向一个包含指定“消息”的数据结构。消息邮箱发送的不是消息本身,而是消息的地址指针。

第七章任务间的同步与通讯消息邮箱什么是消息邮箱消息邮箱的数据结构消息邮箱相关函数消息邮箱的数据结构typedefstruct{INT8UOSEventType;/*事件类型*/INT8UOSEventGrp;/*等待任务所在的组*/INT16UOSEventCnt;

/*计数器(当事件是信号量时)*/void*OSEventPtr;

/*指向消息或者消息队列的指针*/INT8UOSEventTbl[OS_EVENT_TBL_SIZE];

/*等待任务列表*/}OS_EVENT;第七章任务间的同步与通讯消息邮箱什么是消息邮箱消息邮箱的数据结构消息邮箱相关函数

消息邮箱函数一览表消息邮箱函数所属文件是OS_MBOX.COS_CFG.H文件中消息邮箱的配置常数一览表任务、中断服务子程序与消息邮箱之间的关系消息邮箱的操作OSMboxCreate()OSMboxDel()OSMboxPend()OSMboxPost()OSMboxPostOpt()OSMboxAccept()OSMboxQuery()建立一个邮箱,OSMboxCreate()函数OSMboxCreate()的原型如下:OS_EVENT*OSMboxCreate(void*msg);参数:msg参数用来初始化建立的消息邮箱。如果该指针不为空,建立的消息邮箱将含有消息。返回值:指向分配给所建立的消息邮箱的事件控制块的指针。如果没有可用的事件控制块,返回空指针。邮箱作用:通知一个消息(此时,msg为空)作为二值信号量范例OS_EVENT*RxMbox;

/*定义消息邮箱指针 */

voidmain(void){

.

OSInit();

RxMbox=OSMboxCreate((void*)0);

/*建立消息邮箱,初始值为空 */

OSStart();

/*启动多任务内核 */

}OSMboxCreate()函数返回前,事件控制块的内容邮箱特点消息邮箱的内容是一个指向消息的指针,指针指向的内容就是消息邮箱只能接收和发送一条消息,当邮箱满时,将丢弃新消息,保持旧消息当邮箱为满是,邮箱里存放指向消息的指针;当邮箱为空时,邮箱指针为NULL消息邮箱的操作OSMboxCreate()OSMboxDel()OSMboxPend()OSMboxPost()OSMboxPostOpt()OSMboxAccept()OSMboxQuery()删除消息邮箱——OSMboxDel()函数原型如下:

OS_EVENT*OSMboxDel(OS_EVENT*pevent,INT8Uopt,INT8U*err)

该函数用于删除消息邮箱,所属文件为OS_MBOX.C,调用者为任务,开关量为OS_MBOX_EN和OS_MBOX_DEL_EN。参数:

(1) pevent:指向消息邮箱的指针,该指针是消息邮箱建立时,返回给应用程序的指针。(2) opt:用于定义消息邮箱删除条件的选项。可供选择的值如下:

①OS_DEL_NO_PEND:选择只能是没有任何任务在等待该消息邮箱的消息时,才能删除消息邮箱。

②OS_DEL_ALWAYS:不管有没有任务在等待消息邮箱的消息,都立即删除消息邮箱。删除后,所有等待消息邮箱消息的任务都立即进入就绪状态。

(3) err:指向出错代码的指针。

① OS_NO_ERR:调用成功,表明邮箱已被删除。② OS_ERR_DEL_ISR:试图在中断服务子程序中删除邮箱。

③ OS_ERR_INVALID_OPT:无效的opt参数。

④ OS_ERR_EVENT_TYPE:pevent不是指向邮箱的指针。

⑤ OS_ERR_PEVENT_NULL:没有可以使用的OS_EVENT数据结构返回值

OSMboxDel()函数的返回值为如下内容之一:

(1)返回空指针NULL,表示消息邮箱已被删除。

(2)返回pevent,表示消息邮箱没有删除,在这种情况下,应该进一步查看出错代码,找到出错原因。.范例。。。。OS_EVENT*RxMbox

/*定义消息邮箱指针 */

voidtask(void*pdata){

INT8U*err;

pdata=pdata;

for(;;){RxMbox=OSMboxDel(RxMbox,OS_DEL_ALWAYS,&err);

应用程序;

}

}消息邮箱的操作OSMboxCreate()OSMboxDel()OSMboxPend()OSMboxPost()OSMboxPostOpt()OSMboxAccept()OSMboxQuery()等待邮箱中的消息,OSMboxPend()函数OSMboxPend()的原型如下:void*OSMboxPend(OS_EVNNT*pevent,INT16Utimeout,int8u*err);参数:pevent:是指向即将接受消息的消息邮箱的指针。timeout:允许一个任务在经过了指定数目的时钟节拍后还没有得到需要的消息时恢复运行。err:是指向包含错误码的变量的指针。OSMboxPend()函数返回的错误码可能为下述几种:OS_NO_ERR:消息被正确的接收。OS_TIMEOUT:消息没有在指定的周期数内送到。OS_ERR_PEND_ISR:从中断调用该函数。虽然规定了不允许从中断调用该函数,但μC/OS-Ⅱ仍然包含了检测这种情况的功能。OS_ERR_EVENT_TYPE:pevent不是指向消息邮箱的指针。返回值:OSMboxPend()函数返回接收的消息并将*err置为OS_NO_ERR。如果没有在指定数目的时钟节拍内接受到需要的消息,OSMboxPend()函数返回空指针并且将*err设置为OS_TIMEOUT。范例OS_EVENT*RxMbox;

/*定义消息邮箱指针 */

voidTask(void*pdata){

INT8Uerr;

void *msg;

pdata=pdata;

for(;;){

应用程序代码;

msg=OSMboxPend(RxMbox,10,&err);/*返回消息指针,据该指针可获消息 */if(err==OS_NO_ERR){

消息正确接收的应用程序代码;

}else{

在指定时间内没有接收到消息的应用程序代码;

}

应用程序代码;

}

}等待消息说明消息邮箱已经存在需要的消息,那么该消息被返回给OSMboxPend()的调用者,消息邮箱中清除该消息。如果调用OSMboxPend()函数时消息邮箱中没有需要的消息,OSMboxPend()函数挂起当前任务直到得到需要的消息或超出定义等待超时的时间。如果同时有多个任务等待同一个消息,μC/OS-Ⅱ默认最高优先级的任务取得消息并且任务恢复执行。一个由OSTaskSuspend()函数挂起的任务也可以接受消息,但这个任务将一直保持挂起状态直到通过调用OSTaskResume()函数恢复任务的运行。消息邮箱的操作OSMboxCreate()OSMboxDel()OSMboxPend()OSMboxPost()OSMboxPostOpt()OSMboxAccept()OSMboxQuery()向邮箱发送一则消息,OSMboxPost()

函数OSMboxPost()的原型如下:INT8UOSMboxPost(OS_EVENT*pevent,void*msg);参数:pevent:指向即将接受消息的消息邮箱的指针。该指针的值在建立该消息邮箱时可以得到。(参考OSMboxCreate()函数)。msg:即将实际发送给任务的消息。消息是一个指针长度的变量,在不同的程序中消息的使用也可能不同。不允许传递一个空指针,因为这意味着消息邮箱为空。返回值:OS_NO_ERR:消息成功的放到消息邮箱中。OS_MBOX_FULL:消息邮箱已经包含了其他消息,不空。OS_ERR_EVENT_TYPE:pevent不是指向消息邮箱的指针。调用者是任务或中断,开关量是OS_MBOX_EN和OS_MBOX_POST_EN发送消息说明如果消息邮箱中已经存在消息,返回错误码说明消息邮箱已满。如果有多个任务在等待消息邮箱的消息,最高优先级的任务将得到这个消息。如果等待消息的任务优先级比发送消息的任务优先级高,那么高优先级的任务将得到消息而恢复执行,也就是说,发生了一次任务切换。消息邮箱的操作OSMboxCreate()OSMboxDel()OSMboxPend()OSMboxPost()OSMboxPostOpt()OSMboxAccept()OSMboxQuery()广播发送消息到消息邮箱中—OSMboxPostOpt()

函数原型如下:

INT8UOSMboxPostOpt(OS_EVENT*pevent,void*msg,INT8Uopt)

OSMboxPostOpt()函数与OSMboxPost()函数比较两者工作方式相同,都用指针传递消息;不同的是OSMboxPost()只能给一个任务发送消息,而OSMboxPostOpt()允许将消息广播给所有的等待消息邮箱消息的任务,且可以仿真OSMboxPost()。OSMboxPostOpt()函数的开关量是OS_MBOX_EN和OS_MBOX_POST_OPT_EN,调用者可以是任务,也可以是中断。在广播方式下,函数的执行时间取决于等待消息邮箱消息的任务数量。OSMboxpostOpt()函数有如下三个参数:

(1) pevent:指向即将接收消息的消息邮箱的指针。该指针的值可以在建立该消息邮箱时得到。

(2) msg:即将发送给任务的消息。消息是以指针表示的某种数据类型的变量,消息的数据类型可能会因具体应用的不同而有所差异。不允许传递空指针,因为这样意味着消息邮箱为空。

(3) opt:定义发送消息方式的选项。它有两种形式:

①若opt=OS_POST_OPT_NONE,则定义消息只发给等待消息的任务中优先级最高的任务;

②若opt=OS_POST_OPT_BROADCAST,则让所有等待邮箱消息的任务都得到消息。

返回值

(1) OS_NO_ERR:调用成功,消息已经发出。

(2) OS_MBOX_FULL:邮箱中已经有消息。

(3) OS_ERR_EVENT_TYPE:pevent不是指向邮箱的指针。

(4) OS_ERR_PEVENT_NULL:pevent是空指针。

(5) OS_ERR_POST_NULL_PTR:试图发送空指针。根据规则,空指针无效。消息邮箱的操作OSMboxCreate()OSMboxDel()OSMboxPend()OSMboxPost()OSMboxPostOpt()OSMboxAccept()OSMboxQuery()无等待地从邮箱中得到一则消息,OSMboxAccept()

OSMboxAccept()的函数原型如下:void*OSMboxAccept(OS_EVENT*pevent);参数:pevent:是指向需要查看的消息邮箱的指针。返回值:如果消息已经到达,返回指向该消息的指针;如果消息邮箱没有消息,返回空指针。范例OS_EVENT*ComMbox;

voidTask(void*pdata)

{

void*msg;

pdata=pdata;

for(;;){

msg=OSMboxAccept(ComMbox);

/*检查消息邮箱是否有消息 if(msg!=(void*)0){

处理消息;

}else{

处理无消息的情况;

}

.

}

}消息邮箱的操作OSMboxCreate()OSMboxDel()OSMboxPend()OSMboxPost()OSMboxPostOpt()OSMboxAccept()OSMboxQuery()查询一个邮箱的状态,OSMboxQuery()

OSMboxQuery()的函数原型如下:INT8UOSMboxQuery(OS_EVENT*pevent,OS_MBOX_DATA*pdata);参数:pevent:指向即将接收消息的消息邮箱的指针。pdata:指向OS_MBOX_DATA数据结构的指针,该数据结构包含下述成员:Void*OSMsg; /*消息邮箱中消息的复制*/INT8UOSEventTbl[OS_EVENT_TBL_SIZE]; /*消息邮箱等待队列的复制*/INT8UOSEventGrp;返回值:OS_NO_ERR:调用成功。OS_ERR_EVENT_TYPE:pevent不是指向消息邮箱的指针。范例

OS_EVENT*RxMbox;

/*定义指针 /

voidTask(void*pdata){

OS_MBOXDATA cbox_data;

INT8U err;

pdata=pdata;

for(;;){err=OSMboxQuery(RxMbox,&cbox_data);

if(err==OS_NO_ERR){

./*如果cbox_data.OSMsg为非空指针,说明消息邮箱非空 */

}

.

}

}

举例本例中有两个任务:MyTask和YouTask。任务YouTask要向任务MyTask发送消息/************************Test*****************************/#include“includes.h”#defineTASK_STK_SIZE512//任务堆栈长度OS_STKStartTaskStk[TASK_STK_SIZE];//定义任务堆栈区OS_STKMyTaskStk[TASK_STK_SIZE];//定义任务堆栈区OS_STKYouTaskStk[TASK_STK_SIZE];//定义任务堆栈区INT16Skey;//用于退出µc/os-II的键INT8Uy1=0,y2=0;//字符显示位置

OS_EVENT*msg;

//消息邮箱voidStartTask(void*data);//声明起始任务voidMyTask(void*data);//声明任务voidYouTask(void*data);//声明任务/********************主函数*******************/voidmain(void){OSInit();//初始化µc/os-IIPC_DOSSaveReturn();//保存DOS环境

PC_VectSet(uCOS,OSCtxSw);//安装µc/os-II中断msg=OSMboxCreate((void*)0);

OSTaskCreate(StartTask,//创建任务StartTask(void*)0,//给任务传递参数

&StartTaskStk[TASK_STK_SIZE-1],//设置任务堆栈栈顶指针

0);//使任务的优先级别为0OSStart();//启动多任务管理}/****************任务StartTask**********************/voidStartTask(void*pdata){#ifOS_CRITICAL_METHOD==3OS_CPU_SRcpu_sr;#endifpdata=pdata;OS_ENTER_CRITICAL();PC_VectSet(0x08,OSTickISR);//安装时钟中断向量

PC_SetTickRate(OS_TICKS_PER_SEC);//设置µc/os-II时钟频率

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

OSTaskCreate(MyTask,(void*)0,&MyTaskStk[TASK_STK_SIZE-1],

1);OSTaskCreate(YouTask,(void*)0,&YouTaskStk[TASK_STK_SIZE-1],

2);for(;;){//如果按下ESC键,则退出µc/os-IIif(PC_GetKey(&key)==TRUE){if(key==0x1B){PC_DOSReturn();}}OSTimeDlyHMSM(0,0,3,0);//等待3s}}/*****************任务MyTask***********************/voidMyTask(void*pdata){#ifOS_CRITICAL_METHOD==3OS_CPU_SRcpu_sr;#endifpdata=pdata;void*msg_My;INT8Uerr;for(;;){PC_DispStr(5,++y1,“MyTask”,DISP_BGND_BLACK+DISP_FGND_WHITE);

msg_My=OSMboxPend(msg,0,err);if(err==OS_NO_ERR){PC_DispStr(15,y1,msg_My,DISP_BGND_BLACK+DISP_FGND_WHITE);}OSTimeDlyHMSM(0,0,1,0);//等待1s}/***********************任务YouTask*******************/voidYouTask(void*pdata){#ifOS_CRITICAL_METHOD==3OS_CPU_SRcpu_sr;#endifchar*s=“YouTask发送的消息”;//定义消息INT8Uerr;

pdata=pdata;for(;;){PC_DispStr(40,++y2,“YouTask”,DISP_BGND_BLACK+DISP_FGND_WHITE);

err=OSMboxPost(msg,(void*)s);OSTimeDlyHMSM(0,0,2,0);//等待2s}}/********************************END***************************************/再看一个例子设计一个应用程序,该程序有两个任务MyTask和YouTask。在任务MyTask中用一个变量Times记录任务MyTask的运行次数,并将其作为消息用邮箱Str_Box发给任务YouTask且由任务YouTask显示出来。/***********************Test*****************************/#include“includes.h”#defineTASK_STK_SIZE512//任务堆栈长度OS_STKStartTaskStk[TASK_STK_SIZE];//定义任务堆栈区OS_STKMyTaskStk[TASK_STK_SIZE];//定义任务堆栈区OS_STKYouTaskStk[TASK_STK_SIZE];//定义任务堆栈区INT16Skey;//用于退出的键char*s;char*ss;INT8Uerr;INT8Uy=0;//字符显示位置INT32UTimes=0;OS_EVENT*Str_Box;

//定义事件控制块指针voidStartTask(void*data);//声明起始任务voidMyTask(void*data);//声明任务voidYouTask(void*data);//声明任务/*******************主函数****************************/voidmain(void){OSInit();//初始化µc/os-II……

Str_Box=OSMboxCreate((void*)0);

//创建消息邮箱OSTaskCreate(StartTask,//创建任务StartTask(void*)0,//给任务传递参数

&StartTaskStk[TASK_STK_SIZE-1],//设置任务堆栈栈顶指针

0);//使任务的优先级别为0OSStart();//启动多任务管理}/******************任务StartTask*****************************/voidStartTask(void*pdata){……OSStatInit();//初始化统计任务

OSTaskCreate(MyTask,//创建任务MyTask(void*)0,//给任务传递参数

&MyTaskStk[TASK_STK_SIZE-1],//设置任务堆栈栈顶指针

3);//使任务的优先级别为3OSTaskCreate(YouTask,//创建任务YouTask(void*)0,//给任务传递参数

&YouTaskStk[TASK_STK_SIZE-1],//设置任务堆栈栈顶指针

4);//使任务的优先级别为4for(;;){//如果按下ESC键,则退出µc/os-II

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论