STM32跑ucosII系统之信号和消息邮箱介绍

发布者:算法之手最新更新时间:2018-10-06 来源: eefocus关键字:STM32  ucosII系统  信号  消息邮箱 手机看文章 扫描二维码
随时随地手机看文章

写在前面:


“信号”可以单纯的理解为一个信号量(trig触发用),在任务1中传递一个信号给任务2,那么,任务2接收到这个信号,会往下执行。


“消息邮箱”也可以理解为一个信号量,只不过这个消息可以携带内容:比如变量的值。


一、事件——任务之间通信的中间环节


任务间的同步依赖于任务间的通信。 在 UCOSII 中,是使用信号量、邮箱(消息邮箱)和消息队列这些被称作事件的中间环节来实现任务之间的通信的



.       发送事件            请求事件

任务1 ------------> 事件 ------------> 任务2

任务 1 是发信方,任务 2 是收信方。任务 1 负责把信息发送到事件上,这项 

操作叫做发送事件。任务 2 通过读取事件操作对事件进行查询:如果有信息则读取,否则等待。读事件操作叫做请求事件


为了把描述事件的数据结构统一起来, UCOSII 使用叫做事件控制块(ECB)的数据结构来描述诸如信号量、邮箱(消息邮箱)和消息队列这些事件。事件控制块中包含包括等待任务表在内的所有有关事件的数据,事件控制块结构体定义如下:


typedef struct

{

INT8U OSEventType; //事件的类型

INT16U OSEventCnt; //信号量计数器

void *OSEventPtr; //消息或消息队列的指针

INT8U OSEventGrp; //等待事件的任务组

INT8U OSEventTbl[OS_EVENT_TBL_SIZE];//任务等待表

#if OS_EVENT_NAME_EN > 0u

INT8U *OSEventName; //事件名

#endif

} OS_EVENT;

二、 信号量


信号量是一类事件. 使用信号量的最初目的,是为了给共享资源设立一个标志,该标志表示该共享资源的占用情况。这样,当一个任务在访问共享资源之前,就可以先对这个标志进行查询,从而在了解资源被占用的情况之后,再来决定自己的行为。


信号量可以分为两种:一种是二值型信号量(只能一个人占用),另外一种是 N 值信号量(可以有同时多个人使用)


信号量相关函数


1. 创建信号量函数


OS_EVENT *OSSemCreate (INT16U cnt);

该函数返回值为已创建的信号量的指针


cnt 是信号量计数器的初始值


2. 请求信号量函数


void OSSemPend ( OS_EVENT *pevent, INT16U timeout, INT8U *err);

pevent 是被请求信号量的指针


timeout 为等待时限


err 为错误信息


为防止任务因得不到信号量而处于长期的等待状态,函数 OSSemPend 允许用参数timeout 设置一个等待时间的限制,当任务等待的时间超过 timeout 时可以结束等待状态而进入就绪状态。如果参数 timeout 被设置为 0,则表明任务的等待时间为无限长


3. 发送信号量函数


任务获得信号量,并在访问共享资源结束以后,必须要释放信号量,释放信号量也叫做发送信号量,发送信号通过 OSSemPost 函数实现 。 OSSemPost 函数在对信号量的计数器操作之前,首先要检查是否还有等待该信号量的任务。如果没有,就把信号量计数器OSEventCnt 加一;如果有,则调用调度器 OS_Sched( )去运行等待任务中优先级别最高的 

任务。函数 OSSemPost 的原型为:


INT8U OSSemPost(OS_EVENT *pevent);

pevent 为信号量指针


该函数在调用成功后, 返回值为 OS_ON_ERR,否则会 

根据具体错误返回 OS_ERR_EVENT_TYPE、 OS_SEM_OVF


4. 删除信号量函数


OS_EVENT *OSSemDel (OS_EVENT *pevent,INT8U opt, INT8U *err);

1

pevent 为要删除的信号量指针


opt 为删除条件选项


err 为错误信息


三、 邮箱


在多任务操作系统中,常常需要在任务与任务之间通过传递一个数据(这种数据叫做“消息”)的方式来进行通信。为了达到这个目的,可以在内存中创建一个存储空间作为该数据的缓冲区。如果把这个缓冲区称之为消息缓冲区,这样在任务间传递数据(消息)的最简单办法就是传递消息缓冲区的指针。我们把用来传递消息缓冲区指针的数据结构叫做邮箱(消息邮箱)。


UCOSII 中,我们通过事件控制块的 OSEventPtr 来传递消息缓冲区指针, 同时使事件控制块的成员 OSEventType 为常数 OS_EVENT_TYPE_MBOX,则该事件控制块就叫做消息邮箱。


消息邮箱相关的函数


1. 创建邮箱函数


OS_EVENT *OSMboxCreate (void *msg);

