aboutsummaryrefslogtreecommitdiffstats
path: root/drivers
diff options
context:
space:
mode:
authorKent Overstreet <kmo@daterainc.com>2013-11-23 21:34:15 -0500
committerKent Overstreet <kmo@daterainc.com>2013-11-24 01:33:56 -0500
commit196d38bccfcfa32faed8c561868336fdfa0fe8e4 (patch)
tree9dee2bc174436072c9e90ffebbe71f47fde63aaf /drivers
parente90abc8ec323c1fd2a25600097ef7ae1e91f39b0 (diff)
block: Generic bio chaining
This adds a generic mechanism for chaining bio completions. This is going to be used for a bio_split() replacement, and it turns out to be very useful in a fair amount of driver code - a fair number of drivers were implementing this in their own roundabout ways, often painfully. Note that this means it's no longer to call bio_endio() more than once on the same bio! This can cause problems for drivers that save/restore bi_end_io. Arguably they shouldn't be saving/restoring bi_end_io at all - in all but the simplest cases they'd be better off just cloning the bio, and immutable biovecs is making bio cloning cheaper. But for now, we add a bio_endio_nodec() for these cases. Signed-off-by: Kent Overstreet <kmo@daterainc.com> Cc: Jens Axboe <axboe@kernel.dk>
Diffstat (limited to 'drivers')
-rw-r--r--drivers/md/bcache/io.c2
-rw-r--r--drivers/md/dm-cache-target.c6
-rw-r--r--drivers/md/dm-snap.c1
-rw-r--r--drivers/md/dm-thin.c8
-rw-r--r--drivers/md/dm-verity.c2
5 files changed, 15 insertions, 4 deletions
diff --git a/drivers/md/bcache/io.c b/drivers/md/bcache/io.c
index 0f0ab659914d..522f95778443 100644
--- a/drivers/md/bcache/io.c
+++ b/drivers/md/bcache/io.c
@@ -133,7 +133,7 @@ static void bch_bio_submit_split_done(struct closure *cl)
133 133
134 s->bio->bi_end_io = s->bi_end_io; 134 s->bio->bi_end_io = s->bi_end_io;
135 s->bio->bi_private = s->bi_private; 135 s->bio->bi_private = s->bi_private;
136 bio_endio(s->bio, 0); 136 bio_endio_nodec(s->bio, 0);
137 137
138 closure_debug_destroy(&s->cl); 138 closure_debug_destroy(&s->cl);
139 mempool_free(s, s->p->bio_split_hook); 139 mempool_free(s, s->p->bio_split_hook);
diff --git a/drivers/md/dm-cache-target.c b/drivers/md/dm-cache-target.c
index 86f9c83eb30c..bf3a206abd78 100644
--- a/drivers/md/dm-cache-target.c
+++ b/drivers/md/dm-cache-target.c
@@ -765,6 +765,12 @@ static void writethrough_endio(struct bio *bio, int err)
765 765
766 dm_unhook_bio(&pb->hook_info, bio); 766 dm_unhook_bio(&pb->hook_info, bio);
767 767
768 /*
769 * Must bump bi_remaining to allow bio to complete with
770 * restored bi_end_io.
771 */
772 atomic_inc(&bio->bi_remaining);
773
768 if (err) { 774 if (err) {
769 bio_endio(bio, err); 775 bio_endio(bio, err);
770 return; 776 return;
diff --git a/drivers/md/dm-snap.c b/drivers/md/dm-snap.c
index 3ded8c729dfb..80b5cabbea29 100644
--- a/drivers/md/dm-snap.c
+++ b/drivers/md/dm-snap.c
@@ -1415,6 +1415,7 @@ out:
1415 if (full_bio) { 1415 if (full_bio) {
1416 full_bio->bi_end_io = pe->full_bio_end_io; 1416 full_bio->bi_end_io = pe->full_bio_end_io;
1417 full_bio->bi_private = pe->full_bio_private; 1417 full_bio->bi_private = pe->full_bio_private;
1418 atomic_inc(&full_bio->bi_remaining);
1418 } 1419 }
1419 free_pending_exception(pe); 1420 free_pending_exception(pe);
1420 1421
diff --git a/drivers/md/dm-thin.c b/drivers/md/dm-thin.c
index a65402480c8c..1abb4a24c338 100644
--- a/drivers/md/dm-thin.c
+++ b/drivers/md/dm-thin.c
@@ -611,8 +611,10 @@ static void cell_defer_no_holder(struct thin_c *tc, struct dm_bio_prison_cell *c
611 611
612static void process_prepared_mapping_fail(struct dm_thin_new_mapping *m) 612static void process_prepared_mapping_fail(struct dm_thin_new_mapping *m)
613{ 613{
614 if (m->bio) 614 if (m->bio) {
615 m->bio->bi_end_io = m->saved_bi_end_io; 615 m->bio->bi_end_io = m->saved_bi_end_io;
616 atomic_inc(&m->bio->bi_remaining);
617 }
616 cell_error(m->tc->pool, m->cell); 618 cell_error(m->tc->pool, m->cell);
617 list_del(&m->list); 619 list_del(&m->list);
618 mempool_free(m, m->tc->pool->mapping_pool); 620 mempool_free(m, m->tc->pool->mapping_pool);
@@ -626,8 +628,10 @@ static void process_prepared_mapping(struct dm_thin_new_mapping *m)
626 int r; 628 int r;
627 629
628 bio = m->bio; 630 bio = m->bio;
629 if (bio) 631 if (bio) {
630 bio->bi_end_io = m->saved_bi_end_io; 632 bio->bi_end_io = m->saved_bi_end_io;
633 atomic_inc(&bio->bi_remaining);
634 }
631 635
632 if (m->err) { 636 if (m->err) {
633 cell_error(pool, m->cell); 637 cell_error(pool, m->cell);
diff --git a/drivers/md/dm-verity.c b/drivers/md/dm-verity.c
index ac35e959d49b..796007a5e0e1 100644
--- a/drivers/md/dm-verity.c
+++ b/drivers/md/dm-verity.c
@@ -385,7 +385,7 @@ static void verity_finish_io(struct dm_verity_io *io, int error)
385 bio->bi_end_io = io->orig_bi_end_io; 385 bio->bi_end_io = io->orig_bi_end_io;
386 bio->bi_private = io->orig_bi_private; 386 bio->bi_private = io->orig_bi_private;
387 387
388 bio_endio(bio, error); 388 bio_endio_nodec(bio, error);
389} 389}
390 390
391static void verity_work(struct work_struct *w) 391static void verity_work(struct work_struct *w)