diff --git a/include/rtsched.h b/include/rtsched.h index d650e09e3790..38b6f5d3e4b7 100644 --- a/include/rtsched.h +++ b/include/rtsched.h @@ -168,6 +168,7 @@ rt_err_t rt_sched_thread_close(struct rt_thread *thread); rt_err_t rt_sched_thread_ready(struct rt_thread *thread); rt_err_t rt_sched_thread_suspend(struct rt_thread *thread, rt_sched_lock_level_t level); rt_err_t rt_sched_thread_change_priority(struct rt_thread *thread, rt_uint8_t priority); +rt_err_t rt_sched_thread_change_curr_priority(struct rt_thread *thread, rt_uint8_t priority); rt_err_t rt_sched_thread_bind_cpu(struct rt_thread *thread, int cpu); rt_uint8_t rt_sched_thread_is_suspended(struct rt_thread *thread); rt_err_t rt_sched_thread_timer_stop(struct rt_thread *thread); diff --git a/src/ipc.c b/src/ipc.c index 5ffcf017c2ca..190a940e0316 100644 --- a/src/ipc.c +++ b/src/ipc.c @@ -873,7 +873,7 @@ rt_inline void _thread_update_priority(struct rt_thread *thread, rt_uint8_t prio LOG_D("thread:%s priority -> %d", thread->parent.name, priority); /* change priority of the thread */ - ret = rt_sched_thread_change_priority(thread, priority); + ret = rt_sched_thread_change_curr_priority(thread, priority); while ((ret == RT_EOK) && rt_sched_thread_is_suspended(thread)) { @@ -904,7 +904,7 @@ rt_inline void _thread_update_priority(struct rt_thread *thread, rt_uint8_t prio { thread = pending_mutex->owner; - ret = rt_sched_thread_change_priority(thread, mutex_priority); + ret = rt_sched_thread_change_curr_priority(thread, mutex_priority); } else { @@ -931,7 +931,7 @@ static rt_bool_t _check_and_update_prio(rt_thread_t thread, rt_mutex_t mutex) /* get the highest priority in the taken list of thread */ priority = _thread_get_mutex_priority(thread); - rt_sched_thread_change_priority(thread, priority); + rt_sched_thread_change_curr_priority(thread, priority); /** * notify a pending reschedule. Since scheduler is locked, we will not diff --git a/src/scheduler_comm.c b/src/scheduler_comm.c index 4e5495e727cf..73807e8241a4 100644 --- a/src/scheduler_comm.c +++ b/src/scheduler_comm.c @@ -180,7 +180,7 @@ rt_err_t rt_sched_tick_increase(rt_tick_t tick) /** * @brief Update priority of the target thread */ -rt_err_t rt_sched_thread_change_priority(struct rt_thread *thread, rt_uint8_t priority) +rt_err_t rt_sched_thread_change_curr_priority(struct rt_thread *thread, rt_uint8_t priority) { RT_ASSERT(priority < RT_THREAD_PRIORITY_MAX); RT_SCHED_DEBUG_IS_LOCKED; @@ -224,6 +224,60 @@ rt_err_t rt_sched_thread_change_priority(struct rt_thread *thread, rt_uint8_t pr return RT_EOK; } +/** + * @brief Update priority of the target thread + */ +rt_err_t rt_sched_thread_change_priority(struct rt_thread *thread, rt_uint8_t priority) +{ + RT_ASSERT(priority < RT_THREAD_PRIORITY_MAX); + RT_SCHED_DEBUG_IS_LOCKED; + + /* change thread init priority */ + RT_SCHED_PRIV(thread).init_priority = priority; + + /* if this thread takes mutex, its current prio is controlled by mutex */ + if (rt_list_isempty(&thread->taken_object_list) && (rt_object_get_type(thread->pending_object) == RT_Object_Class_Mutex)) + { + /* for ready thread, change queue; otherwise simply update the priority */ + if ((RT_SCHED_CTX(thread).stat & RT_THREAD_STAT_MASK) == RT_THREAD_READY) + { + /* remove thread from schedule queue first */ + rt_sched_remove_thread(thread); + + /* change thread priority */ + RT_SCHED_PRIV(thread).current_priority = priority; + + /* recalculate priority attribute */ +#if RT_THREAD_PRIORITY_MAX > 32 + RT_SCHED_PRIV(thread).number = RT_SCHED_PRIV(thread).current_priority >> 3; /* 5bit */ + RT_SCHED_PRIV(thread).number_mask = 1 << RT_SCHED_PRIV(thread).number; + RT_SCHED_PRIV(thread).high_mask = 1 << (RT_SCHED_PRIV(thread).current_priority & 0x07); /* 3bit */ +#else + RT_SCHED_PRIV(thread).number_mask = 1 << RT_SCHED_PRIV(thread).current_priority; +#endif /* RT_THREAD_PRIORITY_MAX > 32 */ + RT_SCHED_CTX(thread).stat = RT_THREAD_INIT; + + /* insert thread to schedule queue again */ + rt_sched_insert_thread(thread); + } + else + { + RT_SCHED_PRIV(thread).current_priority = priority; + + /* recalculate priority attribute */ +#if RT_THREAD_PRIORITY_MAX > 32 + RT_SCHED_PRIV(thread).number = RT_SCHED_PRIV(thread).current_priority >> 3; /* 5bit */ + RT_SCHED_PRIV(thread).number_mask = 1 << RT_SCHED_PRIV(thread).number; + RT_SCHED_PRIV(thread).high_mask = 1 << (RT_SCHED_PRIV(thread).current_priority & 0x07); /* 3bit */ +#else + RT_SCHED_PRIV(thread).number_mask = 1 << RT_SCHED_PRIV(thread).current_priority; +#endif /* RT_THREAD_PRIORITY_MAX > 32 */ + } + } + + return RT_EOK; +} + #ifdef RT_USING_OVERFLOW_CHECK void rt_scheduler_stack_check(struct rt_thread *thread) {