aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorChristoph Hellwig <hch@lst.de>2007-04-23 15:08:11 -0400
committerArnd Bergmann <arnd@klappe.arndb.de>2007-04-23 15:18:54 -0400
commita475c2f43520cb095452201da57395000cfeb94c (patch)
tree0cfe49111b15cf20b03d308ec8b5a8b5210b0363
parent390c53430498c9973e015432806edd53b2efe6c6 (diff)
[POWERPC] spufs: remove woken threads from the runqueue early
A single context should only be woken once, and we should not have more wakeups for a given priority than the number of contexts on that runqueue position. Also add some asserts to trap future problems in this area more easily. Signed-off-by: Christoph Hellwig <hch@lst.de> Signed-off-by: Arnd Bergmann <arnd.bergmann@de.ibm.com>
-rw-r--r--arch/powerpc/platforms/cell/spufs/context.c2
-rw-r--r--arch/powerpc/platforms/cell/spufs/sched.c44
2 files changed, 19 insertions, 27 deletions
diff --git a/arch/powerpc/platforms/cell/spufs/context.c b/arch/powerpc/platforms/cell/spufs/context.c
index b3954aba424e..065147fb1cc2 100644
--- a/arch/powerpc/platforms/cell/spufs/context.c
+++ b/arch/powerpc/platforms/cell/spufs/context.c
@@ -52,6 +52,7 @@ struct spu_context *alloc_spu_context(struct spu_gang *gang)
52 ctx->state = SPU_STATE_SAVED; 52 ctx->state = SPU_STATE_SAVED;
53 ctx->ops = &spu_backing_ops; 53 ctx->ops = &spu_backing_ops;
54 ctx->owner = get_task_mm(current); 54 ctx->owner = get_task_mm(current);
55 INIT_LIST_HEAD(&ctx->rq);
55 if (gang) 56 if (gang)
56 spu_gang_add_ctx(gang, ctx); 57 spu_gang_add_ctx(gang, ctx);
57 ctx->rt_priority = current->rt_priority; 58 ctx->rt_priority = current->rt_priority;
@@ -76,6 +77,7 @@ void destroy_spu_context(struct kref *kref)
76 spu_fini_csa(&ctx->csa); 77 spu_fini_csa(&ctx->csa);
77 if (ctx->gang) 78 if (ctx->gang)
78 spu_gang_remove_ctx(ctx->gang, ctx); 79 spu_gang_remove_ctx(ctx->gang, ctx);
80 BUG_ON(!list_empty(&ctx->rq));
79 kfree(ctx); 81 kfree(ctx);
80} 82}
81 83
diff --git a/arch/powerpc/platforms/cell/spufs/sched.c b/arch/powerpc/platforms/cell/spufs/sched.c
index 1582d7645237..876828cc95a2 100644
--- a/arch/powerpc/platforms/cell/spufs/sched.c
+++ b/arch/powerpc/platforms/cell/spufs/sched.c
@@ -245,6 +245,14 @@ static void spu_add_to_rq(struct spu_context *ctx)
245 spin_unlock(&spu_prio->runq_lock); 245 spin_unlock(&spu_prio->runq_lock);
246} 246}
247 247
248static void __spu_del_from_rq(struct spu_context *ctx, int prio)
249{
250 if (!list_empty(&ctx->rq))
251 list_del_init(&ctx->rq);
252 if (list_empty(&spu_prio->runq[prio]))
253 clear_bit(ctx->prio, spu_prio->bitmap);
254}
255
248/** 256/**
249 * spu_del_from_rq - remove a context from the runqueue 257 * spu_del_from_rq - remove a context from the runqueue
250 * @ctx: context to remove 258 * @ctx: context to remove
@@ -252,33 +260,10 @@ static void spu_add_to_rq(struct spu_context *ctx)
252static void spu_del_from_rq(struct spu_context *ctx) 260static void spu_del_from_rq(struct spu_context *ctx)
253{ 261{
254 spin_lock(&spu_prio->runq_lock); 262 spin_lock(&spu_prio->runq_lock);
255 list_del_init(&ctx->rq); 263 __spu_del_from_rq(ctx, ctx->prio);
256 if (list_empty(&spu_prio->runq[ctx->prio]))
257 clear_bit(ctx->prio, spu_prio->bitmap);
258 spin_unlock(&spu_prio->runq_lock); 264 spin_unlock(&spu_prio->runq_lock);
259} 265}
260 266
261/**
262 * spu_grab_context - remove one context from the runqueue
263 * @prio: priority of the context to be removed
264 *
265 * This function removes one context from the runqueue for priority @prio.
266 * If there is more than one context with the given priority the first
267 * task on the runqueue will be taken.
268 *
269 * Returns the spu_context it just removed.
270 *
271 * Must be called with spu_prio->runq_lock held.
272 */
273static struct spu_context *spu_grab_context(int prio)
274{
275 struct list_head *rq = &spu_prio->runq[prio];
276
277 if (list_empty(rq))
278 return NULL;
279 return list_entry(rq->next, struct spu_context, rq);
280}
281
282static void spu_prio_wait(struct spu_context *ctx) 267static void spu_prio_wait(struct spu_context *ctx)
283{ 268{
284 DEFINE_WAIT(wait); 269 DEFINE_WAIT(wait);
@@ -309,9 +294,14 @@ static void spu_reschedule(struct spu *spu)
309 spin_lock(&spu_prio->runq_lock); 294 spin_lock(&spu_prio->runq_lock);
310 best = sched_find_first_bit(spu_prio->bitmap); 295 best = sched_find_first_bit(spu_prio->bitmap);
311 if (best < MAX_PRIO) { 296 if (best < MAX_PRIO) {
312 struct spu_context *ctx = spu_grab_context(best); 297 struct list_head *rq = &spu_prio->runq[best];
313 if (ctx) 298 struct spu_context *ctx;
314 wake_up(&ctx->stop_wq); 299
300 BUG_ON(list_empty(rq));
301
302 ctx = list_entry(rq->next, struct spu_context, rq);
303 __spu_del_from_rq(ctx, best);
304 wake_up(&ctx->stop_wq);
315 } 305 }
316 spin_unlock(&spu_prio->runq_lock); 306 spin_unlock(&spu_prio->runq_lock);
317} 307}