嵌入式技术论坛
直播中

CDCNKA

8年用户 1231经验值
擅长:385288
私信 关注
[经验]

在RT-Thread操作系统中的消息队列实验是如何去完成的

消息队列实验
     消息队列实验是在RT-Thread中创建了3个线程,2个是发送消息线程,1个是获取消息线程。3个线程独立运行,key1按下时key1线程发送消息1;key2按下时key2线程连接发送消息2共20次,key2线程发送频率大于接收并打印输出的速率。Key1、key2线程假如发送消息不成功,就把返回的错误情代码在串口打印出来。接收线程用于获取消息线程,在消息队列没有消息之前一直等待消息,一旦获取到消息就把消息打印在串口调试助手里。
    实验时,先按下key2,由于发送频率大于接收线程的处理速率,造成消息队列溢出;在按下key2后,按下按键key1,可能会成功,也可能因消息队列满时,也出现溢出的错误代码。
     本代码在 野火“开发板RT-Thread 3.0 + STM32 消息队列”例程上修改。原代码是2个线程,这里改为3个线程,并在按下key2时连续发送消息。
*************************************************************************
*                             包含的头文件
*************************************************************************
*/
#include "board.h"
#include "rtthread.h"
/*
*************************************************************************
*                               变量
*************************************************************************
*/
/* 定义线程控制块 */
static rt_thread_t receive_thread = RT_NULL;
static rt_thread_t send_thread = RT_NULL;
/* 定义消息队列控制块 */
static rt_mq_t test_mq = RT_NULL;
/*
*************************************************************************
*                             函数声明
*************************************************************************
*/
static void receive_thread_entry(void* parameter);
static void key1_send_thread_entry(void* parameter);
static void key2_send_thread_entry(void* parameter);
/*
*************************************************************************
*                             main 函数
*************************************************************************
*/
int main(void)
{
    /*
* 开发板硬件初始化,RTT系统初始化已经在main函数之前完成,
* 即在component.c文件中的rtthread_startup()函数中完成了。
* 所以在main函数中,只需要创建线程和启动线程即可。
*/
rt_kprintf("这是一个[野火]-STM32F103-指南者-RTT消息队列实验!
");
  rt_kprintf("按下K1或者K2发送队列消息
");
  rt_kprintf("receive任务接收到消息在串口回显
");
   /* 创建一个消息队列 */
test_mq = rt_mq_create("test_mq",/* 消息队列名字 */
                     4,     /* 消息的最大长度 */
                     10,    /* 消息队列的最大容量 */
                     RT_IPC_FLAG_FIFO);/* 队列模式 FIFO(0x00)*/
  if (test_mq != RT_NULL)
    rt_kprintf("消息队列创建成功!

");
receive_thread =                          /* 线程控制块指针 */
    rt_thread_create( "receive",              /* 线程名字 */
                      receive_thread_entry,   /* 线程入口函数 */
                      RT_NULL,             /* 线程入口函数参数 */
                      512,                 /* 线程栈大小 */
                      3,                   /* 线程的优先级 */
                      20);                 /* 线程时间片 */
    /* 启动线程,开启调度 */
   if (receive_thread != RT_NULL)
        rt_thread_startup(receive_thread);
    else
        return -1;
  send_thread =                          /* 线程控制块指针 */
    rt_thread_create( "key1send",              /* 线程名字 */
                      key1_send_thread_entry,   /* 线程入口函数 */
                      RT_NULL,             /* 线程入口函数参数 */
                      512,                 /* 线程栈大小 */
                      2,                   /* 线程的优先级 */
                      20);                 /* 线程时间片 */
    /* 启动线程,开启调度 */
   if (send_thread != RT_NULL)
        rt_thread_startup(send_thread);
    else
        return -1;
  send_thread =                          /* 线程控制块指针 */
    rt_thread_create( "key2send",              /* 线程名字 */
                      key2_send_thread_entry,   /* 线程入口函数 */
                      RT_NULL,             /* 线程入口函数参数 */
                      512,                 /* 线程栈大小 */
                      2,                   /* 线程的优先级 */
                      20);                 /* 线程时间片 */
    /* 启动线程,开启调度 */
   if (send_thread != RT_NULL)
        rt_thread_startup(send_thread);
    else
        return -1;
}
/*
*************************************************************************
*                             线程定义
*************************************************************************
*/
static void receive_thread_entry(void* parameter)
{
  rt_err_t uwRet = RT_EOK;
  uint32_t r_queue;
  /* 任务都是一个无限循环,不能返回 */
  while (1)
  {
    /* 队列读取(接收),等待时间为一直等待 */
uwRet = rt_mq_recv(test_mq, /* 读取(接收)队列的ID(句柄) */
&r_queue, /* 读取(接收)的数据保存位置 */
sizeof(r_queue), /* 读取(接收)的数据的长度 */
RT_WAITING_FOREVER); /* 等待时间:一直等 */
if(RT_EOK == uwRet)
{
rt_kprintf("本次接收到的数据是:%x
",r_queue);
}
else
{
rt_kprintf("数据接收出错,错误代码: 0x%lx
",uwRet);
}
    rt_thread_delay(100);
  }
}
static void key1_send_thread_entry(void* parameter)
{
  rt_err_t uwRet = RT_EOK;
uint32_t send_data1 = 0x1111;
while (1)
{
if( Key_Scan(KEY1_GPIO_PORT,KEY1_GPIO_PIN) == KEY_ON )/* K1 被按下 */
{
/* 将数据写入(发送)到队列中,等待时间为 0  */
uwRet = rt_mq_send( test_mq, /* 写入(发送)队列的ID(句柄) */
&send_data1, /* 写入(发送)的数据 */
sizeof(send_data1)); /* 数据的长度 */
if(RT_EOK != uwRet)
{
rt_kprintf("KEY1 数据不能发送到消息队列!错误代码: %lx
",uwRet);
}
}
rt_thread_delay(20);
}
}
static void key2_send_thread_entry(void* parameter)
{
  rt_err_t uwRet = RT_EOK;
  uint32_t send_data2 = 0x2222;
while (1)
{
if( Key_Scan(KEY2_GPIO_PORT,KEY2_GPIO_PIN) == KEY_ON )/* K1 被按下 */
{
for(int i=0;i<22;i++)
{
/* 将数据写入(发送)到队列中,等待时间为 0  */
uwRet = rt_mq_send(test_mq, /* 写入(发送)队列的ID(句柄) */
&send_data2, /* 写入(发送)的数据 */
sizeof(send_data2)); /* 数据的长度 */
if(RT_EOK != uwRet)
{
rt_kprintf("KEY2 数据不能发送到消息队列!错误代码: %lx
",uwRet);
}
rt_thread_delay(50);
}
}
rt_thread_delay(20);
}
}
/********************************END OF FILE****************************/
先按下key2,后连续按key1,打印输出结果如图。


原作者:liwuxing

更多回帖

发帖
×
20
完善资料,
赚取积分