bio_iov_iter_get_pages() returns only pages for a single non-empty
segment of the input iov_iter's iovec. This may be less than the
number of pages __blkdev_direct_IO_simple() is supposed to process.
Call the new bio_iov_iter_get_all_pages() helper instead to avoid
short reads or writes. Otherwise, __generic_file_write_iter() falls
back to buffered writes, which has been observed to cause data
corruption in certain workloads.

Fixes: 72ecad22d9f1 ("block: support a full bio worth of IO for
 simplified bdev direct-io")
Signed-off-by: Martin Wilck <[email protected]>
---
 fs/block_dev.c | 9 ++++++++-
 1 file changed, 8 insertions(+), 1 deletion(-)

diff --git a/fs/block_dev.c b/fs/block_dev.c
index aba2541..010708a 100644
--- a/fs/block_dev.c
+++ b/fs/block_dev.c
@@ -219,9 +219,16 @@ __blkdev_direct_IO_simple(struct kiocb *iocb, struct 
iov_iter *iter,
        bio.bi_end_io = blkdev_bio_end_io_simple;
        bio.bi_ioprio = iocb->ki_ioprio;
 
-       ret = bio_iov_iter_get_pages(&bio, iter);
+       ret = bio_iov_iter_get_all_pages(&bio, iter);
        if (unlikely(ret))
                goto out;
+
+       /*
+        * Our bi_io_vec should be big enough to hold all data from the
+        * iov_iter, as this has been checked before calling this function.
+        */
+       WARN_ON_ONCE(iov_iter_count(iter));
+
        ret = bio.bi_iter.bi_size;
 
        if (iov_iter_rw(iter) == READ) {
-- 
2.17.1

Reply via email to