]> git.baikalelectronics.ru Git - kernel.git/commitdiff
bio: don't copy bvec for direct IO
authorPavel Begunkov <asml.silence@gmail.com>
Sat, 9 Jan 2021 16:03:03 +0000 (16:03 +0000)
committerJens Axboe <axboe@kernel.dk>
Mon, 25 Jan 2021 15:58:24 +0000 (08:58 -0700)
The block layer spends quite a while in blkdev_direct_IO() to copy and
initialise bio's bvec. However, if we've already got a bvec in the input
iterator it might be reused in some cases, i.e. when new
ITER_BVEC_FLAG_FIXED flag is set. Simple tests show considerable
performance boost, and it also reduces memory footprint.

Suggested-by: Matthew Wilcox <willy@infradead.org>
Reviewed-by: Christoph Hellwig <hch@lst.de>
Signed-off-by: Pavel Begunkov <asml.silence@gmail.com>
Reviewed-by: Ming Lei <ming.lei@redhat.com>
Signed-off-by: Jens Axboe <axboe@kernel.dk>
Documentation/filesystems/porting.rst
block/bio.c
include/linux/bio.h

index c722d94f29ead3819e359e87fcfedf8b60125d8d..1f8cf8e10b340acc7e315c3c9e166896567f5d2c 100644 (file)
@@ -872,3 +872,12 @@ its result is kern_unmount() or kern_unmount_array().
 
 zero-length bvec segments are disallowed, they must be filtered out before
 passed on to an iterator.
+
+---
+
+**mandatory**
+
+For bvec based itererators bio_iov_iter_get_pages() now doesn't copy bvecs but
+uses the one provided. Anyone issuing kiocb-I/O should ensure that the bvec and
+page references stay until I/O has completed, i.e. until ->ki_complete() has
+been called or returned with non -EIOCBQUEUED code.
index 1cd8a2e79048ddbdca54b9f3153996c30126113c..99040a7e6656a1b884aaccf6c5838550d85acb16 100644 (file)
@@ -942,21 +942,17 @@ void bio_release_pages(struct bio *bio, bool mark_dirty)
 }
 EXPORT_SYMBOL_GPL(bio_release_pages);
 
-static int __bio_iov_bvec_add_pages(struct bio *bio, struct iov_iter *iter)
+static int bio_iov_bvec_set(struct bio *bio, struct iov_iter *iter)
 {
-       const struct bio_vec *bv = iter->bvec;
-       unsigned int len;
-       size_t size;
-
-       if (WARN_ON_ONCE(iter->iov_offset > bv->bv_len))
-               return -EINVAL;
-
-       len = min_t(size_t, bv->bv_len - iter->iov_offset, iter->count);
-       size = bio_add_page(bio, bv->bv_page, len,
-                               bv->bv_offset + iter->iov_offset);
-       if (unlikely(size != len))
-               return -EINVAL;
-       iov_iter_advance(iter, size);
+       WARN_ON_ONCE(BVEC_POOL_IDX(bio) != 0);
+
+       bio->bi_vcnt = iter->nr_segs;
+       bio->bi_max_vecs = iter->nr_segs;
+       bio->bi_io_vec = (struct bio_vec *)iter->bvec;
+       bio->bi_iter.bi_bvec_done = iter->iov_offset;
+       bio->bi_iter.bi_size = iter->count;
+
+       iov_iter_advance(iter, iter->count);
        return 0;
 }
 
@@ -1070,12 +1066,12 @@ static int __bio_iov_append_get_pages(struct bio *bio, struct iov_iter *iter)
  * This takes either an iterator pointing to user memory, or one pointing to
  * kernel pages (BVEC iterator). If we're adding user pages, we pin them and
  * map them into the kernel. On IO completion, the caller should put those
- * pages. If we're adding kernel pages, and the caller told us it's safe to
- * do so, we just have to add the pages to the bio directly. We don't grab an
- * extra reference to those pages (the user should already have that), and we
- * don't put the page on IO completion. The caller needs to check if the bio is
- * flagged BIO_NO_PAGE_REF on IO completion. If it isn't, then pages should be
- * released.
+ * pages. For bvec based iterators bio_iov_iter_get_pages() uses the provided
+ * bvecs rather than copying them. Hence anyone issuing kiocb based IO needs
+ * to ensure the bvecs and pages stay referenced until the submitted I/O is
+ * completed by a call to ->ki_complete() or returns with an error other than
+ * -EIOCBQUEUED. The caller needs to check if the bio is flagged BIO_NO_PAGE_REF
+ * on IO completion. If it isn't, then pages should be released.
  *
  * The function tries, but does not guarantee, to pin as many pages as
  * fit into the bio, or are requested in @iter, whatever is smaller. If
@@ -1087,27 +1083,22 @@ static int __bio_iov_append_get_pages(struct bio *bio, struct iov_iter *iter)
  */
 int bio_iov_iter_get_pages(struct bio *bio, struct iov_iter *iter)
 {
-       const bool is_bvec = iov_iter_is_bvec(iter);
-       int ret;
-
-       if (WARN_ON_ONCE(bio->bi_vcnt))
-               return -EINVAL;
+       int ret = 0;
 
-       do {
-               if (bio_op(bio) == REQ_OP_ZONE_APPEND) {
-                       if (WARN_ON_ONCE(is_bvec))
-                               return -EINVAL;
-                       ret = __bio_iov_append_get_pages(bio, iter);
-               } else {
-                       if (is_bvec)
-                               ret = __bio_iov_bvec_add_pages(bio, iter);
+       if (iov_iter_is_bvec(iter)) {
+               if (WARN_ON_ONCE(bio_op(bio) == REQ_OP_ZONE_APPEND))
+                       return -EINVAL;
+               bio_iov_bvec_set(bio, iter);
+               bio_set_flag(bio, BIO_NO_PAGE_REF);
+               return 0;
+       } else {
+               do {
+                       if (bio_op(bio) == REQ_OP_ZONE_APPEND)
+                               ret = __bio_iov_append_get_pages(bio, iter);
                        else
                                ret = __bio_iov_iter_get_pages(bio, iter);
-               }
-       } while (!ret && iov_iter_count(iter) && !bio_full(bio, 0));
-
-       if (is_bvec)
-               bio_set_flag(bio, BIO_NO_PAGE_REF);
+               } while (!ret && iov_iter_count(iter) && !bio_full(bio, 0));
+       }
 
        /* don't account direct I/O as memory stall */
        bio_clear_flag(bio, BIO_WORKINGSET);
index 9ddb19801a0320a97b9a0045f16a01e48567c06f..676870b2c88d807e1fa9d2b6ce2fedf81d6d0058 100644 (file)
@@ -444,10 +444,13 @@ static inline void bio_wouldblock_error(struct bio *bio)
 
 /*
  * Calculate number of bvec segments that should be allocated to fit data
- * pointed by @iter.
+ * pointed by @iter. If @iter is backed by bvec it's going to be reused
+ * instead of allocating a new one.
  */
 static inline int bio_iov_vecs_to_alloc(struct iov_iter *iter, int max_segs)
 {
+       if (iov_iter_is_bvec(iter))
+               return 0;
        return iov_iter_npages(iter, max_segs);
 }