Commit cfea61da authored by Jens Axboe's avatar Jens Axboe Committed by Linus Torvalds

[PATCH] cfq: bad allocation

Arjan (sensibly) put a might_sleep() in mempool_alloc() and it caught a bad
cfq usage.
Signed-off-by: default avatarJens Axboe <axboe@suse.de>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 45532ce5
...@@ -456,37 +456,61 @@ static void cfq_put_queue(struct cfq_data *cfqd, struct cfq_queue *cfqq) ...@@ -456,37 +456,61 @@ static void cfq_put_queue(struct cfq_data *cfqd, struct cfq_queue *cfqq)
mempool_free(cfqq, cfq_mpool); mempool_free(cfqq, cfq_mpool);
} }
static struct cfq_queue *cfq_get_queue(struct cfq_data *cfqd, int pid) static struct cfq_queue *__cfq_get_queue(struct cfq_data *cfqd, int pid,
int gfp_mask)
{ {
const int hashval = hash_long(current->tgid, CFQ_QHASH_SHIFT); const int hashval = hash_long(current->tgid, CFQ_QHASH_SHIFT);
struct cfq_queue *cfqq = __cfq_find_cfq_hash(cfqd, pid, hashval); struct cfq_queue *cfqq = __cfq_find_cfq_hash(cfqd, pid, hashval);
if (!cfqq) { if (!cfqq) {
cfqq = mempool_alloc(cfq_mpool, GFP_NOIO); cfqq = mempool_alloc(cfq_mpool, gfp_mask);
INIT_LIST_HEAD(&cfqq->cfq_hash); if (cfqq) {
INIT_LIST_HEAD(&cfqq->cfq_list); INIT_LIST_HEAD(&cfqq->cfq_hash);
RB_CLEAR_ROOT(&cfqq->sort_list); INIT_LIST_HEAD(&cfqq->cfq_list);
RB_CLEAR_ROOT(&cfqq->sort_list);
cfqq->pid = pid; cfqq->pid = pid;
cfqq->queued[0] = cfqq->queued[1] = 0; cfqq->queued[0] = cfqq->queued[1] = 0;
list_add(&cfqq->cfq_hash, &cfqd->cfq_hash[hashval]); list_add(&cfqq->cfq_hash, &cfqd->cfq_hash[hashval]);
}
} }
return cfqq; return cfqq;
} }
static void cfq_enqueue(struct cfq_data *cfqd, struct cfq_rq *crq) static struct cfq_queue *cfq_get_queue(struct cfq_data *cfqd, int pid,
int gfp_mask)
{ {
request_queue_t *q = cfqd->queue;
struct cfq_queue *cfqq; struct cfq_queue *cfqq;
cfqq = cfq_get_queue(cfqd, current->tgid); spin_lock_irq(q->queue_lock);
cfqq = __cfq_get_queue(cfqd, pid, gfp_mask);
spin_unlock_irq(q->queue_lock);
return cfqq;
}
cfq_add_crq_rb(cfqd, cfqq, crq); static void cfq_enqueue(struct cfq_data *cfqd, struct cfq_rq *crq)
{
struct cfq_queue *cfqq;
if (list_empty(&cfqq->cfq_list)) { cfqq = __cfq_get_queue(cfqd, current->tgid, GFP_ATOMIC);
list_add(&cfqq->cfq_list, &cfqd->rr_list); if (cfqq) {
cfqd->busy_queues++; cfq_add_crq_rb(cfqd, cfqq, crq);
if (list_empty(&cfqq->cfq_list)) {
list_add(&cfqq->cfq_list, &cfqd->rr_list);
cfqd->busy_queues++;
}
} else {
/*
* should can only happen if the request wasn't allocated
* through blk_alloc_request(), eg stack requests from ide-cd
* (those should be removed) _and_ we are in OOM.
*/
list_add_tail(&crq->request->queuelist, cfqd->dispatch);
} }
} }
...@@ -617,8 +641,17 @@ static void cfq_put_request(request_queue_t *q, struct request *rq) ...@@ -617,8 +641,17 @@ static void cfq_put_request(request_queue_t *q, struct request *rq)
static int cfq_set_request(request_queue_t *q, struct request *rq, int gfp_mask) static int cfq_set_request(request_queue_t *q, struct request *rq, int gfp_mask)
{ {
struct cfq_data *cfqd = q->elevator.elevator_data; struct cfq_data *cfqd = q->elevator.elevator_data;
struct cfq_rq *crq = mempool_alloc(cfqd->crq_pool, gfp_mask); struct cfq_queue *cfqq;
struct cfq_rq *crq;
/*
* prepare a queue up front, so cfq_enqueue() doesn't have to
*/
cfqq = cfq_get_queue(cfqd, current->tgid, gfp_mask);
if (!cfqq)
return 1;
crq = mempool_alloc(cfqd->crq_pool, gfp_mask);
if (crq) { if (crq) {
RB_CLEAR(&crq->rb_node); RB_CLEAR(&crq->rb_node);
crq->request = rq; crq->request = rq;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment