Merge branch 'RT-Thread:master' into master

This commit is contained in:
WaterFishJ 2021-08-20 09:48:16 +08:00 committed by GitHub
commit 3191414610
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 166 additions and 41 deletions

View File

@ -46,12 +46,17 @@ static rt_uint8_t utest_log_lv = UTEST_LOG_ALL;
static utest_tc_export_t tc_table = RT_NULL; static utest_tc_export_t tc_table = RT_NULL;
static rt_size_t tc_num; static rt_size_t tc_num;
static rt_uint32_t tc_loop; static rt_uint32_t tc_loop;
static rt_uint8_t *tc_fail_list;
static struct utest local_utest = {UTEST_PASSED, 0, 0}; static struct utest local_utest = {UTEST_PASSED, 0, 0};
#if defined(__ICCARM__) || defined(__ICCRX__) /* for IAR compiler */ #if defined(__ICCARM__) || defined(__ICCRX__) /* for IAR compiler */
#pragma section="UtestTcTab" #pragma section="UtestTcTab"
#endif #endif
#define TC_FAIL_LIST_SIZE (RT_ALIGN(tc_num, 8) / 8)
#define TC_FAIL_LIST_MARK_FAILED(index) (tc_fail_list[RT_ALIGN(index, 8) / 8] |= (1UL << (index % 8)))
#define TC_FAIL_LIST_IS_FAILED(index) (tc_fail_list[RT_ALIGN(index, 8) / 8] & (1UL << (index % 8)))
void utest_log_lv_set(rt_uint8_t lv) void utest_log_lv_set(rt_uint8_t lv)
{ {
if (lv == UTEST_LOG_ALL || lv == UTEST_LOG_ASSERT) if (lv == UTEST_LOG_ALL || lv == UTEST_LOG_ASSERT)
@ -80,6 +85,14 @@ int utest_init(void)
LOG_I("utest is initialize success."); LOG_I("utest is initialize success.");
LOG_I("total utest testcase num: (%d)", tc_num); LOG_I("total utest testcase num: (%d)", tc_num);
if (tc_num > 0)
{
tc_fail_list = rt_malloc(TC_FAIL_LIST_SIZE);
if(!tc_fail_list)
{
LOG_E("no memory, tc_fail_list init failed!");
}
}
return tc_num; return tc_num;
} }
INIT_COMPONENT_EXPORT(utest_init); INIT_COMPONENT_EXPORT(utest_init);
@ -147,6 +160,8 @@ static void utest_run(const char *utest_name)
rt_size_t i; rt_size_t i;
rt_uint32_t index; rt_uint32_t index;
rt_bool_t is_find; rt_bool_t is_find;
rt_uint32_t tc_fail_num = 0;
rt_uint32_t tc_run_num = 0;
rt_thread_mdelay(1000); rt_thread_mdelay(1000);
@ -154,6 +169,14 @@ static void utest_run(const char *utest_name)
{ {
i = 0; i = 0;
is_find = RT_FALSE; is_find = RT_FALSE;
tc_fail_num = 0;
tc_run_num = 0;
if (tc_fail_list)
{
memset(tc_fail_list, 0, TC_FAIL_LIST_SIZE);
}
LOG_I("[==========] [ utest ] loop %d/%d", index + 1, tc_loop); LOG_I("[==========] [ utest ] loop %d/%d", index + 1, tc_loop);
LOG_I("[==========] [ utest ] started"); LOG_I("[==========] [ utest ] started");
while(i < tc_num) while(i < tc_num)
@ -192,6 +215,8 @@ static void utest_run(const char *utest_name)
} }
else else
{ {
TC_FAIL_LIST_MARK_FAILED(i);
tc_fail_num ++;
LOG_E("[ FAILED ] [ result ] testcase (%s)", tc_table[i].name); LOG_E("[ FAILED ] [ result ] testcase (%s)", tc_table[i].name);
} }
} }
@ -212,6 +237,7 @@ static void utest_run(const char *utest_name)
__tc_continue: __tc_continue:
LOG_I("[----------] [ testcase ] (%s) finished", tc_table[i].name); LOG_I("[----------] [ testcase ] (%s) finished", tc_table[i].name);
tc_run_num ++;
i++; i++;
} }
@ -223,6 +249,20 @@ static void utest_run(const char *utest_name)
} }
LOG_I("[==========] [ utest ] finished"); LOG_I("[==========] [ utest ] finished");
LOG_I("[==========] [ utest ] %d tests from %d testcase ran.", tc_run_num, tc_num);
LOG_I("[ PASSED ] [ result ] %d tests.", tc_run_num - tc_fail_num);
if(tc_fail_list && (tc_fail_num > 0))
{
LOG_E("[ FAILED ] [ result ] %d tests, listed below:", tc_fail_num);
for(i = 0; i < tc_num; i ++)
{
if (TC_FAIL_LIST_IS_FAILED(i))
{
LOG_E("[ FAILED ] [ result ] %s", tc_table[i].name);
}
}
}
} }
} }

