From 64cd64baf9ba04dfe5967431d97884d976fc87ab Mon Sep 17 00:00:00 2001 From: Shell Date: Mon, 13 Mar 2023 19:21:41 +0800 Subject: [PATCH] [mutex] mutex may suspend a thread already suspended & fix ceilingprio (#7046) * [fix] reinsert should not suspend twice * [mutex] fix prioceiling * [mutex] fix prioceiling --- src/ipc.c | 41 +++++++++++++++++++++++++++++------------ 1 file changed, 29 insertions(+), 12 deletions(-) diff --git a/src/ipc.c b/src/ipc.c index 0aafb2ea6..be220439b 100644 --- a/src/ipc.c +++ b/src/ipc.c @@ -127,12 +127,15 @@ rt_inline rt_err_t _ipc_list_suspend(rt_list_t *list, rt_uint8_t flag, int suspend_flag) { - rt_err_t ret = rt_thread_suspend_with_flag(thread, suspend_flag); - - /* suspend thread */ - if (ret != RT_EOK) + if ((thread->stat & RT_THREAD_SUSPEND_MASK) != RT_THREAD_SUSPEND_MASK) { - return ret; + rt_err_t ret = rt_thread_suspend_with_flag(thread, suspend_flag); + + /* suspend thread */ + if (ret != RT_EOK) + { + return ret; + } } switch (flag) @@ -740,6 +743,7 @@ RTM_EXPORT(rt_sem_control); #endif /* RT_USING_SEMAPHORE */ #ifdef RT_USING_MUTEX +/* iterate over each suspended thread to update highest priority in pending threads */ rt_inline rt_uint8_t _mutex_update_priority(struct rt_mutex *mutex) { struct rt_thread *thread; @@ -757,6 +761,7 @@ rt_inline rt_uint8_t _mutex_update_priority(struct rt_mutex *mutex) return mutex->priority; } +/* get highest priority inside its taken object and its init priority */ rt_inline rt_uint8_t _thread_get_mutex_priority(struct rt_thread* thread) { rt_list_t *node = RT_NULL; @@ -766,15 +771,20 @@ rt_inline rt_uint8_t _thread_get_mutex_priority(struct rt_thread* thread) rt_list_for_each(node, &(thread->taken_object_list)) { mutex = rt_list_entry(node, struct rt_mutex, taken_list); - if (priority > mutex->priority) + rt_uint8_t mutex_prio = mutex->priority; + /* prio at least be priority ceiling */ + mutex_prio = mutex_prio < mutex->ceiling_priority ? mutex_prio : mutex->ceiling_priority; + + if (priority > mutex_prio) { - priority = mutex->priority; + priority = mutex_prio; } } return priority; } +/* update priority of target thread and the thread suspended it if any */ rt_inline void _thread_update_priority(struct rt_thread *thread, rt_uint8_t priority, int suspend_flag) { rt_err_t ret; @@ -987,8 +997,17 @@ rt_uint8_t rt_mutex_setprioceiling(rt_mutex_t mutex, rt_uint8_t priority) if ((mutex) && (priority < RT_THREAD_PRIORITY_MAX)) { + /* critical section here if multiple updates to one mutex happen */ + rt_ubase_t level = rt_hw_interrupt_disable(); ret_priority = mutex->ceiling_priority; mutex->ceiling_priority = priority; + if (mutex->owner) + { + rt_uint8_t priority = _thread_get_mutex_priority(mutex->owner); + if (priority != mutex->owner->current_priority) + _thread_update_priority(mutex->owner, priority, RT_UNINTERRUPTIBLE); + } + rt_hw_interrupt_enable(level); } else { @@ -1197,11 +1216,9 @@ static rt_err_t _rt_mutex_take(rt_mutex_t mutex, rt_int32_t timeout, int suspend if (mutex->ceiling_priority < mutex->owner->current_priority) _thread_update_priority(mutex->owner, mutex->ceiling_priority, suspend_flag); } - else - { - /* insert mutex to thread's taken object list */ - rt_list_insert_after(&thread->taken_object_list, &mutex->taken_list); - } + + /* insert mutex to thread's taken object list */ + rt_list_insert_after(&thread->taken_object_list, &mutex->taken_list); } else { -- GitLab