mailbox_urgent_sample.c 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  1. /*
  2. * Copyright (c) 2006-2022, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2018-08-24 yangjie the first version
  9. * 2021-01-03 Meco Man add the demo of rt_mb_urgent()
  10. */
  11. /*
  12. * 程序清单:邮箱例程
  13. *
  14. * 这个程序会创建2个动态线程,一个静态的邮箱对象,其中一个线程往邮箱中发送邮件,
  15. * 一个线程往邮箱中收取邮件。
  16. * 本例程也会展示紧急邮件和正常邮件的区别。
  17. */
  18. #include <rtthread.h>
  19. #define THREAD_PRIORITY 10
  20. #define THREAD_TIMESLICE 5
  21. /* 邮箱控制块 */
  22. static struct rt_mailbox mb;
  23. /* 用于放邮件的内存池 */
  24. static char mb_pool[128];
  25. static char mb_urgent[] = "I'm a urgent mail!";
  26. static char mb_normal[] = "I'm a normal mail!";
  27. static char mb_str3[] = "over";
  28. #ifdef rt_align
  29. rt_align(RT_ALIGN_SIZE)
  30. #else
  31. ALIGN(RT_ALIGN_SIZE)
  32. #endif
  33. static char thread1_stack[1024];
  34. static struct rt_thread thread1;
  35. /* 线程1入口 */
  36. static void thread1_entry(void *parameter)
  37. {
  38. char *str;
  39. while (1)
  40. {
  41. rt_kprintf("thread1: try to recv a mail\n");
  42. /* 从邮箱中收取邮件 */
  43. if (rt_mb_recv(&mb, (rt_ubase_t *)&str, RT_WAITING_FOREVER) == RT_EOK)
  44. {
  45. rt_kprintf("thread1: get a mail from mailbox, the content:%s\n", str);
  46. if (str == mb_str3)
  47. break;
  48. /* 延时100ms */
  49. rt_thread_mdelay(100);
  50. }
  51. }
  52. /* 执行邮箱对象脱离 */
  53. rt_mb_detach(&mb);
  54. }
  55. #ifdef rt_align
  56. rt_align(RT_ALIGN_SIZE)
  57. #else
  58. ALIGN(RT_ALIGN_SIZE)
  59. #endif
  60. static char thread2_stack[1024];
  61. static struct rt_thread thread2;
  62. /* 线程2入口 */
  63. static void thread2_entry(void *parameter)
  64. {
  65. rt_uint8_t count;
  66. count = 0;
  67. while (count < 10)
  68. {
  69. count ++;
  70. /* 分别发送正常邮件和紧急邮件到邮箱中 */
  71. /* 关调度器是为了同时给线程1两封邮件,观察线程1如何处理紧急邮件和正常邮件 */
  72. rt_enter_critical();
  73. rt_mb_send(&mb, (rt_uint32_t)&mb_normal);
  74. rt_mb_urgent(&mb, (rt_uint32_t)&mb_urgent);
  75. rt_exit_critical();
  76. /* 可以看到虽然正常邮件和紧急优先同时发送,但是邮箱选择优先处理紧急邮件 */
  77. /* 延时200ms */
  78. rt_thread_mdelay(200);
  79. }
  80. /* 发送邮件告诉线程1,线程2已经运行结束 */
  81. rt_mb_send(&mb, (rt_uint32_t)&mb_str3);
  82. }
  83. int mailbox_urgent_sample(void)
  84. {
  85. rt_err_t result;
  86. /* 初始化一个mailbox */
  87. result = rt_mb_init(&mb,
  88. "mbt", /* 名称是mbt */
  89. &mb_pool[0], /* 邮箱用到的内存池是mb_pool */
  90. sizeof(mb_pool) / sizeof(rt_ubase_t), /* 邮箱中的邮件数目,sizeof(rt_ubase_t)表示指针大小 */
  91. RT_IPC_FLAG_PRIO); /* 采用PRIO方式进行线程等待 */
  92. if (result != RT_EOK)
  93. {
  94. rt_kprintf("init mailbox failed.\n");
  95. return -1;
  96. }
  97. rt_thread_init(&thread1,
  98. "thread1",
  99. thread1_entry,
  100. RT_NULL,
  101. &thread1_stack[0],
  102. sizeof(thread1_stack),
  103. THREAD_PRIORITY, THREAD_TIMESLICE);
  104. #ifdef RT_USING_SMP
  105. /* 绑定线程到同一个核上,避免启用多核时的输出混乱 */
  106. rt_thread_control(&thread1, RT_THREAD_CTRL_BIND_CPU, (void*)0);
  107. #endif
  108. rt_thread_startup(&thread1);
  109. rt_thread_init(&thread2,
  110. "thread2",
  111. thread2_entry,
  112. RT_NULL,
  113. &thread2_stack[0],
  114. sizeof(thread2_stack),
  115. THREAD_PRIORITY, THREAD_TIMESLICE);
  116. #ifdef RT_USING_SMP
  117. /* 绑定线程到同一个核上,避免启用多核时的输出混乱 */
  118. rt_thread_control(&thread2, RT_THREAD_CTRL_BIND_CPU, (void*)0);
  119. #endif
  120. rt_thread_startup(&thread2);
  121. return 0;
  122. }
  123. /* 导出到 msh 命令列表中 */
  124. MSH_CMD_EXPORT(mailbox_urgent_sample, mailbox urgent sample);