| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155 |
- /*
- * Copyright (c) 2006-2024 RT-Thread Development Team
- *
- * SPDX-License-Identifier: Apache-2.0
- *
- * Change Logs:
- * Date Author Notes
- * 2024-11-20 zhujiale the first version
- * 2025-11-24 ChuanN-sudo add standardized utest documentation block
- */
- /**
- * Test Case Name: Serial Bypass Register Test
- *
- * Test Objectives:
- * - Validate serial bypass function registration and unregistration mechanisms.
- * - Verify correct ordering of bypass functions by priority level in concurrent scenarios.
- * - Ensure thread-safe registration of upper and lower bypass handlers.
- * - Test core APIs: rt_bypass_upper_register(), rt_bypass_lower_register(),
- * rt_bypass_upper_unregister(), rt_bypass_lower_unregister()
- *
- * Test Scenarios:
- * - bypass_register_001: Main thread and worker thread concurrently register upper bypass
- * functions with even/odd priority levels to test insertion ordering and thread safety.
- * - bypass_register_002: Two worker threads simultaneously register upper and lower bypass
- * functions with levels 1-9 to verify independent queue management.
- * - Tests validate that registered functions maintain strict ascending order by level.
- * - Unregistration operations verify complete cleanup of bypass function chains.
- *
- * Verification Metrics:
- * - Bypass functions are inserted in correct priority order (level 0-9 ascending).
- * - Concurrent registrations from multiple threads maintain list integrity.
- * - Upper and lower bypass chains operate independently without interference.
- * - All registered bypass functions can be successfully unregistered by level.
- * - No memory leaks or list corruption after registration/unregistration cycles.
- *
- * Dependencies:
- * - Hardware requirements: Platform with serial console device (UART) support.
- * - Software configuration:
- * - RT_USING_UTESTCASES must be enabled (select "RT-Thread Utestcases" in menuconfig).
- * - RT_UTEST_SERIAL_BYPASS must be enabled (enable via: RT-Thread Utestcases -> Kernel Components -> Drivers -> Serial Test -> Serial Bypass Test).
- * - Environmental Assumptions: Serial device initialized and operational before test execution.
- *
- * Expected Results:
- * - Final output: "[ PASSED ] [ result ] testcase (components.drivers.serial.bypass_register)"
- * - All uassert_true() checks pass, confirming correct level ordering.
- */
- #include <rtthread.h>
- #include <rtdevice.h>
- #include "utest.h"
- static struct rt_serial_device* _serial0;
- static struct rt_spinlock lock;
- static rt_err_t utest_001_run(struct rt_serial_device* serial, char ch, void* data)
- {
- return 0;
- }
- static void thread_serial_register1(void* parameter)
- {
- for (int i = 2; i < 10; i += 2)
- {
- rt_bypass_upper_register(_serial0, "test", i, utest_001_run, RT_NULL);
- }
- }
- static void thread_serial_register_upper(void* parameter)
- {
- for (int i = 1; i < 10; i++)
- {
- rt_bypass_upper_register(_serial0, "test", i, utest_001_run, RT_NULL);
- }
- }
- static void thread_serial_register_lower(void* parameter)
- {
- for (int i = 1; i < 10; i++)
- {
- rt_bypass_lower_register(_serial0, "test", i, utest_001_run, RT_NULL);
- }
- }
- static void bypass_register_001(void)
- {
- rt_thread_t t1 = rt_thread_create("serial_register", thread_serial_register1, RT_NULL, 2048, RT_THREAD_PRIORITY_MAX - 5, 10);
- rt_bypass_upper_register(_serial0, "test", 0, utest_001_run, RT_NULL);
- rt_thread_startup(t1);
- for (int i = 1; i < 10; i += 2)
- {
- rt_bypass_upper_register(_serial0, "test", i, utest_001_run, RT_NULL);
- }
- rt_thread_mdelay(1000);
- rt_list_t* node = _serial0->bypass->upper_h->head.next;
- for (int i = 0; i < 10;i++)
- {
- rt_list_t* next = node->next;
- struct rt_serial_bypass_func* temp = rt_container_of(node, struct rt_serial_bypass_func, node);
- uassert_true(temp->level == i);
- rt_bypass_upper_unregister(_serial0, temp->level);
- node = next;
- }
- }
- static void bypass_register_002(void)
- {
- rt_thread_t t1 = rt_thread_create("serial_register", thread_serial_register_upper, RT_NULL, 2048, RT_THREAD_PRIORITY_MAX - 5, 10);
- rt_thread_t t2 = rt_thread_create("serial_register", thread_serial_register_lower, RT_NULL, 2048, RT_THREAD_PRIORITY_MAX - 5, 10);
- rt_bypass_upper_register(_serial0, "test", 0, utest_001_run, RT_NULL);
- rt_thread_startup(t1);
- rt_thread_startup(t2);
- rt_thread_mdelay(1000);
- rt_list_t* node = _serial0->bypass->upper_h->head.next;
- for (int i = 0; i < 10;i++)
- {
- rt_list_t* next = node->next;
- struct rt_serial_bypass_func* temp = rt_container_of(node, struct rt_serial_bypass_func, node);
- uassert_true(temp->level == i);
- rt_bypass_upper_unregister(_serial0, temp->level);
- node = next;
- }
- node = _serial0->bypass->lower_h->head.next;
- for (int i = 1; i < 10;i++)
- {
- rt_list_t* next = node->next;
- struct rt_serial_bypass_func* temp = rt_container_of(node, struct rt_serial_bypass_func, node);
- uassert_true(temp->level == i);
- rt_bypass_lower_unregister(_serial0, temp->level);
- node = next;
- }
- }
- static rt_err_t utest_tc_init(void)
- {
- _serial0 = (struct rt_serial_device*)rt_console_get_device();
- rt_spin_lock_init(&lock);
- return RT_EOK;
- }
- static rt_err_t utest_tc_cleanup(void)
- {
- return RT_EOK;
- }
- static void _testcase(void)
- {
- UTEST_UNIT_RUN(bypass_register_001);
- UTEST_UNIT_RUN(bypass_register_002);
- }
- UTEST_TC_EXPORT(_testcase, "components.drivers.serial.bypass_register", utest_tc_init, utest_tc_cleanup, 10);
|