e25fc8b511
* [libcpu/arm64] add C11 atomic ticket spinlock Replace the former implementation of flag-based spinlock which is unfair Besides, C11 atomic implementation is more readable (it's C anyway), and maintainable. Cause toolchain can use their builtin optimization and tune for different micro-architectures. For example armv8.5 introduces a better instruction. The compiler can help with that when it knows your target platform in support of it. Signed-off-by: Shell <smokewood@qq.com> * fixup: RT_CPUS_NR --------- Signed-off-by: Shell <smokewood@qq.com>
199 lines
5.4 KiB
C
199 lines
5.4 KiB
C
/*
|
|
* Copyright (c) 2006-2024, RT-Thread Development Team
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*
|
|
* Change Logs:
|
|
* Date Author Notes
|
|
* 2024-01-17 Shell the first version
|
|
*/
|
|
#define __RT_IPC_SOURCE__
|
|
|
|
#include <rtthread.h>
|
|
#include "rthw.h"
|
|
#include "utest.h"
|
|
|
|
#define KERN_TEST_CONFIG_LOOP_TIMES 160
|
|
#define KERN_TEST_CONCURRENT_THREADS (RT_CPUS_NR * 2)
|
|
#define KERN_TEST_CONFIG_HIGHEST_PRIO 3
|
|
#define KERN_TEST_CONFIG_LOWEST_PRIO (RT_THREAD_PRIORITY_MAX - 2)
|
|
|
|
#define TEST_LEVEL_COUNTS (KERN_TEST_CONFIG_LOWEST_PRIO - KERN_TEST_CONFIG_HIGHEST_PRIO + 1)
|
|
#if TEST_LEVEL_COUNTS <= RT_CPUS_NR
|
|
#warning for the best of this test, TEST_LEVEL_COUNTS should greater than RT_CPUS_NR
|
|
#endif
|
|
#if KERN_TEST_CONCURRENT_THREADS < RT_CPUS_NR
|
|
#warning for the best of this test, KERN_TEST_CONCURRENT_THREADS should greater than RT_CPUS_NR
|
|
#endif
|
|
#if KERN_TEST_CONFIG_LOWEST_PRIO >= RT_THREAD_PRIORITY_MAX - 1
|
|
#error the thread priority should at least be greater than idle
|
|
#endif
|
|
|
|
static rt_atomic_t _star_counter;
|
|
static struct rt_semaphore _thr_exit_sem;
|
|
static struct rt_semaphore _level_waiting[TEST_LEVEL_COUNTS];
|
|
static rt_thread_t _thread_matrix[TEST_LEVEL_COUNTS][KERN_TEST_CONCURRENT_THREADS];
|
|
static rt_atomic_t _load_average[RT_CPUS_NR];
|
|
|
|
static void _print_char(rt_thread_t thr_self, int character)
|
|
{
|
|
rt_base_t current_counter;
|
|
|
|
#ifdef RT_USING_SMP
|
|
rt_kprintf("%c%d", character, RT_SCHED_CTX(thr_self).oncpu);
|
|
#else
|
|
rt_kprintf("%c0", character);
|
|
#endif /* RT_USING_SMP */
|
|
|
|
current_counter = rt_atomic_add(&_star_counter, 1);
|
|
if (current_counter % 30 == 0)
|
|
{
|
|
rt_kprintf("\n");
|
|
}
|
|
}
|
|
|
|
static void _stats_load_avg_inc(void)
|
|
{
|
|
int cpuid;
|
|
|
|
cpuid = rt_hw_cpu_id();
|
|
rt_atomic_add(&_load_average[cpuid], 1);
|
|
}
|
|
|
|
static void _stats_load_avg_print(void)
|
|
{
|
|
rt_base_t counts = 0;
|
|
const rt_base_t total_test_counts = KERN_TEST_CONFIG_LOOP_TIMES * TEST_LEVEL_COUNTS * KERN_TEST_CONCURRENT_THREADS;
|
|
|
|
for (size_t i = 0; i < RT_CPUS_NR; i++)
|
|
{
|
|
rt_kprintf("%ld ", _load_average[i]);
|
|
counts += _load_average[i];
|
|
}
|
|
|
|
rt_kprintf("\n");
|
|
uassert_int_equal(counts, total_test_counts);
|
|
}
|
|
|
|
static void _thread_entry(void *param)
|
|
{
|
|
int level = (rt_ubase_t)param;
|
|
rt_thread_t thr_self = rt_thread_self();
|
|
|
|
if (level == 0)
|
|
{
|
|
/* always the first to execute among other working threads */
|
|
for (size_t i = 0; i < KERN_TEST_CONFIG_LOOP_TIMES; i++)
|
|
{
|
|
/* notify our consumer */
|
|
rt_sem_release(&_level_waiting[level + 1]);
|
|
|
|
_stats_load_avg_inc();
|
|
|
|
/* waiting for resource of ours */
|
|
rt_sem_take(&_level_waiting[level], RT_WAITING_FOREVER);
|
|
}
|
|
}
|
|
else if (level == TEST_LEVEL_COUNTS - 1)
|
|
{
|
|
|
|
for (size_t i = 0; i < KERN_TEST_CONFIG_LOOP_TIMES; i++)
|
|
{
|
|
/* waiting for our resource first */
|
|
rt_sem_take(&_level_waiting[level], RT_WAITING_FOREVER);
|
|
|
|
_stats_load_avg_inc();
|
|
|
|
_print_char(thr_self, '*');
|
|
|
|
rt_thread_delay(1);
|
|
|
|
/* produce for level 0 worker */
|
|
rt_sem_release(&_level_waiting[0]);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (size_t i = 0; i < KERN_TEST_CONFIG_LOOP_TIMES; i++)
|
|
{
|
|
/* waiting for resource of ours */
|
|
rt_sem_take(&_level_waiting[level], RT_WAITING_FOREVER);
|
|
|
|
_stats_load_avg_inc();
|
|
|
|
/* notify our consumer */
|
|
rt_sem_release(&_level_waiting[level + 1]);
|
|
}
|
|
}
|
|
|
|
uassert_true(1);
|
|
rt_sem_release(&_thr_exit_sem);
|
|
|
|
return;
|
|
}
|
|
|
|
static void scheduler_tc(void)
|
|
{
|
|
LOG_I("Test starts...");
|
|
for (size_t i = 0; i < TEST_LEVEL_COUNTS; i++)
|
|
{
|
|
for (size_t j = 0; j < KERN_TEST_CONCURRENT_THREADS; j++)
|
|
{
|
|
rt_thread_startup(_thread_matrix[i][j]);
|
|
}
|
|
}
|
|
LOG_I("%d threads startup...", TEST_LEVEL_COUNTS * KERN_TEST_CONCURRENT_THREADS);
|
|
|
|
/* waiting for sub-threads to exit */
|
|
for (size_t i = 0; i < TEST_LEVEL_COUNTS * KERN_TEST_CONCURRENT_THREADS; i++)
|
|
{
|
|
rt_sem_take(&_thr_exit_sem, RT_WAITING_FOREVER);
|
|
}
|
|
|
|
/* print load average */
|
|
_stats_load_avg_print();
|
|
}
|
|
|
|
static rt_err_t utest_tc_init(void)
|
|
{
|
|
LOG_I("Setup environment...");
|
|
_star_counter = 1;
|
|
rt_memset(_load_average, 0, sizeof(_load_average));
|
|
rt_sem_init(&_thr_exit_sem, "test", 0, RT_IPC_FLAG_PRIO);
|
|
|
|
for (size_t i = 0; i < TEST_LEVEL_COUNTS; i++)
|
|
{
|
|
rt_sem_init(&_level_waiting[i], "test", 0, RT_IPC_FLAG_PRIO);
|
|
|
|
for (size_t j = 0; j < KERN_TEST_CONCURRENT_THREADS; j++)
|
|
{
|
|
_thread_matrix[i][j] =
|
|
rt_thread_create("test",
|
|
_thread_entry,
|
|
(void *)i,
|
|
UTEST_THR_STACK_SIZE,
|
|
KERN_TEST_CONFIG_HIGHEST_PRIO+i,
|
|
5);
|
|
if (!_thread_matrix[i][j])
|
|
uassert_not_null(_thread_matrix[i][j]);
|
|
}
|
|
}
|
|
return RT_EOK;
|
|
}
|
|
|
|
static rt_err_t utest_tc_cleanup(void)
|
|
{
|
|
rt_sem_detach(&_thr_exit_sem);
|
|
for (size_t i = 0; i < TEST_LEVEL_COUNTS; i++)
|
|
{
|
|
rt_sem_detach(&_level_waiting[i]);
|
|
}
|
|
return RT_EOK;
|
|
}
|
|
|
|
static void testcase(void)
|
|
{
|
|
UTEST_UNIT_RUN(scheduler_tc);
|
|
}
|
|
UTEST_TC_EXPORT(testcase, "testcases.kernel.scheduler.sem", utest_tc_init, utest_tc_cleanup, 10);
|