mutex_sample.c 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118
  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. */
  10. /*
  11. * 程序清单:互斥锁例程
  12. *
  13. * 互斥锁是一种保护共享资源的方法。当一个线程拥有互斥锁的时候,
  14. * 可以保护共享资源不被其他线程破坏。线程1对2个number分别进行加1操作
  15. * 线程2也会对2个number分别进行加1操作。使用互斥量保证2个number值保持一致
  16. */
  17. #include <rtthread.h>
  18. #define THREAD_PRIORITY 8
  19. #define THREAD_TIMESLICE 5
  20. /* 指向互斥量的指针 */
  21. static rt_mutex_t dynamic_mutex = RT_NULL;
  22. static rt_uint8_t number1, number2 = 0;
  23. #ifdef rt_align
  24. rt_align(RT_ALIGN_SIZE)
  25. #else
  26. ALIGN(RT_ALIGN_SIZE)
  27. #endif
  28. static char thread1_stack[1024];
  29. static struct rt_thread thread1;
  30. static void rt_thread_entry1(void *parameter)
  31. {
  32. while (1)
  33. {
  34. /* 线程1获取到互斥量后,先后对number1、number2进行加1操作,然后释放互斥量 */
  35. rt_mutex_take(dynamic_mutex, RT_WAITING_FOREVER);
  36. number1++;
  37. rt_thread_mdelay(10);
  38. number2++;
  39. rt_mutex_release(dynamic_mutex);
  40. }
  41. }
  42. #ifdef rt_align
  43. rt_align(RT_ALIGN_SIZE)
  44. #else
  45. ALIGN(RT_ALIGN_SIZE)
  46. #endif
  47. static char thread2_stack[1024];
  48. static struct rt_thread thread2;
  49. static void rt_thread_entry2(void *parameter)
  50. {
  51. while (1)
  52. {
  53. /* 线程2获取到互斥量后,检查number1、number2的值是否相同,相同则表示mutex起到了锁的作用 */
  54. rt_mutex_take(dynamic_mutex, RT_WAITING_FOREVER);
  55. if (number1 != number2)
  56. {
  57. rt_kprintf("not protect.number1 = %d, mumber2 = %d \n", number1, number2);
  58. }
  59. else
  60. {
  61. rt_kprintf("mutex protect ,number1 = mumber2 is %d\n", number1);
  62. }
  63. number1++;
  64. number2++;
  65. rt_mutex_release(dynamic_mutex);
  66. if (number1 >= 50)
  67. return;
  68. }
  69. }
  70. /* 互斥量示例的初始化 */
  71. int mutex_sample(void)
  72. {
  73. /* 创建一个动态互斥量 */
  74. dynamic_mutex = rt_mutex_create("dmutex", RT_IPC_FLAG_PRIO);
  75. if (dynamic_mutex == RT_NULL)
  76. {
  77. rt_kprintf("create dynamic mutex failed.\n");
  78. return -1;
  79. }
  80. rt_thread_init(&thread1,
  81. "thread1",
  82. rt_thread_entry1,
  83. RT_NULL,
  84. &thread1_stack[0],
  85. sizeof(thread1_stack),
  86. THREAD_PRIORITY, THREAD_TIMESLICE);
  87. #ifdef RT_USING_SMP
  88. /* 绑定线程到同一个核上,避免启用多核时的输出混乱 */
  89. rt_thread_control(&thread1, RT_THREAD_CTRL_BIND_CPU, (void*)0);
  90. #endif
  91. rt_thread_startup(&thread1);
  92. rt_thread_init(&thread2,
  93. "thread2",
  94. rt_thread_entry2,
  95. RT_NULL,
  96. &thread2_stack[0],
  97. sizeof(thread2_stack),
  98. THREAD_PRIORITY - 1, THREAD_TIMESLICE);
  99. #ifdef RT_USING_SMP
  100. /* 绑定线程到同一个核上,避免启用多核时的输出混乱 */
  101. rt_thread_control(&thread2, RT_THREAD_CTRL_BIND_CPU, (void*)0);
  102. #endif
  103. rt_thread_startup(&thread2);
  104. return 0;
  105. }
  106. /* 导出到 msh 命令列表中 */
  107. MSH_CMD_EXPORT(mutex_sample, mutex sample);