bypass_register.c 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. /*
  2. * Copyright (c) 2006-2024 RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2024-11-20 zhujiale the first version
  9. * 2025-11-24 ChuanN-sudo add standardized utest documentation block
  10. */
  11. /**
  12. * Test Case Name: Serial Bypass Register Test
  13. *
  14. * Test Objectives:
  15. * - Validate serial bypass function registration and unregistration mechanisms.
  16. * - Verify correct ordering of bypass functions by priority level in concurrent scenarios.
  17. * - Ensure thread-safe registration of upper and lower bypass handlers.
  18. * - Test core APIs: rt_bypass_upper_register(), rt_bypass_lower_register(),
  19. * rt_bypass_upper_unregister(), rt_bypass_lower_unregister()
  20. *
  21. * Test Scenarios:
  22. * - bypass_register_001: Main thread and worker thread concurrently register upper bypass
  23. * functions with even/odd priority levels to test insertion ordering and thread safety.
  24. * - bypass_register_002: Two worker threads simultaneously register upper and lower bypass
  25. * functions with levels 1-9 to verify independent queue management.
  26. * - Tests validate that registered functions maintain strict ascending order by level.
  27. * - Unregistration operations verify complete cleanup of bypass function chains.
  28. *
  29. * Verification Metrics:
  30. * - Bypass functions are inserted in correct priority order (level 0-9 ascending).
  31. * - Concurrent registrations from multiple threads maintain list integrity.
  32. * - Upper and lower bypass chains operate independently without interference.
  33. * - All registered bypass functions can be successfully unregistered by level.
  34. * - No memory leaks or list corruption after registration/unregistration cycles.
  35. *
  36. * Dependencies:
  37. * - Hardware requirements: Platform with serial console device (UART) support.
  38. * - Software configuration:
  39. * - RT_USING_UTESTCASES must be enabled (select "RT-Thread Utestcases" in menuconfig).
  40. * - RT_UTEST_SERIAL_BYPASS must be enabled (enable via: RT-Thread Utestcases -> Kernel Components -> Drivers -> Serial Test -> Serial Bypass Test).
  41. * - Environmental Assumptions: Serial device initialized and operational before test execution.
  42. *
  43. * Expected Results:
  44. * - Final output: "[ PASSED ] [ result ] testcase (components.drivers.serial.bypass_register)"
  45. * - All uassert_true() checks pass, confirming correct level ordering.
  46. */
  47. #include <rtthread.h>
  48. #include <rtdevice.h>
  49. #include "utest.h"
  50. static struct rt_serial_device* _serial0;
  51. static struct rt_spinlock lock;
  52. static rt_err_t utest_001_run(struct rt_serial_device* serial, char ch, void* data)
  53. {
  54. return 0;
  55. }
  56. static void thread_serial_register1(void* parameter)
  57. {
  58. for (int i = 2; i < 10; i += 2)
  59. {
  60. rt_bypass_upper_register(_serial0, "test", i, utest_001_run, RT_NULL);
  61. }
  62. }
  63. static void thread_serial_register_upper(void* parameter)
  64. {
  65. for (int i = 1; i < 10; i++)
  66. {
  67. rt_bypass_upper_register(_serial0, "test", i, utest_001_run, RT_NULL);
  68. }
  69. }
  70. static void thread_serial_register_lower(void* parameter)
  71. {
  72. for (int i = 1; i < 10; i++)
  73. {
  74. rt_bypass_lower_register(_serial0, "test", i, utest_001_run, RT_NULL);
  75. }
  76. }
  77. static void bypass_register_001(void)
  78. {
  79. rt_thread_t t1 = rt_thread_create("serial_register", thread_serial_register1, RT_NULL, 2048, RT_THREAD_PRIORITY_MAX - 5, 10);
  80. rt_bypass_upper_register(_serial0, "test", 0, utest_001_run, RT_NULL);
  81. rt_thread_startup(t1);
  82. for (int i = 1; i < 10; i += 2)
  83. {
  84. rt_bypass_upper_register(_serial0, "test", i, utest_001_run, RT_NULL);
  85. }
  86. rt_thread_mdelay(1000);
  87. rt_list_t* node = _serial0->bypass->upper_h->head.next;
  88. for (int i = 0; i < 10;i++)
  89. {
  90. rt_list_t* next = node->next;
  91. struct rt_serial_bypass_func* temp = rt_container_of(node, struct rt_serial_bypass_func, node);
  92. uassert_true(temp->level == i);
  93. rt_bypass_upper_unregister(_serial0, temp->level);
  94. node = next;
  95. }
  96. }
  97. static void bypass_register_002(void)
  98. {
  99. rt_thread_t t1 = rt_thread_create("serial_register", thread_serial_register_upper, RT_NULL, 2048, RT_THREAD_PRIORITY_MAX - 5, 10);
  100. rt_thread_t t2 = rt_thread_create("serial_register", thread_serial_register_lower, RT_NULL, 2048, RT_THREAD_PRIORITY_MAX - 5, 10);
  101. rt_bypass_upper_register(_serial0, "test", 0, utest_001_run, RT_NULL);
  102. rt_thread_startup(t1);
  103. rt_thread_startup(t2);
  104. rt_thread_mdelay(1000);
  105. rt_list_t* node = _serial0->bypass->upper_h->head.next;
  106. for (int i = 0; i < 10;i++)
  107. {
  108. rt_list_t* next = node->next;
  109. struct rt_serial_bypass_func* temp = rt_container_of(node, struct rt_serial_bypass_func, node);
  110. uassert_true(temp->level == i);
  111. rt_bypass_upper_unregister(_serial0, temp->level);
  112. node = next;
  113. }
  114. node = _serial0->bypass->lower_h->head.next;
  115. for (int i = 1; i < 10;i++)
  116. {
  117. rt_list_t* next = node->next;
  118. struct rt_serial_bypass_func* temp = rt_container_of(node, struct rt_serial_bypass_func, node);
  119. uassert_true(temp->level == i);
  120. rt_bypass_lower_unregister(_serial0, temp->level);
  121. node = next;
  122. }
  123. }
  124. static rt_err_t utest_tc_init(void)
  125. {
  126. _serial0 = (struct rt_serial_device*)rt_console_get_device();
  127. rt_spin_lock_init(&lock);
  128. return RT_EOK;
  129. }
  130. static rt_err_t utest_tc_cleanup(void)
  131. {
  132. return RT_EOK;
  133. }
  134. static void _testcase(void)
  135. {
  136. UTEST_UNIT_RUN(bypass_register_001);
  137. UTEST_UNIT_RUN(bypass_register_002);
  138. }
  139. UTEST_TC_EXPORT(_testcase, "components.drivers.serial.bypass_register", utest_tc_init, utest_tc_cleanup, 10);