| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126 |
- /*
- * Copyright (c) 2006-2024 RT-Thread Development Team
- *
- * SPDX-License-Identifier: Apache-2.0
- *
- * Change Logs:
- * Date Author Notes
- * 2024/9/12 zhujiale the first version
- * 2024/10/28 Shell Added more assertions
- * 2025/12/3 ChuanN-sudo add standardized utest documentation block
- */
- /**
- * Test Case Name: SMP Call Smoke 001 Test
- *
- * Test Objectives:
- * - Validate rt_smp_call_cpu_mask() with SMP_CALL_WAIT_ALL flag for blocking synchronous calls.
- * - Verify callback execution on all targeted CPUs.
- * - Ensure callback runs in interrupt-disabled context.
- * - Test core APIs: rt_smp_call_cpu_mask().
- *
- * Test Scenarios:
- * - Main thread generates random CPU masks and dispatches callbacks to targeted CPUs.
- * - Callback function clears corresponding CPU bit from shared mask variable.
- * - Blocking call waits for all targeted CPUs to complete execution.
- * - Random CPU selection simulates various multi-core scenarios.
- *
- * Verification Metrics:
- * - rt_smp_call_cpu_mask() must block until all targeted CPUs complete callback execution.
- * - Shared mask variable must be zero after function returns, confirming all targets executed.
- * - Callback must execute in interrupt-disabled context.
- * - All available CPUs must be tested throughout the run.
- * - Data integrity maintained across CPUs with spinlock protection.
- *
- * Dependencies:
- * - Hardware requirements: QEMU emulator or any multi-core hardware platform that supports RT-Thread.
- * - Software configuration:
- * - RT_USING_UTEST must be enabled (select "RT-Thread Utestcases" in menuconfig).
- * - RT_UTEST_SMP_CALL_FUNC must be enabled(enable via: RT-Thread Utestcases -> Kernel Components -> Drivers -> SMP-Call Test -> SMP-Call Smoke Test).
- * - Environmental Assumptions: System scheduler and SMP services working normally.
- *
- * Expected Results:
- * - Final output: "[ PASSED ] [ result ] testcase (components.drivers.smp_call.smoke_001)"
- * - No assertions triggered during test execution.
- */
- #include <rtdevice.h>
- #include <utest.h>
- #include <utest_assert.h>
- #include <smp_call.h>
- #define TEST_COUNT 10000
- static int pass_count = 0;
- static RT_DEFINE_SPINLOCK(_test_data_lock);
- static void _test_smp_cb(void *data)
- {
- int *maskp;
- int oncpu;
- if (!rt_hw_interrupt_is_disabled())
- {
- /* SYNC.004 */
- uassert_true(0);
- }
- rt_spin_lock(&_test_data_lock);
- oncpu = rt_hw_cpu_id();
- maskp = (int *)data;
- *maskp &= ~(1 << oncpu);
- rt_spin_unlock(&_test_data_lock);
- }
- static void _blocking_call(void)
- {
- volatile int cpu_mask;
- rt_ubase_t tested_cpus = 0;
- for (int i = 0; i < TEST_COUNT; i++)
- {
- cpu_mask = rand() % RT_ALL_CPU;
- tested_cpus |= cpu_mask;
- rt_smp_call_cpu_mask(cpu_mask, _test_smp_cb, (void *)&cpu_mask, SMP_CALL_WAIT_ALL);
- if (!cpu_mask)
- {
- pass_count++;
- }
- else
- {
- /* TARG.001, MP.001 */
- uassert_true(0);
- break;
- }
- }
- LOG_D("pass_count %d", pass_count);
- /* TARG.001 */
- uassert_true(pass_count == TEST_COUNT);
- /* TOP.001, TOP.002 */
- uassert_true(tested_cpus == RT_ALL_CPU);
- }
- static rt_err_t utest_tc_init(void)
- {
- pass_count = 0;
- srand(rt_tick_get());
- return RT_EOK;
- }
- static rt_err_t utest_tc_cleanup(void)
- {
- return RT_EOK;
- }
- static void _testcase(void)
- {
- UTEST_UNIT_RUN(_blocking_call);
- }
- UTEST_TC_EXPORT(_testcase, "components.drivers.smp_call.smoke_001", utest_tc_init, utest_tc_cleanup, 10);
|