[POWERPC] spu sched: make addition to stop_wq and runque atomic vs wakeup
Addition to stop_wq needs to happen before adding to the runqeueue and under the same lock so that we don't have a race window for a lost wake up in the spu scheduler. Signed-off-by: Luke Browning <lukebrowning@us.ibm.com> Signed-off-by: Christoph Hellwig <hch@lst.de> Signed-off-by: Arnd Bergmann <arnd.bergmann@de.ibm.com>
This commit is contained in:
Родитель
7ec18ab923
Коммит
4e0f4ed0df
|
@ -236,44 +236,40 @@ static void spu_unbind_context(struct spu *spu, struct spu_context *ctx)
|
||||||
* spu_add_to_rq - add a context to the runqueue
|
* spu_add_to_rq - add a context to the runqueue
|
||||||
* @ctx: context to add
|
* @ctx: context to add
|
||||||
*/
|
*/
|
||||||
static void spu_add_to_rq(struct spu_context *ctx)
|
static void __spu_add_to_rq(struct spu_context *ctx)
|
||||||
{
|
{
|
||||||
spin_lock(&spu_prio->runq_lock);
|
int prio = ctx->prio;
|
||||||
list_add_tail(&ctx->rq, &spu_prio->runq[ctx->prio]);
|
|
||||||
set_bit(ctx->prio, spu_prio->bitmap);
|
list_add_tail(&ctx->rq, &spu_prio->runq[prio]);
|
||||||
mb();
|
set_bit(prio, spu_prio->bitmap);
|
||||||
spin_unlock(&spu_prio->runq_lock);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void __spu_del_from_rq(struct spu_context *ctx, int prio)
|
static void __spu_del_from_rq(struct spu_context *ctx)
|
||||||
{
|
{
|
||||||
|
int prio = ctx->prio;
|
||||||
|
|
||||||
if (!list_empty(&ctx->rq))
|
if (!list_empty(&ctx->rq))
|
||||||
list_del_init(&ctx->rq);
|
list_del_init(&ctx->rq);
|
||||||
if (list_empty(&spu_prio->runq[prio]))
|
if (list_empty(&spu_prio->runq[prio]))
|
||||||
clear_bit(ctx->prio, spu_prio->bitmap);
|
clear_bit(prio, spu_prio->bitmap);
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* spu_del_from_rq - remove a context from the runqueue
|
|
||||||
* @ctx: context to remove
|
|
||||||
*/
|
|
||||||
static void spu_del_from_rq(struct spu_context *ctx)
|
|
||||||
{
|
|
||||||
spin_lock(&spu_prio->runq_lock);
|
|
||||||
__spu_del_from_rq(ctx, ctx->prio);
|
|
||||||
spin_unlock(&spu_prio->runq_lock);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void spu_prio_wait(struct spu_context *ctx)
|
static void spu_prio_wait(struct spu_context *ctx)
|
||||||
{
|
{
|
||||||
DEFINE_WAIT(wait);
|
DEFINE_WAIT(wait);
|
||||||
|
|
||||||
|
spin_lock(&spu_prio->runq_lock);
|
||||||
prepare_to_wait_exclusive(&ctx->stop_wq, &wait, TASK_INTERRUPTIBLE);
|
prepare_to_wait_exclusive(&ctx->stop_wq, &wait, TASK_INTERRUPTIBLE);
|
||||||
if (!signal_pending(current)) {
|
if (!signal_pending(current)) {
|
||||||
|
__spu_add_to_rq(ctx);
|
||||||
|
spin_unlock(&spu_prio->runq_lock);
|
||||||
mutex_unlock(&ctx->state_mutex);
|
mutex_unlock(&ctx->state_mutex);
|
||||||
schedule();
|
schedule();
|
||||||
mutex_lock(&ctx->state_mutex);
|
mutex_lock(&ctx->state_mutex);
|
||||||
|
spin_lock(&spu_prio->runq_lock);
|
||||||
|
__spu_del_from_rq(ctx);
|
||||||
}
|
}
|
||||||
|
spin_unlock(&spu_prio->runq_lock);
|
||||||
__set_current_state(TASK_RUNNING);
|
__set_current_state(TASK_RUNNING);
|
||||||
remove_wait_queue(&ctx->stop_wq, &wait);
|
remove_wait_queue(&ctx->stop_wq, &wait);
|
||||||
}
|
}
|
||||||
|
@ -300,7 +296,7 @@ static void spu_reschedule(struct spu *spu)
|
||||||
BUG_ON(list_empty(rq));
|
BUG_ON(list_empty(rq));
|
||||||
|
|
||||||
ctx = list_entry(rq->next, struct spu_context, rq);
|
ctx = list_entry(rq->next, struct spu_context, rq);
|
||||||
__spu_del_from_rq(ctx, best);
|
__spu_del_from_rq(ctx);
|
||||||
wake_up(&ctx->stop_wq);
|
wake_up(&ctx->stop_wq);
|
||||||
}
|
}
|
||||||
spin_unlock(&spu_prio->runq_lock);
|
spin_unlock(&spu_prio->runq_lock);
|
||||||
|
@ -427,9 +423,7 @@ int spu_activate(struct spu_context *ctx, unsigned long flags)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
spu_add_to_rq(ctx);
|
|
||||||
spu_prio_wait(ctx);
|
spu_prio_wait(ctx);
|
||||||
spu_del_from_rq(ctx);
|
|
||||||
} while (!signal_pending(current));
|
} while (!signal_pending(current));
|
||||||
|
|
||||||
return -ERESTARTSYS;
|
return -ERESTARTSYS;
|
||||||
|
|
Загрузка…
Ссылка в новой задаче