aboutsummaryrefslogtreecommitdiffstats
path: root/drivers
diff options
context:
space:
mode:
Diffstat (limited to 'drivers')
-rw-r--r--drivers/md/dm-crypt.c41
1 files changed, 40 insertions, 1 deletions
diff --git a/drivers/md/dm-crypt.c b/drivers/md/dm-crypt.c
index ad98ded3008..046ee516074 100644
--- a/drivers/md/dm-crypt.c
+++ b/drivers/md/dm-crypt.c
@@ -56,6 +56,7 @@ struct dm_crypt_io {
56 atomic_t pending; 56 atomic_t pending;
57 int error; 57 int error;
58 sector_t sector; 58 sector_t sector;
59 struct dm_crypt_io *base_io;
59}; 60};
60 61
61struct dm_crypt_request { 62struct dm_crypt_request {
@@ -534,6 +535,7 @@ static struct dm_crypt_io *crypt_io_alloc(struct dm_target *ti,
534 io->base_bio = bio; 535 io->base_bio = bio;
535 io->sector = sector; 536 io->sector = sector;
536 io->error = 0; 537 io->error = 0;
538 io->base_io = NULL;
537 atomic_set(&io->pending, 0); 539 atomic_set(&io->pending, 0);
538 540
539 return io; 541 return io;
@@ -547,6 +549,7 @@ static void crypt_inc_pending(struct dm_crypt_io *io)
547/* 549/*
548 * One of the bios was finished. Check for completion of 550 * One of the bios was finished. Check for completion of
549 * the whole request and correctly clean up the buffer. 551 * the whole request and correctly clean up the buffer.
552 * If base_io is set, wait for the last fragment to complete.
550 */ 553 */
551static void crypt_dec_pending(struct dm_crypt_io *io) 554static void crypt_dec_pending(struct dm_crypt_io *io)
552{ 555{
@@ -555,7 +558,14 @@ static void crypt_dec_pending(struct dm_crypt_io *io)
555 if (!atomic_dec_and_test(&io->pending)) 558 if (!atomic_dec_and_test(&io->pending))
556 return; 559 return;
557 560
558 bio_endio(io->base_bio, io->error); 561 if (likely(!io->base_io))
562 bio_endio(io->base_bio, io->error);
563 else {
564 if (io->error && !io->base_io->error)
565 io->base_io->error = io->error;
566 crypt_dec_pending(io->base_io);
567 }
568
559 mempool_free(io, cc->io_pool); 569 mempool_free(io, cc->io_pool);
560} 570}
561 571
@@ -699,6 +709,7 @@ static void kcryptd_crypt_write_convert(struct dm_crypt_io *io)
699{ 709{
700 struct crypt_config *cc = io->target->private; 710 struct crypt_config *cc = io->target->private;
701 struct bio *clone; 711 struct bio *clone;
712 struct dm_crypt_io *new_io;
702 int crypt_finished; 713 int crypt_finished;
703 unsigned out_of_pages = 0; 714 unsigned out_of_pages = 0;
704 unsigned remaining = io->base_bio->bi_size; 715 unsigned remaining = io->base_bio->bi_size;
@@ -753,6 +764,34 @@ static void kcryptd_crypt_write_convert(struct dm_crypt_io *io)
753 if (unlikely(out_of_pages)) 764 if (unlikely(out_of_pages))
754 congestion_wait(WRITE, HZ/100); 765 congestion_wait(WRITE, HZ/100);
755 766
767 /*
768 * With async crypto it is unsafe to share the crypto context
769 * between fragments, so switch to a new dm_crypt_io structure.
770 */
771 if (unlikely(!crypt_finished && remaining)) {
772 new_io = crypt_io_alloc(io->target, io->base_bio,
773 sector);
774 crypt_inc_pending(new_io);
775 crypt_convert_init(cc, &new_io->ctx, NULL,
776 io->base_bio, sector);
777 new_io->ctx.idx_in = io->ctx.idx_in;
778 new_io->ctx.offset_in = io->ctx.offset_in;
779
780 /*
781 * Fragments after the first use the base_io
782 * pending count.
783 */
784 if (!io->base_io)
785 new_io->base_io = io;
786 else {
787 new_io->base_io = io->base_io;
788 crypt_inc_pending(io->base_io);
789 crypt_dec_pending(io);
790 }
791
792 io = new_io;
793 }
794
756 if (unlikely(remaining)) 795 if (unlikely(remaining))
757 wait_event(cc->writeq, !atomic_read(&io->ctx.pending)); 796 wait_event(cc->writeq, !atomic_read(&io->ctx.pending));
758 } 797 }