]> git.baikalelectronics.ru Git - kernel.git/commitdiff
block: change the blk_queue_split calling convention
authorChristoph Hellwig <hch@lst.de>
Wed, 27 Jul 2022 16:22:55 +0000 (12:22 -0400)
committerJens Axboe <axboe@kernel.dk>
Tue, 2 Aug 2022 23:22:53 +0000 (17:22 -0600)
The double indirect bio leads to somewhat suboptimal code generation.
Instead return the (original or split) bio, and make sure the
request_queue arguments to the lower level helpers is passed after the
bio to avoid constant reshuffling of the argument passing registers.

Also give it and the helpers used to implement it more descriptive names.

Signed-off-by: Christoph Hellwig <hch@lst.de>
Link: https://lore.kernel.org/r/20220727162300.3089193-2-hch@lst.de
Signed-off-by: Jens Axboe <axboe@kernel.dk>
block/blk-merge.c
block/blk-mq.c
block/blk.h
drivers/block/drbd/drbd_req.c
drivers/block/pktcdvd.c
drivers/block/ps3vram.c
drivers/md/dm.c
drivers/md/md.c
drivers/nvme/host/multipath.c
drivers/s390/block/dcssblk.c
include/linux/blkdev.h

index 4c8a699754c9706783711c49987e5d75cf53b66d..6e29fb28584ef8551a81d4f059cc79769e191a96 100644 (file)
@@ -95,10 +95,8 @@ static inline bool req_gap_front_merge(struct request *req, struct bio *bio)
        return bio_will_gap(req->q, NULL, bio, req->bio);
 }
 