msg 为消息的指针


函数的返回值为消息邮箱的指针


调用函数 OSMboxCreate 需先定义 msg 的初始值。在一般的情况下,这个初始值为 

NULL;但也可以事先定义一个邮箱,然后把这个邮箱的指针作为参数传递到函数 

OSMboxCreate 中,使之一开始就指向一个邮箱。


2. 向邮箱发送消息函数


INT8U OSMboxPost (OS_EVENT *pevent,void *msg);

pevent 为消息邮箱的指针


msg 为消息指针


3. 请求邮箱函数


当一个任务请求邮箱时需要调用函数 OSMboxPend,这个函数的主要作用就是查看邮箱指针 OSEventPtr 是否为 NULL,如果不是 NULL 就把邮箱中的消息指针返回给调用函数的任务,同时用 OS_NO_ERR 通过函数的参数 err 通知任务获取消息成功; 如果邮箱指针OSEventPtr 是 NULL,则使任务进入等待状态,并引发一次任务调度。


void *OSMboxPend (OS_EVENT *pevent, INT16U timeout, INT8U *err);

pevent 为请求邮箱指针


timeout 为等待时限


err 为错误信息


4. 查询邮箱状态函数


INT8U OSMboxQuery(OS_EVENT *pevent,OS_MBOX_DATA *pdata);

pevent 为消息邮箱指针


pdata 为存放邮箱信息的结构


5. 删除邮箱函数


在邮箱不再使用的时候,我们可以通过调用函数 OSMboxDel 来删除一个邮箱,该函 

数原型为:


OS_EVENT *OSMboxDel(OS_EVENT *pevent,INT8U opt,INT8U *err);

pevent 为消息邮箱指针


opt 为删除选项


err 为错误信息


# 四、STM32使用UCOSII


使用正在原子公司的STM32mini板,在ucosii上使用信号量和邮箱,通过key来控制灯的亮灭


键盘扫描任务——->键盘键邮箱——->key值判断任务——->key值信号量——>亮灯任务


key0按下led0闪 key1按下led1闪 key_up按下led0和led1都闪


/////////////////////////UCOSII任务设置///////////////////////////////////

//START 任务

//设置任务优先级

#define START_TASK_PRIO                 10 //开始任务的优先级设置为最低

//设置任务堆栈大小

#define START_STK_SIZE                  64

//任务堆栈  

OS_STK START_TASK_STK[START_STK_SIZE];

//任务函数

void start_task(void *pdata);   


//LED0任务

//设置任务优先级

#define LED0_TASK_PRIO                  7 

//设置任务堆栈大小

#define LED0_STK_SIZE                   64

//任务堆栈  

OS_STK LED0_TASK_STK[LED0_STK_SIZE];

//任务函数

void led0_task(void *pdata);



//LED1任务

//设置任务优先级

#define LED1_TASK_PRIO                  6 

//设置任务堆栈大小

#define LED1_STK_SIZE                   64

//任务堆栈

OS_STK LED1_TASK_STK[LED1_STK_SIZE];

//任务函数

void led1_task(void *pdata);



//key传递函数

//设置任务优先级

#define KEY_TASK_PRIO                   5

//设置任务堆栈大小

#define KEY_STK_SIZE                        64

//任务堆栈

OS_STK KEY_TASK_STK[KEY_STK_SIZE];

//任务函数

void key_task(void *pdata);



//按键扫描任务

//设置任务优先级

#define SCAN_TASK_PRIO                  4

//设置任务堆栈大小

#define SCAN_STK_SIZE                   64

//任务堆栈

OS_STK SCAN_TASK_STK[SCAN_STK_SIZE];

//任务函数

void scan_task(void *pdata);    


///////////////////////////////////////////////////////////////////////////////////////////////

OS_EVENT * msg_key; //按键邮箱时间块指针

OS_EVENT * sem_led0; //LED0信号量指针

OS_EVENT * sem_led1; //LED1信号量指针


 int main(void)

 {  

    delay_init();            //延时函数初始化  

  NVIC_Configuration();  

    LED_Init();         //初始化与LED连接的硬件接口

    KEY_Init();

    OSInit();   

    OSTaskCreate(start_task,(void *)0,(OS_STK *)&START_TASK_STK[START_STK_SIZE-1],START_TASK_PRIO );//创建起始任务

    OSStart();  

 }



//开始任务

void start_task(void *pdata)

