mirror of
https://github.com/adulau/aha.git
synced 2024-12-26 18:56:14 +00:00
rt_mutex: add proxy lock routines
This patch is a prerequisite for futex requeue_pi. It basically splits rt_mutex_slowlock() right down the middle, just before the first call to schedule(). It further adds helper functions which make use of the split and provide the rt-mutex preliminaries for futex requeue_pi. Signed-off-by: Darren Hart <dvhltc@us.ibm.com> Reviewed-by: Thomas Gleixner <tglx@linutronix.de> Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
This commit is contained in:
parent
dd9739980b
commit
8dac456a68
2 changed files with 219 additions and 77 deletions
288
kernel/rtmutex.c
288
kernel/rtmutex.c
|
@ -300,7 +300,8 @@ static int rt_mutex_adjust_prio_chain(struct task_struct *task,
|
||||||
* assigned pending owner [which might not have taken the
|
* assigned pending owner [which might not have taken the
|
||||||
* lock yet]:
|
* lock yet]:
|
||||||
*/
|
*/
|
||||||
static inline int try_to_steal_lock(struct rt_mutex *lock)
|
static inline int try_to_steal_lock(struct rt_mutex *lock,
|
||||||
|
struct task_struct *task)
|
||||||
{
|
{
|
||||||
struct task_struct *pendowner = rt_mutex_owner(lock);
|
struct task_struct *pendowner = rt_mutex_owner(lock);
|
||||||
struct rt_mutex_waiter *next;
|
struct rt_mutex_waiter *next;
|
||||||
|
@ -309,11 +310,11 @@ static inline int try_to_steal_lock(struct rt_mutex *lock)
|
||||||
if (!rt_mutex_owner_pending(lock))
|
if (!rt_mutex_owner_pending(lock))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
if (pendowner == current)
|
if (pendowner == task)
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
spin_lock_irqsave(&pendowner->pi_lock, flags);
|
spin_lock_irqsave(&pendowner->pi_lock, flags);
|
||||||
if (current->prio >= pendowner->prio) {
|
if (task->prio >= pendowner->prio) {
|
||||||
spin_unlock_irqrestore(&pendowner->pi_lock, flags);
|
spin_unlock_irqrestore(&pendowner->pi_lock, flags);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -338,21 +339,21 @@ static inline int try_to_steal_lock(struct rt_mutex *lock)
|
||||||
* We are going to steal the lock and a waiter was
|
* We are going to steal the lock and a waiter was
|
||||||
* enqueued on the pending owners pi_waiters queue. So
|
* enqueued on the pending owners pi_waiters queue. So
|
||||||
* we have to enqueue this waiter into
|
* we have to enqueue this waiter into
|
||||||
* current->pi_waiters list. This covers the case,
|
* task->pi_waiters list. This covers the case,
|
||||||
* where current is boosted because it holds another
|
* where task is boosted because it holds another
|
||||||
* lock and gets unboosted because the booster is
|
* lock and gets unboosted because the booster is
|
||||||
* interrupted, so we would delay a waiter with higher
|
* interrupted, so we would delay a waiter with higher
|
||||||
* priority as current->normal_prio.
|
* priority as task->normal_prio.
|
||||||
*
|
*
|
||||||
* Note: in the rare case of a SCHED_OTHER task changing
|
* Note: in the rare case of a SCHED_OTHER task changing
|
||||||
* its priority and thus stealing the lock, next->task
|
* its priority and thus stealing the lock, next->task
|
||||||
* might be current:
|
* might be task:
|
||||||
*/
|
*/
|
||||||
if (likely(next->task != current)) {
|
if (likely(next->task != task)) {
|
||||||
spin_lock_irqsave(¤t->pi_lock, flags);
|
spin_lock_irqsave(&task->pi_lock, flags);
|
||||||
plist_add(&next->pi_list_entry, ¤t->pi_waiters);
|
plist_add(&next->pi_list_entry, &task->pi_waiters);
|
||||||
__rt_mutex_adjust_prio(current);
|
__rt_mutex_adjust_prio(task);
|
||||||
spin_unlock_irqrestore(¤t->pi_lock, flags);
|
spin_unlock_irqrestore(&task->pi_lock, flags);
|
||||||
}
|
}
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
@ -389,7 +390,7 @@ static int try_to_take_rt_mutex(struct rt_mutex *lock)
|
||||||
*/
|
*/
|
||||||
mark_rt_mutex_waiters(lock);
|
mark_rt_mutex_waiters(lock);
|
||||||
|
|
||||||
if (rt_mutex_owner(lock) && !try_to_steal_lock(lock))
|
if (rt_mutex_owner(lock) && !try_to_steal_lock(lock, current))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/* We got the lock. */
|
/* We got the lock. */
|
||||||
|
@ -411,6 +412,7 @@ static int try_to_take_rt_mutex(struct rt_mutex *lock)
|
||||||
*/
|
*/
|
||||||
static int task_blocks_on_rt_mutex(struct rt_mutex *lock,
|
static int task_blocks_on_rt_mutex(struct rt_mutex *lock,
|
||||||
struct rt_mutex_waiter *waiter,
|
struct rt_mutex_waiter *waiter,
|
||||||
|
struct task_struct *task,
|
||||||
int detect_deadlock)
|
int detect_deadlock)
|
||||||
{
|
{
|
||||||
struct task_struct *owner = rt_mutex_owner(lock);
|
struct task_struct *owner = rt_mutex_owner(lock);
|
||||||
|
@ -418,21 +420,21 @@ static int task_blocks_on_rt_mutex(struct rt_mutex *lock,
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
int chain_walk = 0, res;
|
int chain_walk = 0, res;
|
||||||
|
|
||||||
spin_lock_irqsave(¤t->pi_lock, flags);
|
spin_lock_irqsave(&task->pi_lock, flags);
|
||||||
__rt_mutex_adjust_prio(current);
|
__rt_mutex_adjust_prio(task);
|
||||||
waiter->task = current;
|
waiter->task = task;
|
||||||
waiter->lock = lock;
|
waiter->lock = lock;
|
||||||
plist_node_init(&waiter->list_entry, current->prio);
|
plist_node_init(&waiter->list_entry, task->prio);
|
||||||
plist_node_init(&waiter->pi_list_entry, current->prio);
|
plist_node_init(&waiter->pi_list_entry, task->prio);
|
||||||
|
|
||||||
/* Get the top priority waiter on the lock */
|
/* Get the top priority waiter on the lock */
|
||||||
if (rt_mutex_has_waiters(lock))
|
if (rt_mutex_has_waiters(lock))
|
||||||
top_waiter = rt_mutex_top_waiter(lock);
|
top_waiter = rt_mutex_top_waiter(lock);
|
||||||
plist_add(&waiter->list_entry, &lock->wait_list);
|
plist_add(&waiter->list_entry, &lock->wait_list);
|
||||||
|
|
||||||
current->pi_blocked_on = waiter;
|
task->pi_blocked_on = waiter;
|
||||||
|
|
||||||
spin_unlock_irqrestore(¤t->pi_lock, flags);
|
spin_unlock_irqrestore(&task->pi_lock, flags);
|
||||||
|
|
||||||
if (waiter == rt_mutex_top_waiter(lock)) {
|
if (waiter == rt_mutex_top_waiter(lock)) {
|
||||||
spin_lock_irqsave(&owner->pi_lock, flags);
|
spin_lock_irqsave(&owner->pi_lock, flags);
|
||||||
|
@ -460,7 +462,7 @@ static int task_blocks_on_rt_mutex(struct rt_mutex *lock,
|
||||||
spin_unlock(&lock->wait_lock);
|
spin_unlock(&lock->wait_lock);
|
||||||
|
|
||||||
res = rt_mutex_adjust_prio_chain(owner, detect_deadlock, lock, waiter,
|
res = rt_mutex_adjust_prio_chain(owner, detect_deadlock, lock, waiter,
|
||||||
current);
|
task);
|
||||||
|
|
||||||
spin_lock(&lock->wait_lock);
|
spin_lock(&lock->wait_lock);
|
||||||
|
|
||||||
|
@ -605,6 +607,85 @@ void rt_mutex_adjust_pi(struct task_struct *task)
|
||||||
rt_mutex_adjust_prio_chain(task, 0, NULL, NULL, task);
|
rt_mutex_adjust_prio_chain(task, 0, NULL, NULL, task);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* __rt_mutex_slowlock() - Perform the wait-wake-try-to-take loop
|
||||||
|
* @lock: the rt_mutex to take
|
||||||
|
* @state: the state the task should block in (TASK_INTERRUPTIBLE
|
||||||
|
* or TASK_UNINTERRUPTIBLE)
|
||||||
|
* @timeout: the pre-initialized and started timer, or NULL for none
|
||||||
|
* @waiter: the pre-initialized rt_mutex_waiter
|
||||||
|
* @detect_deadlock: passed to task_blocks_on_rt_mutex
|
||||||
|
*
|
||||||
|
* lock->wait_lock must be held by the caller.
|
||||||
|
*/
|
||||||
|
static int __sched
|
||||||
|
__rt_mutex_slowlock(struct rt_mutex *lock, int state,
|
||||||
|
struct hrtimer_sleeper *timeout,
|
||||||
|
struct rt_mutex_waiter *waiter,
|
||||||
|
int detect_deadlock)
|
||||||
|
{
|
||||||
|
int ret = 0;
|
||||||
|
|
||||||
|
for (;;) {
|
||||||
|
/* Try to acquire the lock: */
|
||||||
|
if (try_to_take_rt_mutex(lock))
|
||||||
|
break;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* TASK_INTERRUPTIBLE checks for signals and
|
||||||
|
* timeout. Ignored otherwise.
|
||||||
|
*/
|
||||||
|
if (unlikely(state == TASK_INTERRUPTIBLE)) {
|
||||||
|
/* Signal pending? */
|
||||||
|
if (signal_pending(current))
|
||||||
|
ret = -EINTR;
|
||||||
|
if (timeout && !timeout->task)
|
||||||
|
ret = -ETIMEDOUT;
|
||||||
|
if (ret)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* waiter->task is NULL the first time we come here and
|
||||||
|
* when we have been woken up by the previous owner
|
||||||
|
* but the lock got stolen by a higher prio task.
|
||||||
|
*/
|
||||||
|
if (!waiter->task) {
|
||||||
|
ret = task_blocks_on_rt_mutex(lock, waiter, current,
|
||||||
|
detect_deadlock);
|
||||||
|
/*
|
||||||
|
* If we got woken up by the owner then start loop
|
||||||
|
* all over without going into schedule to try
|
||||||
|
* to get the lock now:
|
||||||
|
*/
|
||||||
|
if (unlikely(!waiter->task)) {
|
||||||
|
/*
|
||||||
|
* Reset the return value. We might
|
||||||
|
* have returned with -EDEADLK and the
|
||||||
|
* owner released the lock while we
|
||||||
|
* were walking the pi chain.
|
||||||
|
*/
|
||||||
|
ret = 0;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (unlikely(ret))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
spin_unlock(&lock->wait_lock);
|
||||||
|
|
||||||
|
debug_rt_mutex_print_deadlock(waiter);
|
||||||
|
|
||||||
|
if (waiter->task)
|
||||||
|
schedule_rt_mutex(lock);
|
||||||
|
|
||||||
|
spin_lock(&lock->wait_lock);
|
||||||
|
set_current_state(state);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Slow path lock function:
|
* Slow path lock function:
|
||||||
*/
|
*/
|
||||||
|
@ -636,62 +717,8 @@ rt_mutex_slowlock(struct rt_mutex *lock, int state,
|
||||||
timeout->task = NULL;
|
timeout->task = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (;;) {
|
ret = __rt_mutex_slowlock(lock, state, timeout, &waiter,
|
||||||
/* Try to acquire the lock: */
|
detect_deadlock);
|
||||||
if (try_to_take_rt_mutex(lock))
|
|
||||||
break;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* TASK_INTERRUPTIBLE checks for signals and
|
|
||||||
* timeout. Ignored otherwise.
|
|
||||||
*/
|
|
||||||
if (unlikely(state == TASK_INTERRUPTIBLE)) {
|
|
||||||
/* Signal pending? */
|
|
||||||
if (signal_pending(current))
|
|
||||||
ret = -EINTR;
|
|
||||||
if (timeout && !timeout->task)
|
|
||||||
ret = -ETIMEDOUT;
|
|
||||||
if (ret)
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* waiter.task is NULL the first time we come here and
|
|
||||||
* when we have been woken up by the previous owner
|
|
||||||
* but the lock got stolen by a higher prio task.
|
|
||||||
*/
|
|
||||||
if (!waiter.task) {
|
|
||||||
ret = task_blocks_on_rt_mutex(lock, &waiter,
|
|
||||||
detect_deadlock);
|
|
||||||
/*
|
|
||||||
* If we got woken up by the owner then start loop
|
|
||||||
* all over without going into schedule to try
|
|
||||||
* to get the lock now:
|
|
||||||
*/
|
|
||||||
if (unlikely(!waiter.task)) {
|
|
||||||
/*
|
|
||||||
* Reset the return value. We might
|
|
||||||
* have returned with -EDEADLK and the
|
|
||||||
* owner released the lock while we
|
|
||||||
* were walking the pi chain.
|
|
||||||
*/
|
|
||||||
ret = 0;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (unlikely(ret))
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
spin_unlock(&lock->wait_lock);
|
|
||||||
|
|
||||||
debug_rt_mutex_print_deadlock(&waiter);
|
|
||||||
|
|
||||||
if (waiter.task)
|
|
||||||
schedule_rt_mutex(lock);
|
|
||||||
|
|
||||||
spin_lock(&lock->wait_lock);
|
|
||||||
set_current_state(state);
|
|
||||||
}
|
|
||||||
|
|
||||||
set_current_state(TASK_RUNNING);
|
set_current_state(TASK_RUNNING);
|
||||||
|
|
||||||
|
@ -985,6 +1012,59 @@ void rt_mutex_proxy_unlock(struct rt_mutex *lock,
|
||||||
rt_mutex_deadlock_account_unlock(proxy_owner);
|
rt_mutex_deadlock_account_unlock(proxy_owner);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* rt_mutex_start_proxy_lock() - Start lock acquisition for another task
|
||||||
|
* @lock: the rt_mutex to take
|
||||||
|
* @waiter: the pre-initialized rt_mutex_waiter
|
||||||
|
* @task: the task to prepare
|
||||||
|
* @detect_deadlock: perform deadlock detection (1) or not (0)
|
||||||
|
*
|
||||||
|
* Returns:
|
||||||
|
* 0 - task blocked on lock
|
||||||
|
* 1 - acquired the lock for task, caller should wake it up
|
||||||
|
* <0 - error
|
||||||
|
*
|
||||||
|
* Special API call for FUTEX_REQUEUE_PI support.
|
||||||
|
*/
|
||||||
|
int rt_mutex_start_proxy_lock(struct rt_mutex *lock,
|
||||||
|
struct rt_mutex_waiter *waiter,
|
||||||
|
struct task_struct *task, int detect_deadlock)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
spin_lock(&lock->wait_lock);
|
||||||
|
|
||||||
|
mark_rt_mutex_waiters(lock);
|
||||||
|
|
||||||
|
if (!rt_mutex_owner(lock) || try_to_steal_lock(lock, task)) {
|
||||||
|
/* We got the lock for task. */
|
||||||
|
debug_rt_mutex_lock(lock);
|
||||||
|
|
||||||
|
rt_mutex_set_owner(lock, task, 0);
|
||||||
|
|
||||||
|
rt_mutex_deadlock_account_lock(lock, task);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = task_blocks_on_rt_mutex(lock, waiter, task, detect_deadlock);
|
||||||
|
|
||||||
|
|
||||||
|
if (ret && !waiter->task) {
|
||||||
|
/*
|
||||||
|
* Reset the return value. We might have
|
||||||
|
* returned with -EDEADLK and the owner
|
||||||
|
* released the lock while we were walking the
|
||||||
|
* pi chain. Let the waiter sort it out.
|
||||||
|
*/
|
||||||
|
ret = 0;
|
||||||
|
}
|
||||||
|
spin_unlock(&lock->wait_lock);
|
||||||
|
|
||||||
|
debug_rt_mutex_print_deadlock(waiter);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* rt_mutex_next_owner - return the next owner of the lock
|
* rt_mutex_next_owner - return the next owner of the lock
|
||||||
*
|
*
|
||||||
|
@ -1004,3 +1084,57 @@ struct task_struct *rt_mutex_next_owner(struct rt_mutex *lock)
|
||||||
|
|
||||||
return rt_mutex_top_waiter(lock)->task;
|
return rt_mutex_top_waiter(lock)->task;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* rt_mutex_finish_proxy_lock() - Complete lock acquisition
|
||||||
|
* @lock: the rt_mutex we were woken on
|
||||||
|
* @to: the timeout, null if none. hrtimer should already have
|
||||||
|
* been started.
|
||||||
|
* @waiter: the pre-initialized rt_mutex_waiter
|
||||||
|
* @detect_deadlock: perform deadlock detection (1) or not (0)
|
||||||
|
*
|
||||||
|
* Complete the lock acquisition started our behalf by another thread.
|
||||||
|
*
|
||||||
|
* Returns:
|
||||||
|
* 0 - success
|
||||||
|
* <0 - error, one of -EINTR, -ETIMEDOUT, or -EDEADLK
|
||||||
|
*
|
||||||
|
* Special API call for PI-futex requeue support
|
||||||
|
*/
|
||||||
|
int rt_mutex_finish_proxy_lock(struct rt_mutex *lock,
|
||||||
|
struct hrtimer_sleeper *to,
|
||||||
|
struct rt_mutex_waiter *waiter,
|
||||||
|
int detect_deadlock)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
spin_lock(&lock->wait_lock);
|
||||||
|
|
||||||
|
set_current_state(TASK_INTERRUPTIBLE);
|
||||||
|
|
||||||
|
ret = __rt_mutex_slowlock(lock, TASK_INTERRUPTIBLE, to, waiter,
|
||||||
|
detect_deadlock);
|
||||||
|
|
||||||
|
set_current_state(TASK_RUNNING);
|
||||||
|
|
||||||
|
if (unlikely(waiter->task))
|
||||||
|
remove_waiter(lock, waiter);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* try_to_take_rt_mutex() sets the waiter bit unconditionally. We might
|
||||||
|
* have to fix that up.
|
||||||
|
*/
|
||||||
|
fixup_rt_mutex_waiters(lock);
|
||||||
|
|
||||||
|
spin_unlock(&lock->wait_lock);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Readjust priority, when we did not get the lock. We might have been
|
||||||
|
* the pending owner and boosted. Since we did not take the lock, the
|
||||||
|
* PI boost has to go.
|
||||||
|
*/
|
||||||
|
if (unlikely(ret))
|
||||||
|
rt_mutex_adjust_prio(current);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
|
@ -120,6 +120,14 @@ extern void rt_mutex_init_proxy_locked(struct rt_mutex *lock,
|
||||||
struct task_struct *proxy_owner);
|
struct task_struct *proxy_owner);
|
||||||
extern void rt_mutex_proxy_unlock(struct rt_mutex *lock,
|
extern void rt_mutex_proxy_unlock(struct rt_mutex *lock,
|
||||||
struct task_struct *proxy_owner);
|
struct task_struct *proxy_owner);
|
||||||
|
extern int rt_mutex_start_proxy_lock(struct rt_mutex *lock,
|
||||||
|
struct rt_mutex_waiter *waiter,
|
||||||
|
struct task_struct *task,
|
||||||
|
int detect_deadlock);
|
||||||
|
extern int rt_mutex_finish_proxy_lock(struct rt_mutex *lock,
|
||||||
|
struct hrtimer_sleeper *to,
|
||||||
|
struct rt_mutex_waiter *waiter,
|
||||||
|
int detect_deadlock);
|
||||||
|
|
||||||
#ifdef CONFIG_DEBUG_RT_MUTEXES
|
#ifdef CONFIG_DEBUG_RT_MUTEXES
|
||||||
# include "rtmutex-debug.h"
|
# include "rtmutex-debug.h"
|
||||||
|
|
Loading…
Reference in a new issue