View File

@ -5,6 +5,10 @@ config UTEST_MEMHEAP_TC
default y default y
depends on RT_USING_MEMHEAP depends on RT_USING_MEMHEAP
config UTEST_IRQ_TC
bool "IRQ test"
default n
config UTEST_SEMAPHORE_TC config UTEST_SEMAPHORE_TC
bool "semaphore test" bool "semaphore test"
default n default n

View File

@ -8,6 +8,9 @@ src = Split('''
if GetDepend(['UTEST_MEMHEAP_TC']): if GetDepend(['UTEST_MEMHEAP_TC']):
src += ['memheap_tc.c'] src += ['memheap_tc.c']
if GetDepend(['UTEST_IRQ_TC']):
src += ['irq_tc.c']
if GetDepend(['UTEST_SEMAPHORE_TC']): if GetDepend(['UTEST_SEMAPHORE_TC']):
src += ['semaphore_tc.c'] src += ['semaphore_tc.c']

View File

@ -0,0 +1,78 @@
/*
* Copyright (c) 2006-2021, RT-Thread Development Team
*
* SPDX-License-Identifier: Apache-2.0
*
* Change Logs:
* Date Author Notes
* 2021-08-15 supperthomas add irq_test
*/
#include <rtthread.h>
#include "utest.h"
#include "rthw.h"
#define UTEST_NAME "irq_tc"
static uint32_t irq_count = 0;
static uint32_t max_get_nest_count = 0;
static void irq_callback()
{
if(rt_interrupt_get_nest() > max_get_nest_count)
{
max_get_nest_count = rt_interrupt_get_nest();
}
irq_count ++;
}
static void irq_test(void)
{
irq_count = 0;
rt_interrupt_enter_sethook(irq_callback);
rt_interrupt_leave_sethook(irq_callback);
rt_thread_mdelay(2);
LOG_D("%s test irq_test! irq_count %d max_get_nest_count %d\n", UTEST_NAME, irq_count, max_get_nest_count);
uassert_int_not_equal(0, irq_count);
uassert_int_not_equal(0, max_get_nest_count);
rt_interrupt_enter_sethook(RT_NULL);
rt_interrupt_leave_sethook(RT_NULL);
LOG_D("irq_test OK!\n");
}
static rt_err_t utest_tc_init(void)
{
irq_count = 0;
max_get_nest_count = 0;
return RT_EOK;
}
static rt_err_t utest_tc_cleanup(void)
{
return RT_EOK;
}
static void interrupt_test(void)
{
rt_base_t level;
uint32_t i = 1000;
rt_interrupt_enter_sethook(irq_callback);
rt_interrupt_leave_sethook(irq_callback);
irq_count = 0;
level = rt_hw_interrupt_disable();
while(i)
{
i --;
}
uassert_int_equal(0, irq_count);
rt_hw_interrupt_enable(level);
rt_interrupt_enter_sethook(RT_NULL);
rt_interrupt_leave_sethook(RT_NULL);
}
static void testcase(void)
{
UTEST_UNIT_RUN(irq_test);
UTEST_UNIT_RUN(interrupt_test);
}
UTEST_TC_EXPORT(testcase, "testcases.kernel.irq_tc", utest_tc_init, utest_tc_cleanup, 10);

View File

@ -23,7 +23,7 @@
#include <rthw.h> #include <rthw.h>
/* hard timer list */ /* hard timer list */
static rt_list_t rt_timer_list[RT_TIMER_SKIP_LIST_LEVEL]; static rt_list_t _timer_list[RT_TIMER_SKIP_LIST_LEVEL];
#ifdef RT_USING_TIMER_SOFT #ifdef RT_USING_TIMER_SOFT
@ -39,12 +39,12 @@ static rt_list_t rt_timer_list[RT_TIMER_SKIP_LIST_LEVEL];
#endif /* RT_TIMER_THREAD_PRIO */ #endif /* RT_TIMER_THREAD_PRIO */
/* soft timer status */ /* soft timer status */
static rt_uint8_t soft_timer_status = RT_SOFT_TIMER_IDLE; static rt_uint8_t _soft_timer_status = RT_SOFT_TIMER_IDLE;
/* soft timer list */ /* soft timer list */
static rt_list_t rt_soft_timer_list[RT_TIMER_SKIP_LIST_LEVEL]; static rt_list_t _soft_timer_list[RT_TIMER_SKIP_LIST_LEVEL];
static struct rt_thread timer_thread; static struct rt_thread _timer_thread;
ALIGN(RT_ALIGN_SIZE) ALIGN(RT_ALIGN_SIZE)
static rt_uint8_t timer_thread_stack[RT_TIMER_THREAD_STACK_SIZE]; static rt_uint8_t _timer_thread_stack[RT_TIMER_THREAD_STACK_SIZE];
#endif /* RT_USING_TIMER_SOFT */ #endif /* RT_USING_TIMER_SOFT */
#ifdef RT_USING_HOOK #ifdef RT_USING_HOOK
@ -98,7 +98,7 @@ void rt_timer_exit_sethook(void (*hook)(struct rt_timer *timer))
* @param time the tick of timer * @param time the tick of timer
* @param flag the flag of timer * @param flag the flag of timer
*/ */
static void _rt_timer_init(rt_timer_t timer, static void _timer_init(rt_timer_t timer,
void (*timeout)(void *parameter), void (*timeout)(void *parameter),
void *parameter, void *parameter,
rt_tick_t time, rt_tick_t time,
@ -132,7 +132,7 @@ static void _rt_timer_init(rt_timer_t timer,
* *
* @return rt_tick_t the point of timer * @return rt_tick_t the point of timer
*/ */
static rt_tick_t rt_timer_list_next_timeout(rt_list_t timer_list[]) static rt_tick_t _timer_list_next_timeout(rt_list_t timer_list[])
{ {
struct rt_timer *timer; struct rt_timer *timer;
register rt_base_t level; register rt_base_t level;
@ -159,7 +159,7 @@ static rt_tick_t rt_timer_list_next_timeout(rt_list_t timer_list[])
* *
* @param timer the point of timer * @param timer the point of timer
*/ */
rt_inline void _rt_timer_remove(rt_timer_t timer) rt_inline void _timer_remove(rt_timer_t timer)
{ {
int i; int i;
@ -176,7 +176,7 @@ rt_inline void _rt_timer_remove(rt_timer_t timer)
* @param timer * @param timer
* @return int the count * @return int the count
*/ */
static int rt_timer_count_height(struct rt_timer *timer) static int _timer_count_height(struct rt_timer *timer)
{ {
int i, cnt = 0; int i, cnt = 0;
@ -203,7 +203,7 @@ void rt_timer_dump(rt_list_t timer_heads[])
struct rt_timer *timer = rt_list_entry(list, struct rt_timer *timer = rt_list_entry(list,
struct rt_timer, struct rt_timer,
row[RT_TIMER_SKIP_LIST_LEVEL - 1]); row[RT_TIMER_SKIP_LIST_LEVEL - 1]);
rt_kprintf("%d", rt_timer_count_height(timer)); rt_kprintf("%d", _timer_count_height(timer));
} }
rt_kprintf("\n"); rt_kprintf("\n");
} }
@ -238,7 +238,7 @@ void rt_timer_init(rt_timer_t timer,
/* timer object initialization */ /* timer object initialization */
rt_object_init(&(timer->parent), RT_Object_Class_Timer, name); rt_object_init(&(timer->parent), RT_Object_Class_Timer, name);
_rt_timer_init(timer, timeout, parameter, time, flag); _timer_init(timer, timeout, parameter, time, flag);
} }
RTM_EXPORT(rt_timer_init); RTM_EXPORT(rt_timer_init);
@ -260,7 +260,7 @@ rt_err_t rt_timer_detach(rt_timer_t timer)
/* disable interrupt */ /* disable interrupt */
level = rt_hw_interrupt_disable(); level = rt_hw_interrupt_disable();
_rt_timer_remove(timer); _timer_remove(timer);
/* stop timer */ /* stop timer */
timer->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED; timer->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED;
@ -300,7 +300,7 @@ rt_timer_t rt_timer_create(const char *name,
return RT_NULL; return RT_NULL;
} }
_rt_timer_init(timer, timeout, parameter, time, flag); _timer_init(timer, timeout, parameter, time, flag);
return timer; return timer;
} }
@ -325,7 +325,7 @@ rt_err_t rt_timer_delete(rt_timer_t timer)
/* disable interrupt */ /* disable interrupt */
level = rt_hw_interrupt_disable(); level = rt_hw_interrupt_disable();
_rt_timer_remove(timer); _timer_remove(timer);
/* stop timer */ /* stop timer */
timer->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED; timer->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED;
@ -362,7 +362,7 @@ rt_err_t rt_timer_start(rt_timer_t timer)
/* stop timer firstly */ /* stop timer firstly */
level = rt_hw_interrupt_disable(); level = rt_hw_interrupt_disable();
/* remove timer from list */ /* remove timer from list */
_rt_timer_remove(timer); _timer_remove(timer);
/* change status of timer */ /* change status of timer */
timer->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED; timer->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED;
@ -379,13 +379,13 @@ rt_err_t rt_timer_start(rt_timer_t timer)
if (timer->parent.flag & RT_TIMER_FLAG_SOFT_TIMER) if (timer->parent.flag & RT_TIMER_FLAG_SOFT_TIMER)
{ {
/* insert timer to soft timer list */ /* insert timer to soft timer list */
timer_list = rt_soft_timer_list; timer_list = _soft_timer_list;
} }
else else
#endif /* RT_USING_TIMER_SOFT */ #endif /* RT_USING_TIMER_SOFT */
{ {
/* insert timer to system timer list */ /* insert timer to system timer list */
timer_list = rt_timer_list; timer_list = _timer_list;
} }
row_head[0] = &timer_list[0]; row_head[0] = &timer_list[0];
@ -448,11 +448,11 @@ rt_err_t rt_timer_start(rt_timer_t timer)
if (timer->parent.flag & RT_TIMER_FLAG_SOFT_TIMER) if (timer->parent.flag & RT_TIMER_FLAG_SOFT_TIMER)
{ {
/* check whether timer thread is ready */ /* check whether timer thread is ready */
if ((soft_timer_status == RT_SOFT_TIMER_IDLE) && if ((_soft_timer_status == RT_SOFT_TIMER_IDLE) &&
((timer_thread.stat & RT_THREAD_STAT_MASK) == RT_THREAD_SUSPEND)) ((_timer_thread.stat & RT_THREAD_STAT_MASK) == RT_THREAD_SUSPEND))
{ {
/* resume timer thread to check soft timer */ /* resume timer thread to check soft timer */
rt_thread_resume(&timer_thread); rt_thread_resume(&_timer_thread);
rt_schedule(); rt_schedule();
} }
} }
@ -485,7 +485,7 @@ rt_err_t rt_timer_stop(rt_timer_t timer)
/* disable interrupt */ /* disable interrupt */
level = rt_hw_interrupt_disable(); level = rt_hw_interrupt_disable();
_rt_timer_remove(timer); _timer_remove(timer);
/* change status */ /* change status */
timer->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED; timer->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED;
@ -576,9 +576,9 @@ void rt_timer_check(void)
/* disable interrupt */ /* disable interrupt */
level = rt_hw_interrupt_disable(); level = rt_hw_interrupt_disable();
while (!rt_list_isempty(&rt_timer_list[RT_TIMER_SKIP_LIST_LEVEL - 1])) while (!rt_list_isempty(&_timer_list[RT_TIMER_SKIP_LIST_LEVEL - 1]))
{ {
t = rt_list_entry(rt_timer_list[RT_TIMER_SKIP_LIST_LEVEL - 1].next, t = rt_list_entry(_timer_list[RT_TIMER_SKIP_LIST_LEVEL - 1].next,
struct rt_timer, row[RT_TIMER_SKIP_LIST_LEVEL - 1]); struct rt_timer, row[RT_TIMER_SKIP_LIST_LEVEL - 1]);
/* /*
@ -590,7 +590,7 @@ void rt_timer_check(void)
RT_OBJECT_HOOK_CALL(rt_timer_enter_hook, (t)); RT_OBJECT_HOOK_CALL(rt_timer_enter_hook, (t));
/* remove timer from timer list firstly */ /* remove timer from timer list firstly */
_rt_timer_remove(t); _timer_remove(t);
if (!(t->parent.flag & RT_TIMER_FLAG_PERIODIC)) if (!(t->parent.flag & RT_TIMER_FLAG_PERIODIC))
{ {
t->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED; t->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED;
@ -636,7 +636,7 @@ void rt_timer_check(void)
*/ */
rt_tick_t rt_timer_next_timeout_tick(void) rt_tick_t rt_timer_next_timeout_tick(void)
{ {
return rt_timer_list_next_timeout(rt_timer_list); return _timer_list_next_timeout(_timer_list);
} }
#ifdef RT_USING_TIMER_SOFT #ifdef RT_USING_TIMER_SOFT
@ -658,9 +658,9 @@ void rt_soft_timer_check(void)
/* disable interrupt */ /* disable interrupt */
level = rt_hw_interrupt_disable(); level = rt_hw_interrupt_disable();
while (!rt_list_isempty(&rt_soft_timer_list[RT_TIMER_SKIP_LIST_LEVEL - 1])) while (!rt_list_isempty(&_soft_timer_list[RT_TIMER_SKIP_LIST_LEVEL - 1]))
{ {
t = rt_list_entry(rt_soft_timer_list[RT_TIMER_SKIP_LIST_LEVEL - 1].next, t = rt_list_entry(_soft_timer_list[RT_TIMER_SKIP_LIST_LEVEL - 1].next,
struct rt_timer, row[RT_TIMER_SKIP_LIST_LEVEL - 1]); struct rt_timer, row[RT_TIMER_SKIP_LIST_LEVEL - 1]);
current_tick = rt_tick_get(); current_tick = rt_tick_get();
@ -674,7 +674,7 @@ void rt_soft_timer_check(void)
RT_OBJECT_HOOK_CALL(rt_timer_enter_hook, (t)); RT_OBJECT_HOOK_CALL(rt_timer_enter_hook, (t));
/* remove timer from timer list firstly */ /* remove timer from timer list firstly */
_rt_timer_remove(t); _timer_remove(t);
if (!(t->parent.flag & RT_TIMER_FLAG_PERIODIC)) if (!(t->parent.flag & RT_TIMER_FLAG_PERIODIC))
{ {
t->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED; t->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED;
@ -682,7 +682,7 @@ void rt_soft_timer_check(void)
/* add timer to temporary list */ /* add timer to temporary list */
rt_list_insert_after(&list, &(t->row[RT_TIMER_SKIP_LIST_LEVEL - 1])); rt_list_insert_after(&list, &(t->row[RT_TIMER_SKIP_LIST_LEVEL - 1]));
soft_timer_status = RT_SOFT_TIMER_BUSY; _soft_timer_status = RT_SOFT_TIMER_BUSY;
/* enable interrupt */ /* enable interrupt */
rt_hw_interrupt_enable(level); rt_hw_interrupt_enable(level);
@ -695,7 +695,7 @@ void rt_soft_timer_check(void)
/* disable interrupt */ /* disable interrupt */
level = rt_hw_interrupt_disable(); level = rt_hw_interrupt_disable();
soft_timer_status = RT_SOFT_TIMER_IDLE; _soft_timer_status = RT_SOFT_TIMER_IDLE;
/* Check whether the timer object is detached or started again */ /* Check whether the timer object is detached or started again */
if (rt_list_isempty(&list)) if (rt_list_isempty(&list))
{ {
@ -723,14 +723,14 @@ void rt_soft_timer_check(void)
* *
* @param parameter * @param parameter
*/ */
static void rt_thread_timer_entry(void *parameter) static void _timer_thread_entry(void *parameter)
{ {
rt_tick_t next_timeout; rt_tick_t next_timeout;
while (1) while (1)
{ {
/* get the next timeout tick */ /* get the next timeout tick */
next_timeout = rt_timer_list_next_timeout(rt_soft_timer_list); next_timeout = _timer_list_next_timeout(_soft_timer_list);
if (next_timeout == RT_TICK_MAX) if (next_timeout == RT_TICK_MAX)
{ {
/* no software timer exist, suspend self. */ /* no software timer exist, suspend self. */
@ -767,9 +767,9 @@ void rt_system_timer_init(void)
{ {
int i; int i;
for (i = 0; i < sizeof(rt_timer_list) / sizeof(rt_timer_list[0]); i++) for (i = 0; i < sizeof(_timer_list) / sizeof(_timer_list[0]); i++)
{ {
rt_list_init(rt_timer_list + i); rt_list_init(_timer_list + i);
} }
} }
@ -784,24 +784,24 @@ void rt_system_timer_thread_init(void)
int i; int i;
for (i = 0; for (i = 0;
i < sizeof(rt_soft_timer_list) / sizeof(rt_soft_timer_list[0]); i < sizeof(_soft_timer_list) / sizeof(_soft_timer_list[0]);
i++) i++)
{ {
rt_list_init(rt_soft_timer_list + i); rt_list_init(_soft_timer_list + i);
} }
/* start software timer thread */ /* start software timer thread */
rt_thread_init(&timer_thread, rt_thread_init(&_timer_thread,
"timer", "timer",
rt_thread_timer_entry, _timer_thread_entry,
RT_NULL, RT_NULL,
&timer_thread_stack[0], &_timer_thread_stack[0],
sizeof(timer_thread_stack), sizeof(_timer_thread_stack),
RT_TIMER_THREAD_PRIO, RT_TIMER_THREAD_PRIO,
10); 10);
/* startup */ /* startup */
rt_thread_startup(&timer_thread); rt_thread_startup(&_timer_thread);
#endif /* RT_USING_TIMER_SOFT */ #endif /* RT_USING_TIMER_SOFT */
} }