{

  OS_CPU_SR cpu_sr=0;

    pdata = pdata; 

    msg_key=OSMboxCreate((void *)0);//创建消息邮箱

    sem_led0=OSSemCreate(0);

    sem_led1=OSSemCreate(0);//创建信号量

    OS_ENTER_CRITICAL();            //进入临界区(无法被中断打断)    

    OSTaskCreate(led0_task,(void *)0,(OS_STK*)&LED0_TASK_STK[LED0_STK_SIZE-1],LED0_TASK_PRIO);                         

    OSTaskCreate(led1_task,(void *)0,(OS_STK*)&LED1_TASK_STK[LED1_STK_SIZE-1],LED1_TASK_PRIO);

    OSTaskCreate(key_task,(void *)0,(OS_STK*)&KEY_TASK_STK[KEY_STK_SIZE-1],KEY_TASK_PRIO);

    OSTaskCreate(scan_task,(void *)0,(OS_STK*)&SCAN_TASK_STK[SCAN_STK_SIZE-1],SCAN_TASK_PRIO);

    OSTaskSuspend(START_TASK_PRIO); //挂起起始任务.

    OS_EXIT_CRITICAL();             //退出临界区(可以被中断打断)

}


//LED0任务

void led0_task(void *pdata)

{

    u8 err; 

    while(1)

    {

        OSSemPend(sem_led0,0,&err);

        LED0=0;

        delay_ms(500);

        LED0=1;

        delay_ms(500);

    };

}


//LED1任务

void led1_task(void *pdata)

{   

    u8 err;

    while(1)

    {

        OSSemPend(sem_led1,0,&err);

        LED1=0;

        delay_ms(500);

        LED1=1;

        delay_ms(500);

    }

}



void key_task(void *pdata)

{

    int key=0;

    u8 err;

    while(1)

    {

        key=(int)OSMboxPend(msg_key,10,&err);

        switch(key)

        {

            case KEY0_PRES://发送信号量0

            OSSemPost(sem_led0);

            break;

            case KEY1_PRES://发送信号量1

            OSSemPost(sem_led1);

            break;

            case WKUP_PRES:

            OSSemPost(sem_led0);

            OSSemPost(sem_led1);

            break;

        }

    }

}


//按键扫描任务

void scan_task(void *pdata)

{

    u8 key;

    while(1)

    {

        key=KEY_Scan(0);

        if(key)OSMboxPost(msg_key,(void*)key);//发送消息

        delay_ms(10);

    }

}


关键字:STM32  ucosII系统  信号  消息邮箱 引用地址:STM32跑ucosII系统之信号和消息邮箱介绍

上一篇:stm32单片机检测12V电路
下一篇:STM32启动文件分析——startup_stm32f10x_hd.s

推荐阅读最新更新时间:2024-03-16 16:15

