tc_comm.c 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. #include "tc_comm.h"
  2. #ifdef RT_USING_FINSH
  3. #include <finsh.h>
  4. #endif
  5. #ifdef RT_USING_TC
  6. #define TC_PRIORITY 25
  7. #define TC_STACK_SIZE 0x400
  8. static rt_uint8_t _tc_stat;
  9. static struct rt_semaphore _tc_sem;
  10. static struct rt_thread _tc_thread;
  11. static rt_uint8_t _tc_stack[TC_STACK_SIZE];
  12. static char _tc_prefix[64];
  13. static const char* _tc_current;
  14. static void (*_tc_cleanup)(void) = RT_NULL;
  15. static rt_uint32_t _tc_scale = 1;
  16. FINSH_VAR_EXPORT(_tc_scale, finsh_type_int, the testcase timer timeout scale)
  17. void tc_thread_entry(void* parameter)
  18. {
  19. struct finsh_syscall* index;
  20. /* create tc semaphore */
  21. rt_sem_init(&_tc_sem, "tc", 0, RT_IPC_FLAG_FIFO);
  22. while (_tc_stat & TC_STAT_RUNNING)
  23. {
  24. for (index = _syscall_table_begin; index < _syscall_table_end; index ++)
  25. {
  26. /* search testcase */
  27. if (rt_strstr(index->name, _tc_prefix) == index->name)
  28. {
  29. long tick;
  30. _tc_current = index->name + 4;
  31. rt_kprintf("Run TestCase: %s\n", _tc_current);
  32. _tc_stat = TC_STAT_PASSED | TC_STAT_RUNNING;
  33. tick = index->func();
  34. if (tick > 0)
  35. {
  36. rt_sem_take(&_tc_sem, tick * _tc_scale);
  37. if (_tc_cleanup != RT_NULL)
  38. {
  39. /* perform testcase cleanup */
  40. _tc_cleanup();
  41. _tc_cleanup = RT_NULL;
  42. }
  43. rt_sem_trytake(&_tc_sem);/* by nl1031 */
  44. if (_tc_stat & TC_STAT_FAILED)
  45. rt_kprintf("TestCase[%s] failed\n", _tc_current);
  46. else
  47. rt_kprintf("TestCase[%s] passed\n", _tc_current);
  48. }
  49. else
  50. {
  51. if (_tc_cleanup != RT_NULL)
  52. {
  53. /* perform testcase cleanup */
  54. _tc_cleanup();
  55. _tc_cleanup = RT_NULL;
  56. }
  57. }
  58. }
  59. }
  60. }
  61. rt_kprintf("RT-Thread TestCase Running Done!\n");
  62. /* detach tc semaphore */
  63. rt_sem_detach(&_tc_sem);
  64. }
  65. void tc_stop()
  66. {
  67. _tc_stat &= ~TC_STAT_RUNNING;
  68. rt_thread_delay(RT_TICK_PER_SECOND/2);
  69. if (_tc_thread.stat != RT_THREAD_INIT)
  70. {
  71. /* lock scheduler */
  72. rt_enter_critical();
  73. /* detach old tc thread */
  74. rt_thread_detach(&_tc_thread);
  75. rt_sem_detach(&_tc_sem);
  76. /* unlock scheduler */
  77. rt_exit_critical();
  78. }
  79. rt_thread_delay(RT_TICK_PER_SECOND/2);
  80. }
  81. FINSH_FUNCTION_EXPORT(tc_stop, stop testcase thread);
  82. void tc_done(rt_uint8_t stat)
  83. {
  84. _tc_stat |= stat;
  85. _tc_stat &= ~TC_STAT_RUNNING;
  86. /* release semaphore */
  87. rt_sem_release(&_tc_sem);
  88. }
  89. void tc_stat(rt_uint8_t stat)
  90. {
  91. if (stat & TC_STAT_FAILED)
  92. {
  93. rt_kprintf("TestCases[%s] failed\n", _tc_current);
  94. }
  95. _tc_stat |= stat;
  96. }
  97. void tc_cleanup(void (*cleanup)())
  98. {
  99. _tc_cleanup = cleanup;
  100. }
  101. void tc_start(const char* tc_prefix)
  102. {
  103. rt_err_t result;
  104. /* tesecase prefix is null */
  105. if (tc_prefix == RT_NULL)
  106. {
  107. rt_kprintf("TestCase Usage: tc_start(prefix)\n\n");
  108. rt_kprintf("list_tc() can list all testcases.\n");
  109. return ;
  110. }
  111. /* init tc thread */
  112. if (_tc_stat & TC_STAT_RUNNING)
  113. {
  114. /* stop old tc thread */
  115. tc_stop();
  116. }
  117. rt_memset(_tc_prefix, 0, sizeof(_tc_prefix));
  118. rt_snprintf(_tc_prefix, sizeof(_tc_prefix), "_tc_%s", tc_prefix);
  119. result = rt_thread_init(&_tc_thread, "tc",
  120. tc_thread_entry, RT_NULL,
  121. &_tc_stack[0], sizeof(_tc_stack),
  122. TC_PRIORITY - 3, 5);
  123. /* set tc stat */
  124. _tc_stat = TC_STAT_RUNNING | TC_STAT_FAILED;
  125. if (result == RT_EOK)
  126. rt_thread_startup(&_tc_thread);
  127. }
  128. FINSH_FUNCTION_EXPORT(tc_start, start testcase with testcase prefix or name);
  129. void list_tc()
  130. {
  131. struct finsh_syscall* index;
  132. rt_kprintf("TestCases List:\n");
  133. for (index = _syscall_table_begin; index < _syscall_table_end; index ++)
  134. {
  135. /* search testcase */
  136. if (rt_strstr(index->name, "_tc_") == index->name)
  137. {
  138. #ifdef FINSH_USING_DESCRIPTION
  139. rt_kprintf("%-16s -- %s\n", index->name + 4, index->desc);
  140. #else
  141. rt_kprintf("%s\n", index->name + 4);
  142. #endif
  143. }
  144. }
  145. }
  146. FINSH_FUNCTION_EXPORT(list_tc, list all testcases);
  147. #endif