block: Add warning for bi_next not NULL in bio_endio()
authorKent Overstreet <kent.overstreet@gmail.com>
Wed, 9 May 2018 01:33:56 +0000 (21:33 -0400)
committerJens Axboe <axboe@kernel.dk>
Mon, 14 May 2018 19:16:13 +0000 (13:16 -0600)
Recently found a bug where a driver left bi_next not NULL and then
called bio_endio(), and then the submitter of the bio used
bio_copy_data() which was treating src and dst as lists of bios.

Fixed that bug by splitting out bio_list_copy_data(), but in case other
things are depending on bi_next in weird ways, add a warning to help
avoid more bugs like that in the future.

Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Jens Axboe <axboe@kernel.dk>
block/bio.c
block/blk-core.c

index 2112ad01b52cd5d8cd0c20dcebd14b864106efcc..d8bcc12d5aa465576dea2623a1d5954f45c96967 100644 (file)
@@ -1775,6 +1775,9 @@ again:
        if (!bio_integrity_endio(bio))
                return;
 
+       if (WARN_ONCE(bio->bi_next, "driver left bi_next not NULL"))
+               bio->bi_next = NULL;
+
        /*
         * Need to have a real endio function for chained bios, otherwise
         * various corner cases will break (like stacking block devices that
index b431558f39bcdf31d40c5754ed68efdfc47a6614..43370faee93551cdb586ebe48f777209f3988628 100644 (file)
@@ -279,6 +279,10 @@ static void req_bio_endio(struct request *rq, struct bio *bio,
        bio_advance(bio, nbytes);
 
        /* don't actually finish bio if it's part of flush sequence */
+       /*
+        * XXX this code looks suspicious - it's not consistent with advancing
+        * req->bio in caller
+        */
        if (bio->bi_iter.bi_size == 0 && !(rq->rq_flags & RQF_FLUSH_SEQ))
                bio_endio(bio);
 }
@@ -3083,8 +3087,10 @@ bool blk_update_request(struct request *req, blk_status_t error,
                struct bio *bio = req->bio;
                unsigned bio_bytes = min(bio->bi_iter.bi_size, nr_bytes);
 
-               if (bio_bytes == bio->bi_iter.bi_size)
+               if (bio_bytes == bio->bi_iter.bi_size) {
                        req->bio = bio->bi_next;
+                       bio->bi_next = NULL;
+               }
 
                /* Completion has already been traced */
                bio_clear_flag(bio, BIO_TRACE_COMPLETION);