aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/md/md.c
diff options
context:
space:
mode:
authorNeilBrown <neilb@suse.de>2010-06-01 05:37:29 -0400
committerNeilBrown <neilb@suse.de>2010-07-25 22:53:08 -0400
commit2ac8740151b082f045e58010eb92560c3a23a0e9 (patch)
tree39bff686e28f033339c6d1cf47042b6dc1586c2f /drivers/md/md.c
parent11d8a6e3719519fbc0e2c9d61b6fa931b84bf813 (diff)
md/raid5: add simple plugging infrastructure.
md/raid5 uses the plugging infrastructure provided by the block layer and 'struct request_queue'. However when we plug raid5 under dm there is no request queue so we cannot use that. So create a similar infrastructure that is much lighter weight and use it for raid5. Signed-off-by: NeilBrown <neilb@suse.de>
Diffstat (limited to 'drivers/md/md.c')
-rw-r--r--drivers/md/md.c45
1 files changed, 45 insertions, 0 deletions
diff --git a/drivers/md/md.c b/drivers/md/md.c
index f8775699e15a..eec75f130708 100644
--- a/drivers/md/md.c
+++ b/drivers/md/md.c
@@ -386,6 +386,51 @@ void md_barrier_request(mddev_t *mddev, struct bio *bio)
386} 386}
387EXPORT_SYMBOL(md_barrier_request); 387EXPORT_SYMBOL(md_barrier_request);
388 388
389/* Support for plugging.
390 * This mirrors the plugging support in request_queue, but does not
391 * require having a whole queue
392 */
393static void plugger_work(struct work_struct *work)
394{
395 struct plug_handle *plug =
396 container_of(work, struct plug_handle, unplug_work);
397 plug->unplug_fn(plug);
398}
399static void plugger_timeout(unsigned long data)
400{
401 struct plug_handle *plug = (void *)data;
402 kblockd_schedule_work(NULL, &plug->unplug_work);
403}
404void plugger_init(struct plug_handle *plug,
405 void (*unplug_fn)(struct plug_handle *))
406{
407 plug->unplug_flag = 0;
408 plug->unplug_fn = unplug_fn;
409 init_timer(&plug->unplug_timer);
410 plug->unplug_timer.function = plugger_timeout;
411 plug->unplug_timer.data = (unsigned long)plug;
412 INIT_WORK(&plug->unplug_work, plugger_work);
413}
414EXPORT_SYMBOL_GPL(plugger_init);
415
416void plugger_set_plug(struct plug_handle *plug)
417{
418 if (!test_and_set_bit(PLUGGED_FLAG, &plug->unplug_flag))
419 mod_timer(&plug->unplug_timer, jiffies + msecs_to_jiffies(3)+1);
420}
421EXPORT_SYMBOL_GPL(plugger_set_plug);
422
423int plugger_remove_plug(struct plug_handle *plug)
424{
425 if (test_and_clear_bit(PLUGGED_FLAG, &plug->unplug_flag)) {
426 del_timer(&plug->unplug_timer);
427 return 1;
428 } else
429 return 0;
430}
431EXPORT_SYMBOL_GPL(plugger_remove_plug);
432
433
389static inline mddev_t *mddev_get(mddev_t *mddev) 434static inline mddev_t *mddev_get(mddev_t *mddev)
390{ 435{
391 atomic_inc(&mddev->active); 436 atomic_inc(&mddev->active);