memp_simple.c 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. /*
  2. * Copyright (c) 2006-2021, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. *
  8. */
  9. /*
  10. * 程序清单:内存池例程
  11. *
  12. * 这个程序会创建一个静态的内存池对象,2个动态线程。两个线程会试图分别从内存池中获得
  13. * 内存块
  14. */
  15. #include <rtthread.h>
  16. #include "tc_comm.h"
  17. static rt_uint8_t *ptr[48];
  18. static rt_uint8_t mempool[4096];
  19. static struct rt_mempool mp;
  20. /* 指向线程控制块的指针 */
  21. static rt_thread_t tid1 = RT_NULL;
  22. static rt_thread_t tid2 = RT_NULL;
  23. /* 线程1入口 */
  24. static void thread1_entry(void* parameter)
  25. {
  26. int i;
  27. char *block;
  28. while(1)
  29. {
  30. for (i = 0; i < 48; i++)
  31. {
  32. /* 申请内存块 */
  33. rt_kprintf("allocate No.%d\n", i);
  34. if (ptr[i] == RT_NULL)
  35. {
  36. ptr[i] = rt_mp_alloc(&mp, RT_WAITING_FOREVER);
  37. }
  38. }
  39. /* 继续申请一个内存块,因为已经没有内存块,线程应该被挂起 */
  40. block = rt_mp_alloc(&mp, RT_WAITING_FOREVER);
  41. rt_kprintf("allocate the block mem\n");
  42. /* 释放这个内存块 */
  43. rt_mp_free(block);
  44. block = RT_NULL;
  45. }
  46. }
  47. /* 线程2入口,线程2的优先级比线程1低,应该线程1先获得执行。*/
  48. static void thread2_entry(void *parameter)
  49. {
  50. int i;
  51. while(1)
  52. {
  53. rt_kprintf("try to release block\n");
  54. for (i = 0 ; i < 48; i ++)
  55. {
  56. /* 释放所有分配成功的内存块 */
  57. if (ptr[i] != RT_NULL)
  58. {
  59. rt_kprintf("release block %d\n", i);
  60. rt_mp_free(ptr[i]);
  61. ptr[i] = RT_NULL;
  62. }
  63. }
  64. /* 休眠10个OS Tick */
  65. rt_thread_delay(10);
  66. }
  67. }
  68. int mempool_simple_init()
  69. {
  70. int i;
  71. for (i = 0; i < 48; i ++) ptr[i] = RT_NULL;
  72. /* 初始化内存池对象 */
  73. rt_mp_init(&mp, "mp1", &mempool[0], sizeof(mempool), 80);
  74. /* 创建线程1 */
  75. tid1 = rt_thread_create("t1",
  76. thread1_entry, RT_NULL, /* 线程入口是thread1_entry, 入口参数是RT_NULL */
  77. THREAD_STACK_SIZE, THREAD_PRIORITY, THREAD_TIMESLICE);
  78. if (tid1 != RT_NULL)
  79. rt_thread_startup(tid1);
  80. else
  81. tc_stat(TC_STAT_END | TC_STAT_FAILED);
  82. /* 创建线程2 */
  83. tid2 = rt_thread_create("t2",
  84. thread2_entry, RT_NULL, /* 线程入口是thread2_entry, 入口参数是RT_NULL */
  85. THREAD_STACK_SIZE, THREAD_PRIORITY + 1, THREAD_TIMESLICE);
  86. if (tid2 != RT_NULL)
  87. rt_thread_startup(tid2);
  88. else
  89. tc_stat(TC_STAT_END | TC_STAT_FAILED);
  90. return 0;
  91. }
  92. #ifdef RT_USING_TC
  93. static void _tc_cleanup()
  94. {
  95. /* 调度器上锁,上锁后,将不再切换到其他线程,仅响应中断 */
  96. rt_enter_critical();
  97. /* 删除线程 */
  98. if (tid1 != RT_NULL && tid1->stat != RT_THREAD_CLOSE)
  99. rt_thread_delete(tid1);
  100. if (tid2 != RT_NULL && tid2->stat != RT_THREAD_CLOSE)
  101. rt_thread_delete(tid2);
  102. /* 执行内存池脱离 */
  103. rt_mp_detach(&mp);
  104. /* 调度器解锁 */
  105. rt_exit_critical();
  106. /* 设置TestCase状态 */
  107. tc_done(TC_STAT_PASSED);
  108. }
  109. int _tc_mempool_simple()
  110. {
  111. /* 设置TestCase清理回调函数 */
  112. tc_cleanup(_tc_cleanup);
  113. mempool_simple_init();
  114. /* 返回TestCase运行的最长时间 */
  115. return 100;
  116. }
  117. /* 输出函数命令到finsh shell中 */
  118. FINSH_FUNCTION_EXPORT(_tc_mempool_simple, a memory pool example);
  119. #else
  120. /* 用户应用入口 */
  121. int rt_application_init()
  122. {
  123. mempool_simple_init();
  124. return 0;
  125. }
  126. #endif