diff options
author | Christoph Hellwig <hch@infradead.org> | 2014-01-08 12:33:37 -0500 |
---|---|---|
committer | Jens Axboe <axboe@kernel.dk> | 2014-01-08 16:31:27 -0500 |
commit | 3d6efbf62c797a2924785f482e4ce8aa8820ec72 (patch) | |
tree | 2c2000c1a79e4cd0333f2112d231998ee282e5b6 | |
parent | c7b22bb19a24fef1a851a41e5c0657c0c4a41550 (diff) |
blk-mq: use __smp_call_function_single directly
__smp_call_function_single already avoids multiple IPIs by internally
queing up the items, and now also is available for non-SMP builds as
a trivially correct stub, so there is no need to wrap it. If the
additional lock roundtrip cause problems my patch to convert the
generic IPI code to llists is waiting to get merged will fix it.
Signed-off-by: Christoph Hellwig <hch@lst.de>
Signed-off-by: Jens Axboe <axboe@kernel.dk>
-rw-r--r-- | block/blk-mq-cpu.c | 31 | ||||
-rw-r--r-- | block/blk-mq.c | 68 | ||||
-rw-r--r-- | block/blk-mq.h | 1 |
3 files changed, 11 insertions, 89 deletions
diff --git a/block/blk-mq-cpu.c b/block/blk-mq-cpu.c index 0045ace9bdf0..20576e3476e9 100644 --- a/block/blk-mq-cpu.c +++ b/block/blk-mq-cpu.c | |||
@@ -28,32 +28,6 @@ static int blk_mq_main_cpu_notify(struct notifier_block *self, | |||
28 | return NOTIFY_OK; | 28 | return NOTIFY_OK; |
29 | } | 29 | } |
30 | 30 | ||
31 | static void blk_mq_cpu_notify(void *data, unsigned long action, | ||
32 | unsigned int cpu) | ||
33 | { | ||
34 | if (action == CPU_DEAD || action == CPU_DEAD_FROZEN) { | ||
35 | /* | ||
36 | * If the CPU goes away, ensure that we run any pending | ||
37 | * completions. | ||
38 | */ | ||
39 | struct llist_node *node; | ||
40 | struct request *rq; | ||
41 | |||
42 | local_irq_disable(); | ||
43 | |||
44 | node = llist_del_all(&per_cpu(ipi_lists, cpu)); | ||
45 | while (node) { | ||
46 | struct llist_node *next = node->next; | ||
47 | |||
48 | rq = llist_entry(node, struct request, ll_list); | ||
49 | __blk_mq_end_io(rq, rq->errors); | ||
50 | node = next; | ||
51 | } | ||
52 | |||
53 | local_irq_enable(); | ||
54 | } | ||
55 | } | ||
56 | |||
57 | static struct notifier_block __cpuinitdata blk_mq_main_cpu_notifier = { | 31 | static struct notifier_block __cpuinitdata blk_mq_main_cpu_notifier = { |
58 | .notifier_call = blk_mq_main_cpu_notify, | 32 | .notifier_call = blk_mq_main_cpu_notify, |
59 | }; | 33 | }; |
@@ -82,12 +56,7 @@ void blk_mq_init_cpu_notifier(struct blk_mq_cpu_notifier *notifier, | |||
82 | notifier->data = data; | 56 | notifier->data = data; |
83 | } | 57 | } |
84 | 58 | ||
85 | static struct blk_mq_cpu_notifier __cpuinitdata cpu_notifier = { | ||
86 | .notify = blk_mq_cpu_notify, | ||
87 | }; | ||
88 | |||
89 | void __init blk_mq_cpu_init(void) | 59 | void __init blk_mq_cpu_init(void) |
90 | { | 60 | { |
91 | register_hotcpu_notifier(&blk_mq_main_cpu_notifier); | 61 | register_hotcpu_notifier(&blk_mq_main_cpu_notifier); |
92 | blk_mq_register_cpu_notifier(&cpu_notifier); | ||
93 | } | 62 | } |
diff --git a/block/blk-mq.c b/block/blk-mq.c index 473ce4039060..68734f87f1da 100644 --- a/block/blk-mq.c +++ b/block/blk-mq.c | |||
@@ -27,8 +27,6 @@ static LIST_HEAD(all_q_list); | |||
27 | 27 | ||
28 | static void __blk_mq_run_hw_queue(struct blk_mq_hw_ctx *hctx); | 28 | static void __blk_mq_run_hw_queue(struct blk_mq_hw_ctx *hctx); |
29 | 29 | ||
30 | DEFINE_PER_CPU(struct llist_head, ipi_lists); | ||
31 | |||
32 | static struct blk_mq_ctx *__blk_mq_get_ctx(struct request_queue *q, | 30 | static struct blk_mq_ctx *__blk_mq_get_ctx(struct request_queue *q, |
33 | unsigned int cpu) | 31 | unsigned int cpu) |
34 | { | 32 | { |
@@ -339,55 +337,12 @@ void __blk_mq_end_io(struct request *rq, int error) | |||
339 | blk_mq_complete_request(rq, error); | 337 | blk_mq_complete_request(rq, error); |
340 | } | 338 | } |
341 | 339 | ||
342 | #if defined(CONFIG_SMP) | 340 | static void blk_mq_end_io_remote(void *data) |
343 | |||
344 | /* | ||
345 | * Called with interrupts disabled. | ||
346 | */ | ||
347 | static void ipi_end_io(void *data) | ||
348 | { | 341 | { |
349 | struct llist_head *list = &per_cpu(ipi_lists, smp_processor_id()); | 342 | struct request *rq = data; |
350 | struct llist_node *entry, *next; | ||
351 | struct request *rq; | ||
352 | |||
353 | entry = llist_del_all(list); | ||
354 | |||
355 | while (entry) { | ||
356 | next = entry->next; | ||
357 | rq = llist_entry(entry, struct request, ll_list); | ||
358 | __blk_mq_end_io(rq, rq->errors); | ||
359 | entry = next; | ||
360 | } | ||
361 | } | ||
362 | |||
363 | static int ipi_remote_cpu(struct blk_mq_ctx *ctx, const int cpu, | ||
364 | struct request *rq, const int error) | ||
365 | { | ||
366 | struct call_single_data *data = &rq->csd; | ||
367 | |||
368 | rq->errors = error; | ||
369 | rq->ll_list.next = NULL; | ||
370 | |||
371 | /* | ||
372 | * If the list is non-empty, an existing IPI must already | ||
373 | * be "in flight". If that is the case, we need not schedule | ||
374 | * a new one. | ||
375 | */ | ||
376 | if (llist_add(&rq->ll_list, &per_cpu(ipi_lists, ctx->cpu))) { | ||
377 | data->func = ipi_end_io; | ||
378 | data->flags = 0; | ||
379 | __smp_call_function_single(ctx->cpu, data, 0); | ||
380 | } | ||
381 | 343 | ||
382 | return true; | 344 | __blk_mq_end_io(rq, rq->errors); |
383 | } | 345 | } |
384 | #else /* CONFIG_SMP */ | ||
385 | static int ipi_remote_cpu(struct blk_mq_ctx *ctx, const int cpu, | ||
386 | struct request *rq, const int error) | ||
387 | { | ||
388 | return false; | ||
389 | } | ||
390 | #endif | ||
391 | 346 | ||
392 | /* | 347 | /* |
393 | * End IO on this request on a multiqueue enabled driver. We'll either do | 348 | * End IO on this request on a multiqueue enabled driver. We'll either do |
@@ -403,11 +358,15 @@ void blk_mq_end_io(struct request *rq, int error) | |||
403 | return __blk_mq_end_io(rq, error); | 358 | return __blk_mq_end_io(rq, error); |
404 | 359 | ||
405 | cpu = get_cpu(); | 360 | cpu = get_cpu(); |
406 | 361 | if (cpu != ctx->cpu && cpu_online(ctx->cpu)) { | |
407 | if (cpu == ctx->cpu || !cpu_online(ctx->cpu) || | 362 | rq->errors = error; |
408 | !ipi_remote_cpu(ctx, cpu, rq, error)) | 363 | rq->csd.func = blk_mq_end_io_remote; |
364 | rq->csd.info = rq; | ||
365 | rq->csd.flags = 0; | ||
366 | __smp_call_function_single(ctx->cpu, &rq->csd, 0); | ||
367 | } else { | ||
409 | __blk_mq_end_io(rq, error); | 368 | __blk_mq_end_io(rq, error); |
410 | 369 | } | |
411 | put_cpu(); | 370 | put_cpu(); |
412 | } | 371 | } |
413 | EXPORT_SYMBOL(blk_mq_end_io); | 372 | EXPORT_SYMBOL(blk_mq_end_io); |
@@ -1506,11 +1465,6 @@ static int blk_mq_queue_reinit_notify(struct notifier_block *nb, | |||
1506 | 1465 | ||
1507 | static int __init blk_mq_init(void) | 1466 | static int __init blk_mq_init(void) |
1508 | { | 1467 | { |
1509 | unsigned int i; | ||
1510 | |||
1511 | for_each_possible_cpu(i) | ||
1512 | init_llist_head(&per_cpu(ipi_lists, i)); | ||
1513 | |||
1514 | blk_mq_cpu_init(); | 1468 | blk_mq_cpu_init(); |
1515 | 1469 | ||
1516 | /* Must be called after percpu_counter_hotcpu_callback() */ | 1470 | /* Must be called after percpu_counter_hotcpu_callback() */ |
diff --git a/block/blk-mq.h b/block/blk-mq.h index e151a2f4f171..5c3917984b00 100644 --- a/block/blk-mq.h +++ b/block/blk-mq.h | |||
@@ -40,7 +40,6 @@ void blk_mq_init_cpu_notifier(struct blk_mq_cpu_notifier *notifier, | |||
40 | void blk_mq_register_cpu_notifier(struct blk_mq_cpu_notifier *notifier); | 40 | void blk_mq_register_cpu_notifier(struct blk_mq_cpu_notifier *notifier); |
41 | void blk_mq_unregister_cpu_notifier(struct blk_mq_cpu_notifier *notifier); | 41 | void blk_mq_unregister_cpu_notifier(struct blk_mq_cpu_notifier *notifier); |
42 | void blk_mq_cpu_init(void); | 42 | void blk_mq_cpu_init(void); |
43 | DECLARE_PER_CPU(struct llist_head, ipi_lists); | ||
44 | 43 | ||
45 | /* | 44 | /* |
46 | * CPU -> queue mappings | 45 | * CPU -> queue mappings |