Commit 53b03744 authored by Jens Axboe's avatar Jens Axboe Committed by Jens Axboe

[PATCH] cfq-iosched: Kill O(N) runtime of cfq_resort_rr_list()

Currently it scales with number of processes in that priority group,
which is potentially not very nice as it's called quite often.
Basically we always need to do tail inserts, except for the case of a
new process. So just mark/detect a queue as such.
Signed-off-by: default avatarJens Axboe <axboe@suse.de>
parent b5deef90
...@@ -152,7 +152,6 @@ struct cfq_queue { ...@@ -152,7 +152,6 @@ struct cfq_queue {
unsigned long slice_start; unsigned long slice_start;
unsigned long slice_end; unsigned long slice_end;
unsigned long slice_left; unsigned long slice_left;
unsigned long service_last;
/* number of requests that are on the dispatch list */ /* number of requests that are on the dispatch list */
int on_dispatch[2]; int on_dispatch[2];
...@@ -174,6 +173,7 @@ enum cfqq_state_flags { ...@@ -174,6 +173,7 @@ enum cfqq_state_flags {
CFQ_CFQQ_FLAG_fifo_expire, CFQ_CFQQ_FLAG_fifo_expire,
CFQ_CFQQ_FLAG_idle_window, CFQ_CFQQ_FLAG_idle_window,
CFQ_CFQQ_FLAG_prio_changed, CFQ_CFQQ_FLAG_prio_changed,
CFQ_CFQQ_FLAG_queue_new,
}; };
#define CFQ_CFQQ_FNS(name) \ #define CFQ_CFQQ_FNS(name) \
...@@ -198,6 +198,7 @@ CFQ_CFQQ_FNS(must_dispatch); ...@@ -198,6 +198,7 @@ CFQ_CFQQ_FNS(must_dispatch);
CFQ_CFQQ_FNS(fifo_expire); CFQ_CFQQ_FNS(fifo_expire);
CFQ_CFQQ_FNS(idle_window); CFQ_CFQQ_FNS(idle_window);
CFQ_CFQQ_FNS(prio_changed); CFQ_CFQQ_FNS(prio_changed);
CFQ_CFQQ_FNS(queue_new);
#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);
...@@ -350,7 +351,7 @@ cfq_find_next_rq(struct cfq_data *cfqd, struct cfq_queue *cfqq, ...@@ -350,7 +351,7 @@ cfq_find_next_rq(struct cfq_data *cfqd, struct cfq_queue *cfqq,
static void cfq_resort_rr_list(struct cfq_queue *cfqq, int preempted) static void cfq_resort_rr_list(struct cfq_queue *cfqq, int preempted)
{ {
struct cfq_data *cfqd = cfqq->cfqd; struct cfq_data *cfqd = cfqq->cfqd;
struct list_head *list, *entry; struct list_head *list;
BUG_ON(!cfq_cfqq_on_rr(cfqq)); BUG_ON(!cfq_cfqq_on_rr(cfqq));
...@@ -375,31 +376,26 @@ static void cfq_resort_rr_list(struct cfq_queue *cfqq, int preempted) ...@@ -375,31 +376,26 @@ static void cfq_resort_rr_list(struct cfq_queue *cfqq, int preempted)
} }
/* /*
* if queue was preempted, just add to front to be fair. busy_rr * If this queue was preempted or is new (never been serviced), let
* isn't sorted, but insert at the back for fairness. * it be added first for fairness but beind other new queues.
* Otherwise, just add to the back of the list.
*/ */
if (preempted || list == &cfqd->busy_rr) { if (preempted || cfq_cfqq_queue_new(cfqq)) {
if (preempted) struct list_head *n = list;
list = list->prev; struct cfq_queue *__cfqq;
list_add_tail(&cfqq->cfq_list, list); while (n->next != list) {
return; __cfqq = list_entry_cfqq(n->next);
} if (!cfq_cfqq_queue_new(__cfqq))
break;
/* n = n->next;
* sort by when queue was last serviced }
*/
entry = list;
while ((entry = entry->prev) != list) {
struct cfq_queue *__cfqq = list_entry_cfqq(entry);
if (!__cfqq->service_last) list = n;
break;
if (time_before(__cfqq->service_last, cfqq->service_last))
break;
} }
list_add(&cfqq->cfq_list, entry); list_add_tail(&cfqq->cfq_list, list);
} }
/* /*
...@@ -591,13 +587,12 @@ __cfq_slice_expired(struct cfq_data *cfqd, struct cfq_queue *cfqq, ...@@ -591,13 +587,12 @@ __cfq_slice_expired(struct cfq_data *cfqd, struct cfq_queue *cfqq,
if (cfq_cfqq_wait_request(cfqq)) if (cfq_cfqq_wait_request(cfqq))
del_timer(&cfqd->idle_slice_timer); del_timer(&cfqd->idle_slice_timer);
if (!preempted && !cfq_cfqq_dispatched(cfqq)) { if (!preempted && !cfq_cfqq_dispatched(cfqq))
cfqq->service_last = now;
cfq_schedule_dispatch(cfqd); cfq_schedule_dispatch(cfqd);
}
cfq_clear_cfqq_must_dispatch(cfqq); cfq_clear_cfqq_must_dispatch(cfqq);
cfq_clear_cfqq_wait_request(cfqq); cfq_clear_cfqq_wait_request(cfqq);
cfq_clear_cfqq_queue_new(cfqq);
/* /*
* store what was left of this slice, if the queue idled out * store what was left of this slice, if the queue idled out
...@@ -1297,13 +1292,13 @@ cfq_get_queue(struct cfq_data *cfqd, unsigned int key, struct task_struct *tsk, ...@@ -1297,13 +1292,13 @@ cfq_get_queue(struct cfq_data *cfqd, unsigned int key, struct task_struct *tsk,
hlist_add_head(&cfqq->cfq_hash, &cfqd->cfq_hash[hashval]); hlist_add_head(&cfqq->cfq_hash, &cfqd->cfq_hash[hashval]);
atomic_set(&cfqq->ref, 0); atomic_set(&cfqq->ref, 0);
cfqq->cfqd = cfqd; cfqq->cfqd = cfqd;
cfqq->service_last = 0;
/* /*
* set ->slice_left to allow preemption for a new process * set ->slice_left to allow preemption for a new process
*/ */
cfqq->slice_left = 2 * cfqd->cfq_slice_idle; cfqq->slice_left = 2 * cfqd->cfq_slice_idle;
cfq_mark_cfqq_idle_window(cfqq); cfq_mark_cfqq_idle_window(cfqq);
cfq_mark_cfqq_prio_changed(cfqq); cfq_mark_cfqq_prio_changed(cfqq);
cfq_mark_cfqq_queue_new(cfqq);
cfq_init_prio_data(cfqq); cfq_init_prio_data(cfqq);
} }
...@@ -1672,12 +1667,8 @@ static void cfq_completed_request(request_queue_t *q, struct request *rq) ...@@ -1672,12 +1667,8 @@ static void cfq_completed_request(request_queue_t *q, struct request *rq)
if (!cfq_class_idle(cfqq)) if (!cfq_class_idle(cfqq))
cfqd->last_end_request = now; cfqd->last_end_request = now;
if (!cfq_cfqq_dispatched(cfqq)) { if (!cfq_cfqq_dispatched(cfqq) && cfq_cfqq_on_rr(cfqq))
if (cfq_cfqq_on_rr(cfqq)) { cfq_resort_rr_list(cfqq, 0);
cfqq->service_last = now;
cfq_resort_rr_list(cfqq, 0);
}
}
if (sync) if (sync)
RQ_CIC(rq)->last_end_request = now; RQ_CIC(rq)->last_end_request = now;
......
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