SylixOS 下优先级反转与解决方案
1、优先级反转概念
在系统中,有些资源必须是独占使用的,多个任务对这样的资源的并发访问将导致错误的发生。一般来说,对需要独占使用的资源必须使用互斥方法将对其的并发访问串行化。
在优先级多任务系统中引入互斥方案,会导致任务优先级反转的问题:假如某时低优先级的任务占有资源,然后又有高优先级的任务申请资源,但因为不能满足而被挂起了,即低优先级任务阻塞了高优先级任务的运行。假如这时又有一个中优先级任务,那么它会把低优先级任务抢占。最终高优先级任务会间接地被中优先级任务抢占了。这种现象叫作优先级反转。举例说明:
假如 A、D、C 三个任务优先级从高到低排列,任务 A 和 C 共享互斥信号量 R,如果某一时刻任务 C 已经获得互斥信号量 R,而任务 A 此时尝试占用 R,那么任务 A 会因为得不到 R 而阻塞在 R 的任务等待队列中。再假设此时任务 D 因为优先级高于任务 C 从而抢占了 C,进而长期占有处理器资源,那么就相当于中优先级的任务 D 间接阻塞了高优先级任务A的运行。
- T0 时刻,任务 C 处于运行状态,运行过程中,任务 C 获得了共享资源 R。
- T1 时刻,任务 A 就绪。由于任务 A 优先级高于任务 C,所以它抢占了任务 C,任务 A 被调度执行。
- T2 时刻,任务 A 需要共享资源 R,但R被更低优先级的任务 C 所拥有,所以任务 A 被阻塞等待该资源。任务 C得到执行。
- T3 时刻,此时任务 D 就绪,由于任务 D 优先级高于任务 C,所以它抢占了任务 C,任务 D 被调度执行。
从整个流程上看,T3 时刻,高优先级任务 A 被低优先级任务 D 间接地抢占了。此时优先级最高的任务 A 不仅要等任务 C 运行完,还要等优先级低的任务 D 运行完才能被调度,如果任务 D 和任务 C 需要执行很长时间,那么任务 A 的执行就不能得到保证,整个系统的实时性能很差。
优先级反转现象对基于优先级调度的实时系统有很大的影响。在基于优先级调度的系统中,处理器资源是按照优先级分配给任务的,就绪的高优先级任务必须实时获得处理器。系统中的各种资源,如果采用按照任务优先级分配的原则,那么高优先级的任务应该是首先被考虑的。优先级反转的问题将打乱这些原则。
2、解决方法
2.1 优先级继承
当一个任务占有了资源并且随后阻塞了其他申请该资源的任务时,该任务的优先级将临时改变为 “所有尝试申请该资源的任务中的最高优先级”,并以这个临时优先级在临界区执行。当任务释放资源后,则恢复它原有的优先级。
2.2 优先级天花板
将申请(占有)资源的任务的优先级提升到可能访问该资源的所有任务的最高优先级(这个最高优先级称为该资源的优先级天花板)。
2.3 代码详解
SylixOS 下,只有互斥锁(API_SemaphoreMPend),支持解决优先级反转问题。
ULONG API_SemaphoreMPend (LW_OBJECT_HANDLE ulId, ULONG ulTimeout)
{
......
_EventPrioTryBoost(pevent, ptcbCur); /* 尝试提升所属任务优先级 */
......
if (pevent->EVENT_ulOption & LW_OPTION_WAIT_PRIORITY) { /* 按优先级等待 */
_EVENT_INDEX_Q_PRIORITY(ptcbCur->TCB_ucPriority, ucPriorityIndex);
_EVENT_PRIORITY_Q_PTR(EVENT_SEM_Q, ppringList, ucPriorityIndex);
ptcbCur->TCB_ppringPriorityQueue = ppringList; /* 记录等待队列位置 */
_EventWaitPriority(ptcbCur, pevent, ppringList); /* 加入优先级等待表 */
} else { /* 按 FIFO 等待 */
_EVENT_FIFO_Q_PTR(EVENT_SEM_Q, ppringList); /* 确定 FIFO 队列的位置 */
_EventWaitFifo(ptcbCur, pevent, ppringList); /* 加入 FIFO 等待表 */
}
......
}
函数入参:
- ulId:当前互斥锁句柄
- ulTimeout:等待时间(是否无穷等待)
_EventPrioTryBoost 接口中实现了 LW_OPTION_INHERIT_PRIORITY
、LW_OPTION_PRIORITY_CEILING
两种解决优先级翻转的方法。由代码可见,两种方法,修改的是当前互斥锁持有者的优先级
VOID _EventPrioTryBoost (PLW_CLASS_EVENT pevent, PLW_CLASS_TCB ptcbCur)
{
PLW_CLASS_TCB ptcbOwner = (PLW_CLASS_TCB)pevent->EVENT_pvTcbOwn; /* 获取互斥锁持有者的线程句柄 */
if (ptcbOwner->TCB_iDeleteProcStatus) { /* 任务已被删除或正在被删除 */
return;
}
/*
* 如果当前函数调用者 ptcbCur(也就是尝试获取互斥锁的线程)线程优先级,大于 “锁持有者” ptcbOwner 的线程优先级
* 说明就需要提高 “锁持有者” ptcbOwner 的线程优先级
* 优先级继承算法:将 “锁持有者” ptcbOwner 的线程优先级提高成 “函数调用者” ptcbCur 线程优先级
* 优先级天花板算法:将 “锁持有者” ptcbOwner 的线程优先级提高一个固定优先级 EVENT_ucCeilingPriority
*/
if (LW_PRIO_IS_HIGH(ptcbCur->TCB_ucPriority,
ptcbOwner->TCB_ucPriority)) { /* 需要改变优先级 */
if (pevent->EVENT_ulOption & LW_OPTION_INHERIT_PRIORITY) { /* 优先级继承 */
_SchedSetPrio(ptcbOwner, ptcbCur->TCB_ucPriority);
} else if (LW_PRIO_IS_HIGH(pevent->EVENT_ucCeilingPriority,
ptcbOwner->TCB_ucPriority)) { /* 优先级天花板 */
_SchedSetPrio(ptcbOwner, pevent->EVENT_ucCeilingPriority);
}
}
}
函数入参:
- pevent:事件句柄(可以把它简单理解为互斥锁句柄)
- ptcbCur:当前线程句柄(也就是 API_SemaphoreMPend 调用者线程句柄)