2024-10-30 19:45:49 +08:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2006-2024 RT-Thread Development Team
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*
|
|
|
|
* Change Logs:
|
|
|
|
* Date Author Notes
|
|
|
|
* 2024/10/28 Shell Added smp.smoke
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <rtdevice.h>
|
|
|
|
#include <utest.h>
|
|
|
|
#include <utest_assert.h>
|
2024-10-31 13:50:41 +08:00
|
|
|
#include <smp_call.h>
|
2024-10-30 19:45:49 +08:00
|
|
|
|
|
|
|
#define PERCPU_TEST_COUNT 10000
|
|
|
|
#define NEWLINE_ON 80
|
|
|
|
|
|
|
|
static struct rt_semaphore _utestd_exited;
|
|
|
|
static rt_thread_t _utestd[RT_CPUS_NR];
|
|
|
|
static rt_atomic_t _entry_counts[RT_CPUS_NR];
|
|
|
|
|
|
|
|
static void _logging_progress(void)
|
|
|
|
{
|
|
|
|
static rt_atomic_t counts;
|
|
|
|
rt_ubase_t old;
|
|
|
|
|
|
|
|
rt_kputs("#");
|
|
|
|
old = rt_atomic_add(&counts, 1);
|
|
|
|
if (old % NEWLINE_ON == 0)
|
|
|
|
{
|
|
|
|
rt_kputs("\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _test_smp_cb(void *param)
|
|
|
|
{
|
|
|
|
rt_ubase_t req_cpuid = (rt_ubase_t)param;
|
|
|
|
|
|
|
|
if (!rt_hw_interrupt_is_disabled())
|
|
|
|
{
|
|
|
|
/* SYNC.004 */
|
|
|
|
uassert_true(0);
|
|
|
|
}
|
|
|
|
_logging_progress();
|
|
|
|
rt_atomic_add(&_entry_counts[req_cpuid], 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _utestd_entry(void *oncpu_param)
|
|
|
|
{
|
|
|
|
rt_ubase_t oncpu = (rt_ubase_t)oncpu_param;
|
|
|
|
volatile int cpu_mask;
|
|
|
|
volatile int popcount = 0;
|
|
|
|
rt_thread_t curthr = rt_thread_self();
|
|
|
|
|
|
|
|
if (rt_hw_cpu_id() != oncpu)
|
|
|
|
{
|
|
|
|
/* SYNC.004 */
|
|
|
|
uassert_true(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0; i < PERCPU_TEST_COUNT; i++)
|
|
|
|
{
|
|
|
|
cpu_mask = rand() % RT_ALL_CPU;
|
|
|
|
|
|
|
|
rt_smp_call_cpu_mask(cpu_mask, _test_smp_cb, oncpu_param, 0);
|
|
|
|
popcount += __builtin_popcount(cpu_mask);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0; i < RT_CPUS_NR; i++)
|
|
|
|
{
|
|
|
|
rt_thread_control(curthr, RT_THREAD_CTRL_BIND_CPU, (void *)i);
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG_D("popcount %d, _entry_counts[%d] %d", popcount, oncpu, _entry_counts[oncpu]);
|
|
|
|
|
|
|
|
/* MP.002 */
|
|
|
|
uassert_true(popcount == rt_atomic_load(&_entry_counts[oncpu]));
|
|
|
|
|
|
|
|
rt_sem_release(&_utestd_exited);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _async_call(void)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < RT_CPUS_NR; i++)
|
|
|
|
{
|
|
|
|
rt_thread_startup(_utestd[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0; i < RT_CPUS_NR; i++)
|
|
|
|
{
|
|
|
|
rt_sem_take(&_utestd_exited, RT_WAITING_FOREVER);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static rt_err_t utest_tc_init(void)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < RT_CPUS_NR; i++)
|
|
|
|
{
|
|
|
|
rt_atomic_store(&_entry_counts[i], 0);
|
|
|
|
_utestd[i] = rt_thread_create("utestd", _utestd_entry, (void *)i,
|
|
|
|
UTEST_THR_STACK_SIZE, UTEST_THR_PRIORITY,
|
|
|
|
20);
|
|
|
|
rt_thread_control(_utestd[i], RT_THREAD_CTRL_BIND_CPU, (void *)i);
|
|
|
|
|
|
|
|
/* SYNC.001, SYNC.002, SYNC.003 */
|
|
|
|
uassert_true(_utestd[i] != RT_NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
rt_sem_init(&_utestd_exited, "utestd", 0, RT_IPC_FLAG_PRIO);
|
|
|
|
srand(rt_tick_get());
|
|
|
|
|
|
|
|
return RT_EOK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static rt_err_t utest_tc_cleanup(void)
|
|
|
|
{
|
|
|
|
rt_sem_detach(&_utestd_exited);
|
|
|
|
|
|
|
|
return RT_EOK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _testcase(void)
|
|
|
|
{
|
|
|
|
UTEST_UNIT_RUN(_async_call);
|
|
|
|
}
|
|
|
|
|
|
|
|
UTEST_TC_EXPORT(_testcase, "testcase.smp.smoke.003", utest_tc_init, utest_tc_cleanup, 10);
|