提交 60b6a7e6 编写于 作者: E Edward Hsieh 提交者: Jens Axboe

block: fix trace completion for chained bio

For chained bio, trace_block_bio_complete in bio_endio is currently called
only by the parent bio once upon all chained bio completed.
However, the sector and size for the parent bio are modified in bio_split.
Therefore, the size and sector of the complete events might not match the
queue events in blktrace.

The original fix of bio completion trace <fbbaf700> ("block: trace
completion of all bios.") wants multiple complete events to correspond
to one queue event but missed this.

The issue can be reproduced by md/raid5 read with bio cross chunks.

To fix, move trace completion into the loop for every chained bio to call.

Fixes: fbbaf700 ("block: trace completion of all bios.")
Reviewed-by: NWade Liang <wadel@synology.com>
Reviewed-by: NBingJing Chang <bingjingc@synology.com>
Signed-off-by: NEdward Hsieh <edwardh@synology.com>
Reviewed-by: NChristoph Hellwig <hch@lst.de>
Link: https://lore.kernel.org/r/20210624123030.27014-1-edwardh@synology.comSigned-off-by: NJens Axboe <axboe@kernel.dk>
上级 ddcc5c54
...@@ -1375,8 +1375,7 @@ static inline bool bio_remaining_done(struct bio *bio) ...@@ -1375,8 +1375,7 @@ static inline bool bio_remaining_done(struct bio *bio)
* *
* bio_endio() can be called several times on a bio that has been chained * bio_endio() can be called several times on a bio that has been chained
* using bio_chain(). The ->bi_end_io() function will only be called the * using bio_chain(). The ->bi_end_io() function will only be called the
* last time. At this point the BLK_TA_COMPLETE tracing event will be * last time.
* generated if BIO_TRACE_COMPLETION is set.
**/ **/
void bio_endio(struct bio *bio) void bio_endio(struct bio *bio)
{ {
...@@ -1389,6 +1388,11 @@ void bio_endio(struct bio *bio) ...@@ -1389,6 +1388,11 @@ void bio_endio(struct bio *bio)
if (bio->bi_bdev) if (bio->bi_bdev)
rq_qos_done_bio(bio->bi_bdev->bd_disk->queue, bio); rq_qos_done_bio(bio->bi_bdev->bd_disk->queue, bio);
if (bio->bi_bdev && bio_flagged(bio, BIO_TRACE_COMPLETION)) {
trace_block_bio_complete(bio->bi_bdev->bd_disk->queue, bio);
bio_clear_flag(bio, BIO_TRACE_COMPLETION);
}
/* /*
* Need to have a real endio function for chained bios, otherwise * Need to have a real endio function for chained bios, otherwise
* various corner cases will break (like stacking block devices that * various corner cases will break (like stacking block devices that
...@@ -1402,11 +1406,6 @@ void bio_endio(struct bio *bio) ...@@ -1402,11 +1406,6 @@ void bio_endio(struct bio *bio)
goto again; goto again;
} }
if (bio->bi_bdev && bio_flagged(bio, BIO_TRACE_COMPLETION)) {
trace_block_bio_complete(bio->bi_bdev->bd_disk->queue, bio);
bio_clear_flag(bio, BIO_TRACE_COMPLETION);
}
blk_throtl_bio_endio(bio); blk_throtl_bio_endio(bio);
/* release cgroup info */ /* release cgroup info */
bio_uninit(bio); bio_uninit(bio);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册