xref: /nrf52832-nimble/rt-thread/examples/kernel/messageq_simple.c (revision 104654410c56c573564690304ae786df310c91fc)
1*10465441SEvalZero /*
2*10465441SEvalZero  * 程序清单:消息队列例程
3*10465441SEvalZero  *
4*10465441SEvalZero  * 这个程序会创建3个动态线程,一个线程会从消息队列中收取消息;一个线程会定时给消
5*10465441SEvalZero  * 息队列发送消息;一个线程会定时给消息队列发送紧急消息。
6*10465441SEvalZero  */
7*10465441SEvalZero #include <rtthread.h>
8*10465441SEvalZero #include "tc_comm.h"
9*10465441SEvalZero 
10*10465441SEvalZero /* 指向线程控制块的指针 */
11*10465441SEvalZero static rt_thread_t tid1 = RT_NULL;
12*10465441SEvalZero static rt_thread_t tid2 = RT_NULL;
13*10465441SEvalZero static rt_thread_t tid3 = RT_NULL;
14*10465441SEvalZero 
15*10465441SEvalZero /* 消息队列控制块 */
16*10465441SEvalZero static struct rt_messagequeue mq;
17*10465441SEvalZero /* 消息队列中用到的放置消息的内存池 */
18*10465441SEvalZero static char msg_pool[2048];
19*10465441SEvalZero 
20*10465441SEvalZero /* 线程1入口函数 */
thread1_entry(void * parameter)21*10465441SEvalZero static void thread1_entry(void* parameter)
22*10465441SEvalZero {
23*10465441SEvalZero     char buf[128];
24*10465441SEvalZero 
25*10465441SEvalZero     while (1)
26*10465441SEvalZero     {
27*10465441SEvalZero         rt_memset(&buf[0], 0, sizeof(buf));
28*10465441SEvalZero 
29*10465441SEvalZero         /* 从消息队列中接收消息 */
30*10465441SEvalZero         if (rt_mq_recv(&mq, &buf[0], sizeof(buf), RT_WAITING_FOREVER) == RT_EOK)
31*10465441SEvalZero         {
32*10465441SEvalZero             rt_kprintf("thread1: recv msg from message queue, the content:%s\n", buf);
33*10465441SEvalZero         }
34*10465441SEvalZero 
35*10465441SEvalZero         /* 延迟10个OS Tick */
36*10465441SEvalZero         rt_thread_delay(10);
37*10465441SEvalZero     }
38*10465441SEvalZero }
39*10465441SEvalZero 
40*10465441SEvalZero /* 线程2入口函数 */
thread2_entry(void * parameter)41*10465441SEvalZero static void thread2_entry(void* parameter)
42*10465441SEvalZero {
43*10465441SEvalZero     int i, result;
44*10465441SEvalZero     char buf[] = "this is message No.x";
45*10465441SEvalZero 
46*10465441SEvalZero     while (1)
47*10465441SEvalZero     {
48*10465441SEvalZero         for (i = 0; i < 10; i++)
49*10465441SEvalZero         {
50*10465441SEvalZero             buf[sizeof(buf) - 2] = '0' + i;
51*10465441SEvalZero 
52*10465441SEvalZero             rt_kprintf("thread2: send message - %s\n", buf);
53*10465441SEvalZero             /* 发送消息到消息队列中 */
54*10465441SEvalZero             result = rt_mq_send(&mq, &buf[0], sizeof(buf));
55*10465441SEvalZero             if ( result == -RT_EFULL)
56*10465441SEvalZero             {
57*10465441SEvalZero                 /* 消息队列满, 延迟1s时间 */
58*10465441SEvalZero                 rt_kprintf("message queue full, delay 1s\n");
59*10465441SEvalZero                 rt_thread_delay(100);
60*10465441SEvalZero             }
61*10465441SEvalZero         }
62*10465441SEvalZero 
63*10465441SEvalZero         /* 延时10个OS Tick */
64*10465441SEvalZero         rt_thread_delay(10);
65*10465441SEvalZero     }
66*10465441SEvalZero }
67*10465441SEvalZero 
68*10465441SEvalZero /* 线程3入口函数 */
thread3_entry(void * parameter)69*10465441SEvalZero static void thread3_entry(void* parameter)
70*10465441SEvalZero {
71*10465441SEvalZero     char buf[] = "this is an urgent message!";
72*10465441SEvalZero 
73*10465441SEvalZero     while (1)
74*10465441SEvalZero     {
75*10465441SEvalZero         rt_kprintf("thread3: send an urgent message\n");
76*10465441SEvalZero 
77*10465441SEvalZero         /* 发送紧急消息到消息队列中 */
78*10465441SEvalZero         rt_mq_urgent(&mq, &buf[0], sizeof(buf));
79*10465441SEvalZero 
80*10465441SEvalZero         /* 延时25个OS Tick */
81*10465441SEvalZero         rt_thread_delay(25);
82*10465441SEvalZero     }
83*10465441SEvalZero }
84*10465441SEvalZero 
messageq_simple_init()85*10465441SEvalZero int messageq_simple_init()
86*10465441SEvalZero {
87*10465441SEvalZero     /* 初始化消息队列 */
88*10465441SEvalZero     rt_mq_init(&mq, "mqt",
89*10465441SEvalZero                &msg_pool[0],        /* 内存池指向msg_pool */
90*10465441SEvalZero                128 - sizeof(void*), /* 每个消息的大小是 128 - void* */
91*10465441SEvalZero                sizeof(msg_pool),    /* 内存池的大小是msg_pool的大小 */
92*10465441SEvalZero                RT_IPC_FLAG_FIFO);   /* 如果有多个线程等待,按照先来先得到的方法分配消息 */
93*10465441SEvalZero 
94*10465441SEvalZero     /* 创建线程1 */
95*10465441SEvalZero     tid1 = rt_thread_create("t1",
96*10465441SEvalZero                             thread1_entry, RT_NULL, /* 线程入口是thread1_entry, 入口参数是RT_NULL */
97*10465441SEvalZero                             THREAD_STACK_SIZE, THREAD_PRIORITY, THREAD_TIMESLICE);
98*10465441SEvalZero     if (tid1 != RT_NULL)
99*10465441SEvalZero         rt_thread_startup(tid1);
100*10465441SEvalZero     else
101*10465441SEvalZero         tc_stat(TC_STAT_END | TC_STAT_FAILED);
102*10465441SEvalZero 
103*10465441SEvalZero     /* 创建线程2 */
104*10465441SEvalZero     tid2 = rt_thread_create("t2",
105*10465441SEvalZero                             thread2_entry, RT_NULL, /* 线程入口是thread2_entry, 入口参数是RT_NULL */
106*10465441SEvalZero                             THREAD_STACK_SIZE, THREAD_PRIORITY, THREAD_TIMESLICE);
107*10465441SEvalZero     if (tid2 != RT_NULL)
108*10465441SEvalZero         rt_thread_startup(tid2);
109*10465441SEvalZero     else
110*10465441SEvalZero         tc_stat(TC_STAT_END | TC_STAT_FAILED);
111*10465441SEvalZero 
112*10465441SEvalZero     /* 创建线程3 */
113*10465441SEvalZero     tid3 = rt_thread_create("t3",
114*10465441SEvalZero                             thread3_entry, RT_NULL, /* 线程入口是thread2_entry, 入口参数是RT_NULL */
115*10465441SEvalZero                             THREAD_STACK_SIZE, THREAD_PRIORITY, THREAD_TIMESLICE);
116*10465441SEvalZero     if (tid3 != RT_NULL)
117*10465441SEvalZero         rt_thread_startup(tid3);
118*10465441SEvalZero     else
119*10465441SEvalZero         tc_stat(TC_STAT_END | TC_STAT_FAILED);
120*10465441SEvalZero 
121*10465441SEvalZero     return 0;
122*10465441SEvalZero }
123*10465441SEvalZero 
124*10465441SEvalZero #ifdef RT_USING_TC
_tc_cleanup()125*10465441SEvalZero static void _tc_cleanup()
126*10465441SEvalZero {
127*10465441SEvalZero     /* 调度器上锁,上锁后,将不再切换到其他线程,仅响应中断 */
128*10465441SEvalZero     rt_enter_critical();
129*10465441SEvalZero 
130*10465441SEvalZero     /* 删除线程 */
131*10465441SEvalZero     if (tid1 != RT_NULL && tid1->stat != RT_THREAD_CLOSE)
132*10465441SEvalZero         rt_thread_delete(tid1);
133*10465441SEvalZero     if (tid2 != RT_NULL && tid2->stat != RT_THREAD_CLOSE)
134*10465441SEvalZero         rt_thread_delete(tid2);
135*10465441SEvalZero     if (tid3 != RT_NULL && tid3->stat != RT_THREAD_CLOSE)
136*10465441SEvalZero         rt_thread_delete(tid3);
137*10465441SEvalZero 
138*10465441SEvalZero     /* 执行消息队列对象脱离 */
139*10465441SEvalZero     rt_mq_detach(&mq);
140*10465441SEvalZero 
141*10465441SEvalZero     /* 调度器解锁 */
142*10465441SEvalZero     rt_exit_critical();
143*10465441SEvalZero 
144*10465441SEvalZero     /* 设置TestCase状态 */
145*10465441SEvalZero     tc_done(TC_STAT_PASSED);
146*10465441SEvalZero }
147*10465441SEvalZero 
_tc_messageq_simple()148*10465441SEvalZero int _tc_messageq_simple()
149*10465441SEvalZero {
150*10465441SEvalZero     /* 设置TestCase清理回调函数 */
151*10465441SEvalZero     tc_cleanup(_tc_cleanup);
152*10465441SEvalZero     messageq_simple_init();
153*10465441SEvalZero 
154*10465441SEvalZero     /* 返回TestCase运行的最长时间 */
155*10465441SEvalZero     return 100;
156*10465441SEvalZero }
157*10465441SEvalZero /* 输出函数命令到finsh shell中 */
158*10465441SEvalZero FINSH_FUNCTION_EXPORT(_tc_messageq_simple, a simple message queue example);
159*10465441SEvalZero #else
160*10465441SEvalZero /* 用户应用入口 */
rt_application_init()161*10465441SEvalZero int rt_application_init()
162*10465441SEvalZero {
163*10465441SEvalZero     messageq_simple_init();
164*10465441SEvalZero 
165*10465441SEvalZero     return 0;
166*10465441SEvalZero }
167*10465441SEvalZero #endif
168