-static struct bio *blk_bio_discard_split(struct request_queue *q,
-                                        struct bio *bio,
-                                        struct bio_set *bs,
-                                        unsigned *nsegs)
+static struct bio *bio_split_discard(struct bio *bio, struct request_queue *q,
+               unsigned *nsegs, struct bio_set *bs)
 {
        unsigned int max_discard_sectors, granularity;
        int alignment;
@@ -139,8 +137,8 @@ static struct bio *blk_bio_discard_split(struct request_queue *q,
        return bio_split(bio, split_sectors, GFP_NOIO, bs);
 }
 
-static struct bio *blk_bio_write_zeroes_split(struct request_queue *q,
-               struct bio *bio, struct bio_set *bs, unsigned *nsegs)
+static struct bio *bio_split_write_zeroes(struct bio *bio,
+               struct request_queue *q, unsigned *nsegs, struct bio_set *bs)
 {
        *nsegs = 0;
 
@@ -161,8 +159,8 @@ static struct bio *blk_bio_write_zeroes_split(struct request_queue *q,
  * requests that are submitted to a block device if the start of a bio is not
  * aligned to a physical block boundary.
  */
-static inline unsigned get_max_io_size(struct request_queue *q,
-                                      struct bio *bio)
+static inline unsigned get_max_io_size(struct bio *bio,
+               struct request_queue *q)
 {
        unsigned pbs = queue_physical_block_size(q) >> SECTOR_SHIFT;
        unsigned lbs = queue_logical_block_size(q) >> SECTOR_SHIFT;
@@ -247,16 +245,16 @@ static bool bvec_split_segs(const struct request_queue *q,
 }
 
 /**
- * blk_bio_segment_split - split a bio in two bios
- * @q:    [in] request queue pointer
+ * bio_split_rw - split a bio in two bios
  * @bio:  [in] bio to be split
- * @bs:          [in] bio set to allocate the clone from
+ * @q:    [in] request queue pointer
  * @segs: [out] number of segments in the bio with the first half of the sectors
+ * @bs:          [in] bio set to allocate the clone from
  *
  * Clone @bio, update the bi_iter of the clone to represent the first sectors
  * of @bio and update @bio->bi_iter to represent the remaining sectors. The
  * following is guaranteed for the cloned bio:
- * - That it has at most get_max_io_size(@q, @bio) sectors.
+ * - That it has at most get_max_io_size(@bio, @q) sectors.
  * - That it has at most queue_max_segments(@q) segments.
  *
  * Except for discard requests the cloned bio will point at the bi_io_vec of
@@ -265,15 +263,13 @@ static bool bvec_split_segs(const struct request_queue *q,
  * responsible for ensuring that @bs is only destroyed after processing of the
  * split bio has finished.
  */
-static struct bio *blk_bio_segment_split(struct request_queue *q,
-                                        struct bio *bio,
-                                        struct bio_set *bs,
-                                        unsigned *segs)
+static struct bio *bio_split_rw(struct bio *bio, struct request_queue *q,
+               unsigned *segs, struct bio_set *bs)
 {
        struct bio_vec bv, bvprv, *bvprvp = NULL;
        struct bvec_iter iter;
        unsigned nsegs = 0, bytes = 0;
-       const unsigned max_bytes = get_max_io_size(q, bio) << 9;
+       const unsigned max_bytes = get_max_io_size(bio, q) << 9;
        const unsigned max_segs = queue_max_segments(q);
 
        bio_for_each_bvec(bv, bio, iter) {
@@ -320,34 +316,33 @@ split:
 }
 
 /**
- * __blk_queue_split - split a bio and submit the second half
- * @q:       [in] request_queue new bio is being queued at
- * @bio:     [in, out] bio to be split
- * @nr_segs: [out] number of segments in the first bio
+ * __bio_split_to_limits - split a bio to fit the queue limits
+ * @bio:     bio to be split
+ * @q:       request_queue new bio is being queued at
+ * @nr_segs: returns the number of segments in the returned bio
+ *
+ * Check if @bio needs splitting based on the queue limits, and if so split off
+ * a bio fitting the limits from the beginning of @bio and return it.  @bio is
+ * shortened to the remainder and re-submitted.
  *
- * Split a bio into two bios, chain the two bios, submit the second half and
- * store a pointer to the first half in *@bio. If the second bio is still too
- * big it will be split by a recursive call to this function. Since this
- * function may allocate a new bio from q->bio_split, it is the responsibility
- * of the caller to ensure that q->bio_split is only released after processing
- * of the split bio has finished.
+ * The split bio is allocated from @q->bio_split, which is provided by the
+ * block layer.
  */
-void __blk_queue_split(struct request_queue *q, struct bio **bio,
+struct bio *__bio_split_to_limits(struct bio *bio, struct request_queue *q,
                       unsigned int *nr_segs)
 {
-       struct bio *split = NULL;
+       struct bio *split;
 
-       switch (bio_op(*bio)) {
+       switch (bio_op(bio)) {
        case REQ_OP_DISCARD:
        case REQ_OP_SECURE_ERASE:
-               split = blk_bio_discard_split(q, *bio, &q->bio_split, nr_segs);
+               split = bio_split_discard(bio, q, nr_segs, &q->bio_split);
                break;
        case REQ_OP_WRITE_ZEROES:
-               split = blk_bio_write_zeroes_split(q, *bio, &q->bio_split,
-                               nr_segs);
+               split = bio_split_write_zeroes(bio, q, nr_segs, &q->bio_split);
                break;
        default:
-               split = blk_bio_segment_split(q, *bio, &q->bio_split, nr_segs);
+               split = bio_split_rw(bio, q, nr_segs, &q->bio_split);
                break;
        }
 
@@ -356,32 +351,35 @@ void __blk_queue_split(struct request_queue *q, struct bio **bio,
                split->bi_opf |= REQ_NOMERGE;
 
                blkcg_bio_issue_init(split);
-               bio_chain(split, *bio);
-               trace_block_split(split, (*bio)->bi_iter.bi_sector);
-               submit_bio_noacct(*bio);
-               *bio = split;
+               bio_chain(split, bio);
+               trace_block_split(split, bio->bi_iter.bi_sector);
+               submit_bio_noacct(bio);
+               return split;
        }
+       return bio;
 }
 
 /**
- * blk_queue_split - split a bio and submit the second half
- * @bio: [in, out] bio to be split
+ * bio_split_to_limits - split a bio to fit the queue limits
+ * @bio:     bio to be split
+ *
+ * Check if @bio needs splitting based on the queue limits of @bio->bi_bdev, and
+ * if so split off a bio fitting the limits from the beginning of @bio and
+ * return it.  @bio is shortened to the remainder and re-submitted.
  *
- * Split a bio into two bios, chains the two bios, submit the second half and
- * store a pointer to the first half in *@bio. Since this function may allocate
- * a new bio from q->bio_split, it is the responsibility of the caller to ensure
- * that q->bio_split is only released after processing of the split bio has
- * finished.
+ * The split bio is allocated from @q->bio_split, which is provided by the
+ * block layer.
  */
-void blk_queue_split(struct bio **bio)
+struct bio *bio_split_to_limits(struct bio *bio)
 {
-       struct request_queue *q = bdev_get_queue((*bio)->bi_bdev);
+       struct request_queue *q = bdev_get_queue(bio->bi_bdev);
        unsigned int nr_segs;
 
-       if (blk_may_split(q, *bio))
-               __blk_queue_split(q, bio, &nr_segs);
+       if (bio_may_exceed_limits(bio, q))
+               return __bio_split_to_limits(bio, q, &nr_segs);
+       return bio;
 }
-EXPORT_SYMBOL(blk_queue_split);
+EXPORT_SYMBOL(bio_split_to_limits);
 
 unsigned int blk_recalc_rq_segments(struct request *rq)
 {
index 70177ee74295ba76b5973d05d29daab23b2d0654..790f55453f1b1ebd6411be3ec209fddc8e656330 100644 (file)
@@ -2816,8 +2816,8 @@ void blk_mq_submit_bio(struct bio *bio)
        blk_status_t ret;
 
        blk_queue_bounce(q, &bio);
-       if (blk_may_split(q, bio))
-               __blk_queue_split(q, &bio, &nr_segs);
+       if (bio_may_exceed_limits(bio, q))
+               bio = __bio_split_to_limits(bio, q, &nr_segs);
 
        if (!bio_integrity_prep(bio))
                return;
index 1d83b1d41cd101f6937ff22149cd6c8c7f02af4e..623be4c2e60c19e88bf8d10727f9a319ba2abdf7 100644 (file)
@@ -288,7 +288,7 @@ ssize_t part_timeout_show(struct device *, struct device_attribute *, char *);
 ssize_t part_timeout_store(struct device *, struct device_attribute *,
                                const char *, size_t);
 
-static inline bool blk_may_split(struct request_queue *q, struct bio *bio)
+static inline bool bio_may_exceed_limits(struct bio *bio, struct request_queue *q)
 {
        switch (bio_op(bio)) {
        case REQ_OP_DISCARD:
@@ -311,8 +311,8 @@ static inline bool blk_may_split(struct request_queue *q, struct bio *bio)
                bio->bi_io_vec->bv_len + bio->bi_io_vec->bv_offset > PAGE_SIZE;
 }
 
-void __blk_queue_split(struct request_queue *q, struct bio **bio,
-                       unsigned int *nr_segs);
+struct bio *__bio_split_to_limits(struct bio *bio, struct request_queue *q,
+                      unsigned int *nr_segs);
 int ll_back_merge_fn(struct request *req, struct bio *bio,
                unsigned int nr_segs);
 bool blk_attempt_req_merge(struct request_queue *q, struct request *rq,
index 6d8dd14458c69096a4a771f297bcc5c9d1b79f45..8f7f144e54f3a801bf1dc6e64438954f2b082f59 100644 (file)
@@ -1608,7 +1608,7 @@ void drbd_submit_bio(struct bio *bio)
 {
        struct drbd_device *device = bio->bi_bdev->bd_disk->private_data;
 
-       blk_queue_split(&bio);
+       bio = bio_split_to_limits(bio);
 
        /*
         * what we "blindly" assume:
index 01a15dbd9cde2e007fb424870b541a4baf9fffdb..4cea3b08087ed027cf523e808a4593595edb77a2 100644 (file)
@@ -2399,7 +2399,7 @@ static void pkt_submit_bio(struct bio *bio)
        struct pktcdvd_device *pd = bio->bi_bdev->bd_disk->queue->queuedata;
        struct bio *split;
 
-       blk_queue_split(&bio);
+       bio = bio_split_to_limits(bio);
 
        pkt_dbg(2, pd, "start = %6llx stop = %6llx\n",
                (unsigned long long)bio->bi_iter.bi_sector,
index d1e0fefec90ba22d89ad06ae6ea5f3bc930b5037..e1d080f680edfdd257e49f8ba3d69429be9e12a0 100644 (file)
@@ -586,7 +586,7 @@ static void ps3vram_submit_bio(struct bio *bio)
 
        dev_dbg(&dev->core, "%s\n", __func__);
 
-       blk_queue_split(&bio);
+       bio = bio_split_to_limits(bio);
 
        spin_lock_irq(&priv->lock);
        busy = !bio_list_empty(&priv->list);
index 99642f69bfa7f0f0c61ad93f53709f311d369956..fbcffcfb2304241dee20e3ecd746b4516aaed4d0 100644 (file)
@@ -1181,7 +1181,7 @@ static sector_t max_io_len(struct dm_target *ti, sector_t sector)
         * Does the target need to split IO even further?
         * - varied (per target) IO splitting is a tenet of DM; this
         *   explains why stacked chunk_sectors based splitting via
-        *   blk_queue_split() isn't possible here.
+        *   bio_split_to_limits() isn't possible here.
         */
        if (!ti->max_io_len)
                return len;
@@ -1751,10 +1751,10 @@ static void dm_split_and_process_bio(struct mapped_device *md,
        is_abnormal = is_abnormal_io(bio);
        if (unlikely(is_abnormal)) {
                /*
-                * Use blk_queue_split() for abnormal IO (e.g. discard, etc)
+                * Use bio_split_to_limits() for abnormal IO (e.g. discard, etc)
                 * otherwise associated queue_limits won't be imposed.
                 */
-               blk_queue_split(&bio);
+               bio = bio_split_to_limits(bio);
        }
 
        init_clone_info(&ci, md, map, bio, is_abnormal);
index 35b895813c88bc1abb7997e8eafb7107309e0b71..afaf36b2f6ab8b32d53edc24e12ba17fa7a16884 100644 (file)
@@ -442,7 +442,7 @@ static void md_submit_bio(struct bio *bio)
                return;
        }
 
-       blk_queue_split(&bio);
+       bio = bio_split_to_limits(bio);
 
        if (mddev->ro == 1 && unlikely(rw == WRITE)) {
                if (bio_sectors(bio) != 0)
index 64f44d98daaf7b3a9e85b11bfc700f87c9d3acb3..6ef497c75a1655529ef7010bb5fcf07beb6127a5 100644 (file)
@@ -346,7 +346,7 @@ static void nvme_ns_head_submit_bio(struct bio *bio)
         * different queue via blk_steal_bios(), so we need to use the bio_split
         * pool from the original queue to allocate the bvecs from.
         */
-       blk_queue_split(&bio);
+       bio = bio_split_to_limits(bio);
 
        srcu_idx = srcu_read_lock(&head->srcu);
        ns = nvme_find_path(head);
index 4d8d1759775ae7c6b5a2ce16dfb99a42e4e1cd08..5187705bd0f39294026348d354f3f6bce4c578fb 100644 (file)
@@ -863,7 +863,7 @@ dcssblk_submit_bio(struct bio *bio)
        unsigned long source_addr;
        unsigned long bytes_done;
 
-       blk_queue_split(&bio);
+       bio = bio_split_to_limits(bio);
 
        bytes_done = 0;
        dev_info = bio->bi_bdev->bd_disk->private_data;
index d04bdf549efa9511fc5b7f91ab959018a9d222b2..5eef8d2eddc1c6a5a6f0635c0b02a29d04ecdc48 100644 (file)
@@ -864,9 +864,9 @@ void blk_request_module(dev_t devt);
 extern int blk_register_queue(struct gendisk *disk);
 extern void blk_unregister_queue(struct gendisk *disk);
 void submit_bio_noacct(struct bio *bio);
+struct bio *bio_split_to_limits(struct bio *bio);
 
 extern int blk_lld_busy(struct request_queue *q);
-extern void blk_queue_split(struct bio **);
 extern int blk_queue_enter(struct request_queue *q, blk_mq_req_flags_t flags);
 extern void blk_queue_exit(struct request_queue *q);
 extern void blk_sync_queue(struct request_queue *q);