[PATCH] cfq-iosched: kill crq

Get rid of the cfq_rq request type. With the added elevator_private2, we
have enough room in struct request to get rid of any crq allocation/free
for each request.

Signed-off-by: Jens Axboe <axboe@suse.de>
This commit is contained in:
Jens Axboe 2006-07-13 12:39:25 +02:00 committed by Jens Axboe
parent ff7d145fd9
commit 5e70537479

View file

@ -43,9 +43,9 @@ static DEFINE_SPINLOCK(cfq_exit_lock);
#define list_entry_cfqq(ptr) list_entry((ptr), struct cfq_queue, cfq_list) #define list_entry_cfqq(ptr) list_entry((ptr), struct cfq_queue, cfq_list)
#define RQ_DATA(rq) (rq)->elevator_private #define RQ_CIC(rq) ((struct cfq_io_context*)(rq)->elevator_private)
#define RQ_CFQQ(rq) ((rq)->elevator_private2)
static kmem_cache_t *crq_pool;
static kmem_cache_t *cfq_pool; static kmem_cache_t *cfq_pool;
static kmem_cache_t *cfq_ioc_pool; static kmem_cache_t *cfq_ioc_pool;
@ -95,8 +95,6 @@ struct cfq_data {
*/ */
struct hlist_head *cfq_hash; struct hlist_head *cfq_hash;
mempool_t *crq_pool;
int rq_in_driver; int rq_in_driver;
int hw_tag; int hw_tag;
@ -153,7 +151,7 @@ struct cfq_queue {
/* sorted list of pending requests */ /* sorted list of pending requests */
struct rb_root sort_list; struct rb_root sort_list;
/* if fifo isn't expired, next request to serve */ /* if fifo isn't expired, next request to serve */
struct cfq_rq *next_crq; struct request *next_rq;
/* requests queued in sort_list */ /* requests queued in sort_list */
int queued[2]; int queued[2];
/* currently allocated requests */ /* currently allocated requests */
@ -177,13 +175,6 @@ struct cfq_queue {
unsigned int flags; unsigned int flags;
}; };
struct cfq_rq {
struct request *request;
struct cfq_queue *cfq_queue;
struct cfq_io_context *io_context;
};
enum cfqq_state_flags { enum cfqq_state_flags {
CFQ_CFQQ_FLAG_on_rr = 0, CFQ_CFQQ_FLAG_on_rr = 0,
CFQ_CFQQ_FLAG_wait_request, CFQ_CFQQ_FLAG_wait_request,
@ -220,7 +211,7 @@ CFQ_CFQQ_FNS(prio_changed);
#undef CFQ_CFQQ_FNS #undef CFQ_CFQQ_FNS
static struct cfq_queue *cfq_find_cfq_hash(struct cfq_data *, unsigned int, unsigned short); static struct cfq_queue *cfq_find_cfq_hash(struct cfq_data *, unsigned int, unsigned short);
static void cfq_dispatch_insert(request_queue_t *, struct cfq_rq *); static void cfq_dispatch_insert(request_queue_t *, struct request *);
static struct cfq_queue *cfq_get_queue(struct cfq_data *cfqd, unsigned int key, struct task_struct *tsk, gfp_t gfp_mask); static struct cfq_queue *cfq_get_queue(struct cfq_data *cfqd, unsigned int key, struct task_struct *tsk, gfp_t gfp_mask);
/* /*
@ -249,12 +240,12 @@ static inline pid_t cfq_queue_pid(struct task_struct *task, int rw)
} }
/* /*
* Lifted from AS - choose which of crq1 and crq2 that is best served now. * Lifted from AS - choose which of rq1 and rq2 that is best served now.
* We choose the request that is closest to the head right now. Distance * We choose the request that is closest to the head right now. Distance
* behind the head is penalized and only allowed to a certain extent. * behind the head is penalized and only allowed to a certain extent.
*/ */
static struct cfq_rq * static struct request *
cfq_choose_req(struct cfq_data *cfqd, struct cfq_rq *crq1, struct cfq_rq *crq2) cfq_choose_req(struct cfq_data *cfqd, struct request *rq1, struct request *rq2)
{ {
sector_t last, s1, s2, d1 = 0, d2 = 0; sector_t last, s1, s2, d1 = 0, d2 = 0;
unsigned long back_max; unsigned long back_max;
@ -262,18 +253,18 @@ cfq_choose_req(struct cfq_data *cfqd, struct cfq_rq *crq1, struct cfq_rq *crq2)
#define CFQ_RQ2_WRAP 0x02 /* request 2 wraps */ #define CFQ_RQ2_WRAP 0x02 /* request 2 wraps */
unsigned wrap = 0; /* bit mask: requests behind the disk head? */ unsigned wrap = 0; /* bit mask: requests behind the disk head? */
if (crq1 == NULL || crq1 == crq2) if (rq1 == NULL || rq1 == rq2)
return crq2; return rq2;
if (crq2 == NULL) if (rq2 == NULL)
return crq1; return rq1;
if (rq_is_sync(crq1->request) && !rq_is_sync(crq2->request)) if (rq_is_sync(rq1) && !rq_is_sync(rq2))
return crq1; return rq1;
else if (rq_is_sync(crq2->request) && !rq_is_sync(crq1->request)) else if (rq_is_sync(rq2) && !rq_is_sync(rq1))
return crq2; return rq2;
s1 = crq1->request->sector; s1 = rq1->sector;
s2 = crq2->request->sector; s2 = rq2->sector;
last = cfqd->last_sector; last = cfqd->last_sector;
@ -308,23 +299,23 @@ cfq_choose_req(struct cfq_data *cfqd, struct cfq_rq *crq1, struct cfq_rq *crq2)
* check two variables for all permutations: --> faster! * check two variables for all permutations: --> faster!
*/ */
switch (wrap) { switch (wrap) {
case 0: /* common case for CFQ: crq1 and crq2 not wrapped */ case 0: /* common case for CFQ: rq1 and rq2 not wrapped */
if (d1 < d2) if (d1 < d2)
return crq1; return rq1;
else if (d2 < d1) else if (d2 < d1)
return crq2; return rq2;
else { else {
if (s1 >= s2) if (s1 >= s2)
return crq1; return rq1;
else else
return crq2; return rq2;
} }
case CFQ_RQ2_WRAP: case CFQ_RQ2_WRAP:
return crq1; return rq1;
case CFQ_RQ1_WRAP: case CFQ_RQ1_WRAP:
return crq2; return rq2;
case (CFQ_RQ1_WRAP|CFQ_RQ2_WRAP): /* both crqs wrapped */ case (CFQ_RQ1_WRAP|CFQ_RQ2_WRAP): /* both rqs wrapped */
default: default:
/* /*
* Since both rqs are wrapped, * Since both rqs are wrapped,
@ -333,35 +324,34 @@ cfq_choose_req(struct cfq_data *cfqd, struct cfq_rq *crq1, struct cfq_rq *crq2)
* since back seek takes more time than forward. * since back seek takes more time than forward.
*/ */
if (s1 <= s2) if (s1 <= s2)
return crq1; return rq1;
else else
return crq2; return rq2;
} }
} }
/* /*
* would be nice to take fifo expire time into account as well * would be nice to take fifo expire time into account as well
*/ */
static struct cfq_rq * static struct request *
cfq_find_next_crq(struct cfq_data *cfqd, struct cfq_queue *cfqq, cfq_find_next_rq(struct cfq_data *cfqd, struct cfq_queue *cfqq,
struct cfq_rq *last_crq) struct request *last)
{ {
struct request *last = last_crq->request;
struct rb_node *rbnext = rb_next(&last->rb_node); struct rb_node *rbnext = rb_next(&last->rb_node);
struct rb_node *rbprev = rb_prev(&last->rb_node); struct rb_node *rbprev = rb_prev(&last->rb_node);
struct cfq_rq *next = NULL, *prev = NULL; struct request *next = NULL, *prev = NULL;
BUG_ON(RB_EMPTY_NODE(&last->rb_node)); BUG_ON(RB_EMPTY_NODE(&last->rb_node));
if (rbprev) if (rbprev)
prev = RQ_DATA(rb_entry_rq(rbprev)); prev = rb_entry_rq(rbprev);
if (rbnext) if (rbnext)
next = RQ_DATA(rb_entry_rq(rbnext)); next = rb_entry_rq(rbnext);
else { else {
rbnext = rb_first(&cfqq->sort_list); rbnext = rb_first(&cfqq->sort_list);
if (rbnext && rbnext != &last->rb_node) if (rbnext && rbnext != &last->rb_node)
next = RQ_DATA(rb_entry_rq(rbnext)); next = rb_entry_rq(rbnext);
} }
return cfq_choose_req(cfqd, next, prev); return cfq_choose_req(cfqd, next, prev);
@ -450,26 +440,25 @@ cfq_del_cfqq_rr(struct cfq_data *cfqd, struct cfq_queue *cfqq)
/* /*
* rb tree support functions * rb tree support functions
*/ */
static inline void cfq_del_crq_rb(struct cfq_rq *crq) static inline void cfq_del_rq_rb(struct request *rq)
{ {
struct cfq_queue *cfqq = crq->cfq_queue; struct cfq_queue *cfqq = RQ_CFQQ(rq);
struct cfq_data *cfqd = cfqq->cfqd; struct cfq_data *cfqd = cfqq->cfqd;
const int sync = rq_is_sync(crq->request); const int sync = rq_is_sync(rq);
BUG_ON(!cfqq->queued[sync]); BUG_ON(!cfqq->queued[sync]);
cfqq->queued[sync]--; cfqq->queued[sync]--;
elv_rb_del(&cfqq->sort_list, crq->request); elv_rb_del(&cfqq->sort_list, rq);
if (cfq_cfqq_on_rr(cfqq) && RB_EMPTY_ROOT(&cfqq->sort_list)) if (cfq_cfqq_on_rr(cfqq) && RB_EMPTY_ROOT(&cfqq->sort_list))
cfq_del_cfqq_rr(cfqd, cfqq); cfq_del_cfqq_rr(cfqd, cfqq);
} }
static void cfq_add_crq_rb(struct cfq_rq *crq) static void cfq_add_rq_rb(struct request *rq)
{ {
struct cfq_queue *cfqq = crq->cfq_queue; struct cfq_queue *cfqq = RQ_CFQQ(rq);
struct cfq_data *cfqd = cfqq->cfqd; struct cfq_data *cfqd = cfqq->cfqd;
struct request *rq = crq->request;
struct request *__alias; struct request *__alias;
cfqq->queued[rq_is_sync(rq)]++; cfqq->queued[rq_is_sync(rq)]++;
@ -479,17 +468,15 @@ static void cfq_add_crq_rb(struct cfq_rq *crq)
* if that happens, put the alias on the dispatch list * if that happens, put the alias on the dispatch list
*/ */
while ((__alias = elv_rb_add(&cfqq->sort_list, rq)) != NULL) while ((__alias = elv_rb_add(&cfqq->sort_list, rq)) != NULL)
cfq_dispatch_insert(cfqd->queue, RQ_DATA(__alias)); cfq_dispatch_insert(cfqd->queue, __alias);
} }
static inline void static inline void
cfq_reposition_crq_rb(struct cfq_queue *cfqq, struct cfq_rq *crq) cfq_reposition_rq_rb(struct cfq_queue *cfqq, struct request *rq)
{ {
struct request *rq = crq->request;
elv_rb_del(&cfqq->sort_list, rq); elv_rb_del(&cfqq->sort_list, rq);
cfqq->queued[rq_is_sync(rq)]--; cfqq->queued[rq_is_sync(rq)]--;
cfq_add_crq_rb(crq); cfq_add_rq_rb(rq);
} }
static struct request * static struct request *
@ -533,14 +520,13 @@ static void cfq_deactivate_request(request_queue_t *q, struct request *rq)
static void cfq_remove_request(struct request *rq) static void cfq_remove_request(struct request *rq)
{ {
struct cfq_rq *crq = RQ_DATA(rq); struct cfq_queue *cfqq = RQ_CFQQ(rq);
struct cfq_queue *cfqq = crq->cfq_queue;
if (cfqq->next_crq == crq) if (cfqq->next_rq == rq)
cfqq->next_crq = cfq_find_next_crq(cfqq->cfqd, cfqq, crq); cfqq->next_rq = cfq_find_next_rq(cfqq->cfqd, cfqq, rq);
list_del_init(&rq->queuelist); list_del_init(&rq->queuelist);
cfq_del_crq_rb(crq); cfq_del_rq_rb(rq);
} }
static int static int
@ -561,12 +547,10 @@ cfq_merge(request_queue_t *q, struct request **req, struct bio *bio)
static void cfq_merged_request(request_queue_t *q, struct request *req, static void cfq_merged_request(request_queue_t *q, struct request *req,
int type) int type)
{ {
struct cfq_rq *crq = RQ_DATA(req);
if (type == ELEVATOR_FRONT_MERGE) { if (type == ELEVATOR_FRONT_MERGE) {
struct cfq_queue *cfqq = crq->cfq_queue; struct cfq_queue *cfqq = RQ_CFQQ(req);
cfq_reposition_crq_rb(cfqq, crq); cfq_reposition_rq_rb(cfqq, req);
} }
} }
@ -789,11 +773,10 @@ static int cfq_arm_slice_timer(struct cfq_data *cfqd, struct cfq_queue *cfqq)
return 1; return 1;
} }
static void cfq_dispatch_insert(request_queue_t *q, struct cfq_rq *crq) static void cfq_dispatch_insert(request_queue_t *q, struct request *rq)
{ {
struct cfq_data *cfqd = q->elevator->elevator_data; struct cfq_data *cfqd = q->elevator->elevator_data;
struct cfq_queue *cfqq = crq->cfq_queue; struct cfq_queue *cfqq = RQ_CFQQ(rq);
struct request *rq = crq->request;
cfq_remove_request(rq); cfq_remove_request(rq);
cfqq->on_dispatch[rq_is_sync(rq)]++; cfqq->on_dispatch[rq_is_sync(rq)]++;
@ -806,11 +789,10 @@ static void cfq_dispatch_insert(request_queue_t *q, struct cfq_rq *crq)
/* /*
* return expired entry, or NULL to just start from scratch in rbtree * return expired entry, or NULL to just start from scratch in rbtree
*/ */
static inline struct cfq_rq *cfq_check_fifo(struct cfq_queue *cfqq) static inline struct request *cfq_check_fifo(struct cfq_queue *cfqq)
{ {
struct cfq_data *cfqd = cfqq->cfqd; struct cfq_data *cfqd = cfqq->cfqd;
struct request *rq; struct request *rq;
struct cfq_rq *crq;
if (cfq_cfqq_fifo_expire(cfqq)) if (cfq_cfqq_fifo_expire(cfqq))
return NULL; return NULL;
@ -818,11 +800,10 @@ static inline struct cfq_rq *cfq_check_fifo(struct cfq_queue *cfqq)
if (!list_empty(&cfqq->fifo)) { if (!list_empty(&cfqq->fifo)) {
int fifo = cfq_cfqq_class_sync(cfqq); int fifo = cfq_cfqq_class_sync(cfqq);
crq = RQ_DATA(rq_entry_fifo(cfqq->fifo.next)); rq = rq_entry_fifo(cfqq->fifo.next);
rq = crq->request;
if (time_after(jiffies, rq->start_time + cfqd->cfq_fifo_expire[fifo])) { if (time_after(jiffies, rq->start_time + cfqd->cfq_fifo_expire[fifo])) {
cfq_mark_cfqq_fifo_expire(cfqq); cfq_mark_cfqq_fifo_expire(cfqq);
return crq; return rq;
} }
} }
@ -909,25 +890,25 @@ __cfq_dispatch_requests(struct cfq_data *cfqd, struct cfq_queue *cfqq,
BUG_ON(RB_EMPTY_ROOT(&cfqq->sort_list)); BUG_ON(RB_EMPTY_ROOT(&cfqq->sort_list));
do { do {
struct cfq_rq *crq; struct request *rq;
/* /*
* follow expired path, else get first next available * follow expired path, else get first next available
*/ */
if ((crq = cfq_check_fifo(cfqq)) == NULL) if ((rq = cfq_check_fifo(cfqq)) == NULL)
crq = cfqq->next_crq; rq = cfqq->next_rq;
/* /*
* finally, insert request into driver dispatch list * finally, insert request into driver dispatch list
*/ */
cfq_dispatch_insert(cfqd->queue, crq); cfq_dispatch_insert(cfqd->queue, rq);
cfqd->dispatch_slice++; cfqd->dispatch_slice++;
dispatched++; dispatched++;
if (!cfqd->active_cic) { if (!cfqd->active_cic) {
atomic_inc(&crq->io_context->ioc->refcount); atomic_inc(&RQ_CIC(rq)->ioc->refcount);
cfqd->active_cic = crq->io_context; cfqd->active_cic = RQ_CIC(rq);
} }
if (RB_EMPTY_ROOT(&cfqq->sort_list)) if (RB_EMPTY_ROOT(&cfqq->sort_list))
@ -958,13 +939,12 @@ static int
cfq_forced_dispatch_cfqqs(struct list_head *list) cfq_forced_dispatch_cfqqs(struct list_head *list)
{ {
struct cfq_queue *cfqq, *next; struct cfq_queue *cfqq, *next;
struct cfq_rq *crq;
int dispatched; int dispatched;
dispatched = 0; dispatched = 0;
list_for_each_entry_safe(cfqq, next, list, cfq_list) { list_for_each_entry_safe(cfqq, next, list, cfq_list) {
while ((crq = cfqq->next_crq)) { while (cfqq->next_rq) {
cfq_dispatch_insert(cfqq->cfqd->queue, crq); cfq_dispatch_insert(cfqq->cfqd->queue, cfqq->next_rq);
dispatched++; dispatched++;
} }
BUG_ON(!list_empty(&cfqq->fifo)); BUG_ON(!list_empty(&cfqq->fifo));
@ -1040,8 +1020,8 @@ cfq_dispatch_requests(request_queue_t *q, int force)
} }
/* /*
* task holds one reference to the queue, dropped when task exits. each crq * task holds one reference to the queue, dropped when task exits. each rq
* in-flight on this queue also holds a reference, dropped when crq is freed. * in-flight on this queue also holds a reference, dropped when rq is freed.
* *
* queue lock must be held here. * queue lock must be held here.
*/ */
@ -1486,15 +1466,15 @@ cfq_update_io_thinktime(struct cfq_data *cfqd, struct cfq_io_context *cic)
static void static void
cfq_update_io_seektime(struct cfq_data *cfqd, struct cfq_io_context *cic, cfq_update_io_seektime(struct cfq_data *cfqd, struct cfq_io_context *cic,
struct cfq_rq *crq) struct request *rq)
{ {
sector_t sdist; sector_t sdist;
u64 total; u64 total;
if (cic->last_request_pos < crq->request->sector) if (cic->last_request_pos < rq->sector)
sdist = crq->request->sector - cic->last_request_pos; sdist = rq->sector - cic->last_request_pos;
else else
sdist = cic->last_request_pos - crq->request->sector; sdist = cic->last_request_pos - rq->sector;
/* /*
* Don't allow the seek distance to get too large from the * Don't allow the seek distance to get too large from the
@ -1545,7 +1525,7 @@ cfq_update_idle_window(struct cfq_data *cfqd, struct cfq_queue *cfqq,
*/ */
static int static int
cfq_should_preempt(struct cfq_data *cfqd, struct cfq_queue *new_cfqq, cfq_should_preempt(struct cfq_data *cfqd, struct cfq_queue *new_cfqq,
struct cfq_rq *crq) struct request *rq)
{ {
struct cfq_queue *cfqq = cfqd->active_queue; struct cfq_queue *cfqq = cfqd->active_queue;
@ -1564,7 +1544,7 @@ cfq_should_preempt(struct cfq_data *cfqd, struct cfq_queue *new_cfqq,
*/ */
if (new_cfqq->slice_left < cfqd->cfq_slice_idle) if (new_cfqq->slice_left < cfqd->cfq_slice_idle)
return 0; return 0;
if (rq_is_sync(crq->request) && !cfq_cfqq_sync(cfqq)) if (rq_is_sync(rq) && !cfq_cfqq_sync(cfqq))
return 1; return 1;
return 0; return 0;
@ -1603,26 +1583,26 @@ static void cfq_start_queueing(struct cfq_data *cfqd, struct cfq_queue *cfqq)
} }
/* /*
* Called when a new fs request (crq) is added (to cfqq). Check if there's * Called when a new fs request (rq) is added (to cfqq). Check if there's
* something we should do about it * something we should do about it
*/ */
static void static void
cfq_crq_enqueued(struct cfq_data *cfqd, struct cfq_queue *cfqq, cfq_rq_enqueued(struct cfq_data *cfqd, struct cfq_queue *cfqq,
struct cfq_rq *crq) struct request *rq)
{ {
struct cfq_io_context *cic = crq->io_context; struct cfq_io_context *cic = RQ_CIC(rq);
/* /*
* check if this request is a better next-serve candidate)) { * check if this request is a better next-serve candidate)) {
*/ */
cfqq->next_crq = cfq_choose_req(cfqd, cfqq->next_crq, crq); cfqq->next_rq = cfq_choose_req(cfqd, cfqq->next_rq, rq);
BUG_ON(!cfqq->next_crq); BUG_ON(!cfqq->next_rq);
/* /*
* we never wait for an async request and we don't allow preemption * we never wait for an async request and we don't allow preemption
* of an async request. so just return early * of an async request. so just return early
*/ */
if (!rq_is_sync(crq->request)) { if (!rq_is_sync(rq)) {
/* /*
* sync process issued an async request, if it's waiting * sync process issued an async request, if it's waiting
* then expire it and kick rq handling. * then expire it and kick rq handling.
@ -1636,11 +1616,11 @@ cfq_crq_enqueued(struct cfq_data *cfqd, struct cfq_queue *cfqq,
} }
cfq_update_io_thinktime(cfqd, cic); cfq_update_io_thinktime(cfqd, cic);
cfq_update_io_seektime(cfqd, cic, crq); cfq_update_io_seektime(cfqd, cic, rq);
cfq_update_idle_window(cfqd, cfqq, cic); cfq_update_idle_window(cfqd, cfqq, cic);
cic->last_queue = jiffies; cic->last_queue = jiffies;
cic->last_request_pos = crq->request->sector + crq->request->nr_sectors; cic->last_request_pos = rq->sector + rq->nr_sectors;
if (cfqq == cfqd->active_queue) { if (cfqq == cfqd->active_queue) {
/* /*
@ -1653,7 +1633,7 @@ cfq_crq_enqueued(struct cfq_data *cfqd, struct cfq_queue *cfqq,
del_timer(&cfqd->idle_slice_timer); del_timer(&cfqd->idle_slice_timer);
cfq_start_queueing(cfqd, cfqq); cfq_start_queueing(cfqd, cfqq);
} }
} else if (cfq_should_preempt(cfqd, cfqq, crq)) { } else if (cfq_should_preempt(cfqd, cfqq, rq)) {
/* /*
* not the active queue - expire current slice if it is * not the active queue - expire current slice if it is
* idle and has expired it's mean thinktime or this new queue * idle and has expired it's mean thinktime or this new queue
@ -1668,25 +1648,23 @@ cfq_crq_enqueued(struct cfq_data *cfqd, struct cfq_queue *cfqq,
static void cfq_insert_request(request_queue_t *q, struct request *rq) static void cfq_insert_request(request_queue_t *q, struct request *rq)
{ {
struct cfq_data *cfqd = q->elevator->elevator_data; struct cfq_data *cfqd = q->elevator->elevator_data;
struct cfq_rq *crq = RQ_DATA(rq); struct cfq_queue *cfqq = RQ_CFQQ(rq);
struct cfq_queue *cfqq = crq->cfq_queue;
cfq_init_prio_data(cfqq); cfq_init_prio_data(cfqq);
cfq_add_crq_rb(crq); cfq_add_rq_rb(rq);
if (!cfq_cfqq_on_rr(cfqq)) if (!cfq_cfqq_on_rr(cfqq))
cfq_add_cfqq_rr(cfqd, cfqq); cfq_add_cfqq_rr(cfqd, cfqq);
list_add_tail(&rq->queuelist, &cfqq->fifo); list_add_tail(&rq->queuelist, &cfqq->fifo);
cfq_crq_enqueued(cfqd, cfqq, crq); cfq_rq_enqueued(cfqd, cfqq, rq);
} }
static void cfq_completed_request(request_queue_t *q, struct request *rq) static void cfq_completed_request(request_queue_t *q, struct request *rq)
{ {
struct cfq_rq *crq = RQ_DATA(rq); struct cfq_queue *cfqq = RQ_CFQQ(rq);
struct cfq_queue *cfqq = crq->cfq_queue;
struct cfq_data *cfqd = cfqq->cfqd; struct cfq_data *cfqd = cfqq->cfqd;
const int sync = rq_is_sync(rq); const int sync = rq_is_sync(rq);
unsigned long now; unsigned long now;
@ -1709,7 +1687,7 @@ static void cfq_completed_request(request_queue_t *q, struct request *rq)
} }
if (sync) if (sync)
crq->io_context->last_end_request = now; RQ_CIC(rq)->last_end_request = now;
/* /*
* If this is the active queue, check if it needs to be expired, * If this is the active queue, check if it needs to be expired,
@ -1817,20 +1795,18 @@ static void cfq_check_waiters(request_queue_t *q, struct cfq_queue *cfqq)
*/ */
static void cfq_put_request(request_queue_t *q, struct request *rq) static void cfq_put_request(request_queue_t *q, struct request *rq)
{ {
struct cfq_data *cfqd = q->elevator->elevator_data; struct cfq_queue *cfqq = RQ_CFQQ(rq);
struct cfq_rq *crq = RQ_DATA(rq);
if (crq) { if (cfqq) {
struct cfq_queue *cfqq = crq->cfq_queue;
const int rw = rq_data_dir(rq); const int rw = rq_data_dir(rq);
BUG_ON(!cfqq->allocated[rw]); BUG_ON(!cfqq->allocated[rw]);
cfqq->allocated[rw]--; cfqq->allocated[rw]--;
put_io_context(crq->io_context->ioc); put_io_context(RQ_CIC(rq)->ioc);
mempool_free(crq, cfqd->crq_pool);
rq->elevator_private = NULL; rq->elevator_private = NULL;
rq->elevator_private2 = NULL;
cfq_check_waiters(q, cfqq); cfq_check_waiters(q, cfqq);
cfq_put_queue(cfqq); cfq_put_queue(cfqq);
@ -1850,7 +1826,6 @@ cfq_set_request(request_queue_t *q, struct request *rq, struct bio *bio,
const int rw = rq_data_dir(rq); const int rw = rq_data_dir(rq);
pid_t key = cfq_queue_pid(tsk, rw); pid_t key = cfq_queue_pid(tsk, rw);
struct cfq_queue *cfqq; struct cfq_queue *cfqq;
struct cfq_rq *crq;
unsigned long flags; unsigned long flags;
int is_sync = key != CFQ_KEY_ASYNC; int is_sync = key != CFQ_KEY_ASYNC;
@ -1876,23 +1851,13 @@ cfq_set_request(request_queue_t *q, struct request *rq, struct bio *bio,
cfq_clear_cfqq_must_alloc(cfqq); cfq_clear_cfqq_must_alloc(cfqq);
cfqd->rq_starved = 0; cfqd->rq_starved = 0;
atomic_inc(&cfqq->ref); atomic_inc(&cfqq->ref);
spin_unlock_irqrestore(q->queue_lock, flags); spin_unlock_irqrestore(q->queue_lock, flags);
crq = mempool_alloc(cfqd->crq_pool, gfp_mask); rq->elevator_private = cic;
if (crq) { rq->elevator_private2 = cfqq;
crq->request = rq; return 0;
crq->cfq_queue = cfqq;
crq->io_context = cic;
rq->elevator_private = crq;
return 0;
}
spin_lock_irqsave(q->queue_lock, flags);
cfqq->allocated[rw]--;
if (!(cfqq->allocated[0] + cfqq->allocated[1]))
cfq_mark_cfqq_must_alloc(cfqq);
cfq_put_queue(cfqq);
queue_fail: queue_fail:
if (cic) if (cic)
put_io_context(cic->ioc); put_io_context(cic->ioc);
@ -2040,7 +2005,6 @@ static void cfq_exit_queue(elevator_t *e)
cfq_shutdown_timer_wq(cfqd); cfq_shutdown_timer_wq(cfqd);
mempool_destroy(cfqd->crq_pool);
kfree(cfqd->cfq_hash); kfree(cfqd->cfq_hash);
kfree(cfqd); kfree(cfqd);
} }
@ -2067,11 +2031,7 @@ static void *cfq_init_queue(request_queue_t *q, elevator_t *e)
cfqd->cfq_hash = kmalloc(sizeof(struct hlist_head) * CFQ_QHASH_ENTRIES, GFP_KERNEL); cfqd->cfq_hash = kmalloc(sizeof(struct hlist_head) * CFQ_QHASH_ENTRIES, GFP_KERNEL);
if (!cfqd->cfq_hash) if (!cfqd->cfq_hash)
goto out_crqhash; goto out_free;
cfqd->crq_pool = mempool_create_slab_pool(BLKDEV_MIN_RQ, crq_pool);
if (!cfqd->crq_pool)
goto out_crqpool;
for (i = 0; i < CFQ_QHASH_ENTRIES; i++) for (i = 0; i < CFQ_QHASH_ENTRIES; i++)
INIT_HLIST_HEAD(&cfqd->cfq_hash[i]); INIT_HLIST_HEAD(&cfqd->cfq_hash[i]);
@ -2100,17 +2060,13 @@ static void *cfq_init_queue(request_queue_t *q, elevator_t *e)
cfqd->cfq_slice_idle = cfq_slice_idle; cfqd->cfq_slice_idle = cfq_slice_idle;
return cfqd; return cfqd;
out_crqpool: out_free:
kfree(cfqd->cfq_hash);
out_crqhash:
kfree(cfqd); kfree(cfqd);
return NULL; return NULL;
} }
static void cfq_slab_kill(void) static void cfq_slab_kill(void)
{ {
if (crq_pool)
kmem_cache_destroy(crq_pool);
if (cfq_pool) if (cfq_pool)
kmem_cache_destroy(cfq_pool); kmem_cache_destroy(cfq_pool);
if (cfq_ioc_pool) if (cfq_ioc_pool)
@ -2119,11 +2075,6 @@ static void cfq_slab_kill(void)
static int __init cfq_slab_setup(void) static int __init cfq_slab_setup(void)
{ {
crq_pool = kmem_cache_create("crq_pool", sizeof(struct cfq_rq), 0, 0,
NULL, NULL);
if (!crq_pool)
goto fail;
cfq_pool = kmem_cache_create("cfq_pool", sizeof(struct cfq_queue), 0, 0, cfq_pool = kmem_cache_create("cfq_pool", sizeof(struct cfq_queue), 0, 0,
NULL, NULL); NULL, NULL);
if (!cfq_pool) if (!cfq_pool)