将放大器放在探头如何降低探头和线缆对信号造成的损耗
“为什么示波器厂商把放大器放在探头尖端,而不在示波器内?”将放大器放在探头可最大限度地降低探头和线缆对信号造成的损耗,但这是如何实现的?要了解如何及为什么,我们需要对探头和输入阻抗有一基本了解。 示波器探头将示波器的输入连接到要测量的电压节点。传统上,常用的探头分为三种类型:高阻抗无源探头、低容抗传输线探头和有源探头。 最常见的探头类型是高阻抗无源探头。图1是其简化示意图。该探头使用经补偿的分压器(电阻和电容匹配的分压器)以驱动探头线缆和示波器输入电容。这些探头有500MHz额定带宽,但你应考虑由输入电容所带来的限制。 图1:采用电容和电阻匹配的分压器的高阻无源探头。 示波器的输入电容可能在15~25pF之间。同轴电缆
[测试测量]
将放大器放在探头如何降低探头和线缆对<font color='red'>信号</font>造成的损耗
STM32—IWDG独立看门狗的使用
STM32---IWDG独立看门狗的使用 独立看门狗(IWDG)由专用的40kHz的低速时钟驱动, 即使主时钟发生故障它也仍然有效。 窗口看门狗由从APB1时钟分频后得到的时钟驱动, 通过可配置的时间窗口来检测应用程序非正常的过迟或过早的操作。 IWDG最适合应用于那些需要看门狗作为一个在主程序之外, 能够完全独立工作,并且对时间精度要求较低的场合。 WWDG最适合那些要求看门狗在精确计时窗口起作用的应用程序。 IWDG主要性能 ●自由运行的递减计数器 ●时钟由独立的RC振荡器提供(可在停止和待机模式下工作) ●看门狗被激活后,则在计数器计数至0x000时产生复位 在键寄存器(IWD
[单片机]
基于MDK编译器 STM32与12864液晶显示程序 和电路连接
里附上的是主程序部分和电路连接 这里用的12864液晶是5v的,电路链接部分 RS PE2 RW PE4 EN PE6 15口PSB串并 PE3 #include stm32f10x.h GPIO_InitTypeDef GPIO_InitStructure; #define RS_SET GPIO_SetBits(GPIOE, GPIO_Pin_2) #define RS_CLR GPIO_ResetBits(GPIOE, GPIO_Pin_2) #define RW_SET GPIO_SetBits(GPIOE, GPIO_Pin_4) #define RW_CLR GPIO_Re
[单片机]
STM32红外串口接收
1.NEC协议 现有的红外遥控包括两种方式: PWM(脉冲宽度调制)和PPM(脉冲位置调制)。 两种形式编码的代表分别为NEC 和 PHILIPS的RC-5、RC-6以及将来的RC-7。 PWM(脉冲宽度调制):以发射红外载波的占空比代表 0”和 1 。为了节省能量,一般情况下,发射红外载波的时间固定,通过改变不发射载波的时间来改变占空比。例如常用的电视遥控器,TOSHIBA的TC9012,其引导码为载波发射4. 5ms,不发射4.5ms,其 0 为载波发射0.56ms,不发射0.565ms,其 1 为载波发射0.56ms,不发射1.69ms。 PPM(脉冲位置调制)∶以发射载波的位置表示 0 和 1 。从发射载波到不发射
[单片机]
<font color='red'>STM32</font>红外串口接收
天线分析仪的应用广泛 可当信号发生器使用
天线分析仪是用来测量天线各顼特性参数的仪器,有“天线万用表”的美称。随着电子制造业的发展,芯片集成化程度越来越高,,功能电路的设计制造要求也越来越高,业余无线电爱好者自己JY完整的高性能收发信机已变得困难重重,而制作天线则相对简单。无线电发展至今,收发信机的电子线路巳经更新换代多次,而基本的开线形式则一直继承、保留,至今依然广泛应用在业余电台和专业电台上。 传统基础无源天线的制作不需要半导体电子线路,最简单的天线甚至只是一些特定长度的导线(普通电线)的组合,技术要求不高,门槛很低,即便是专业知识不多的新手也很容易上手制作,并可以曲浅入深地逐渐试制一些比蛟复杂的天线。在天线的制作过程中,爱好者会体验到动手的快乐和享受最终成功的的
[测试测量]
天线分析仪的应用广泛 可当<font color='red'>信号</font>发生器使用
基于地面高清机顶盒的PVR系统设计与开发
0 引言   2006年8月18日,国家标准化管理委员会正式发布了我国具有自主知识产权的《数字电视地面广播传输系统帧结构、信道编码和调制》标准(简称“中国数字电视地面传输标准”)。该标准支持高清晰度电视(HDTV)、标准清晰度电视(SDTV)和多媒体数据广播(MMDB)等多种业务,满足大范围固定覆盖和移动接收的需要。地面高清数字广播系统于2008年7月投入使用,已成功进行了北京奥运会的全程直播,并将直播2010年上海世博会。针对地面高清市场的各类产品及技术应运而生,而PVR功能则是相关产品技术中的难点和亮点。   PVR的全称是Personal Video Recorder(个人视频录像机),但其实际功能超出了名称的表述。PV
[家用电子]
基于地面高清机顶盒的PVR<font color='red'>系统</font>设计与开发
TI全新振荡器系列具备业内最低抖动,可优化信号完整性
2015年12月23日,北京讯---日前,德州仪器宣布了一个完全可编程、引脚可选且频率固定的7mmx5mm差分振荡器系列。这一系列的器件可提供90飞秒 (fs) 的业内最低抖动,从而使得设计人员能够优化性能关键应用中的信号完整性,并降低数据传输误差。LMK61xx系列可实现轻松定制、频率裕量,并且在一个器件内支持针对现场可编程门阵列 (FPGA) 、模数转换器 (ADC) 、数模转换器 (DAC) 和高速串行链路的多个时钟频率。 LMK61xx振荡器的主要特点与优势: 业内最低抖动:在12kHz至20MHz频率范围内,均方根 (RMS) 抖动典型值为90fs, 这比市场同类产品减少一半,从而使得设计人员可以提升
[嵌入式]
TI全新振荡器系列具备业内最低抖动,可优化<font color='red'>信号</font>完整性
STM32的硬件I2C设计有BUG
坊间一直流传着一个传说~STM32的硬件I2C设计有BUG,最好不要用,用软件I2C比较靠谱。长久以来,为了不必要的麻烦,我也一直没有用过硬件I2C,主要是软件I2C也比较方便,基本上任意端口都可以用。 最近画了块板子,正好用到了I2C,就顺便来测试一下硬件I2C是不是真的像有些人说的不好用。 测试硬件:STM32F407VET6+AT24C64测试软件:STM32CubeMX v6.1.1HAL库:STM32CubeF4 Firmware Package V1.25.2 STM32CubeMX配置 使用STM32CubeMX配置很方便,时钟等基础配置不再详细介绍,直接看I2C配置如下: 这里的速度模式选择为标准模式,
[单片机]
<font color='red'>STM32</font>的硬件I2C设计有BUG
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved