Commit 096bc7ea authored by Damien Le Moal's avatar Damien Le Moal Committed by Jens Axboe

block: Fix handling of non-empty flush write requests to zones

Zone write plugging ignores empty (no data) flush operations but handles
flush BIOs that have data to ensure that the flush machinery generated
write is processed in order. However, the call to
blk_zone_write_plug_attempt_merge() which sets a request
RQF_ZONE_WRITE_PLUGGING flag is called after blk_insert_flush(), thus
missing indicating that a non empty flush request completion needs
handling by zone write plugging.

Fix this by moving the call to blk_zone_write_plug_attempt_merge()
before blk_insert_flush(). And while at it, rename that function as
blk_zone_write_plug_init_request() to be clear that it is not just about
merging plugged BIOs in the request. While at it, also add a WARN_ONCE()
check that the zone write plug for the request is not NULL.

Fixes: dd291d77 ("block: Introduce zone write plugging")
Signed-off-by: default avatarDamien Le Moal <dlemoal@kernel.org>
Reviewed-by: default avatarChristoph Hellwig <hch@lst.de>
Reviewed-by: default avatarJohannes Thumshirn <johannes.thumshirn@wdc.com>
Link: https://lore.kernel.org/r/20240501110907.96950-10-dlemoal@kernel.orgSigned-off-by: default avatarJens Axboe <axboe@kernel.dk>
parent af147b74
...@@ -3001,12 +3001,12 @@ void blk_mq_submit_bio(struct bio *bio) ...@@ -3001,12 +3001,12 @@ void blk_mq_submit_bio(struct bio *bio)
return; return;
} }
if (bio_zone_write_plugging(bio))
blk_zone_write_plug_init_request(rq);
if (op_is_flush(bio->bi_opf) && blk_insert_flush(rq)) if (op_is_flush(bio->bi_opf) && blk_insert_flush(rq))
return; return;
if (bio_zone_write_plugging(bio))
blk_zone_write_plug_attempt_merge(rq);
if (plug) { if (plug) {
blk_add_rq_to_plug(plug, rq); blk_add_rq_to_plug(plug, rq);
return; return;
......
...@@ -874,8 +874,9 @@ void blk_zone_write_plug_bio_merged(struct bio *bio) ...@@ -874,8 +874,9 @@ void blk_zone_write_plug_bio_merged(struct bio *bio)
/* /*
* If the BIO was already plugged, then we were called through * If the BIO was already plugged, then we were called through
* blk_zone_write_plug_attempt_merge() -> blk_attempt_bio_merge(). * blk_zone_write_plug_init_request() -> blk_attempt_bio_merge().
* For this case, blk_zone_write_plug_attempt_merge() will handle the * For this case, we already hold a reference on the zone write plug for
* the BIO and blk_zone_write_plug_init_request() will handle the
* zone write pointer offset update. * zone write pointer offset update.
*/ */
if (bio_flagged(bio, BIO_ZONE_WRITE_PLUGGING)) if (bio_flagged(bio, BIO_ZONE_WRITE_PLUGGING))
...@@ -899,7 +900,7 @@ void blk_zone_write_plug_bio_merged(struct bio *bio) ...@@ -899,7 +900,7 @@ void blk_zone_write_plug_bio_merged(struct bio *bio)
* already went through zone write plugging (either a new BIO or one that was * already went through zone write plugging (either a new BIO or one that was
* unplugged). * unplugged).
*/ */
void blk_zone_write_plug_attempt_merge(struct request *req) void blk_zone_write_plug_init_request(struct request *req)
{ {
sector_t req_back_sector = blk_rq_pos(req) + blk_rq_sectors(req); sector_t req_back_sector = blk_rq_pos(req) + blk_rq_sectors(req);
struct request_queue *q = req->q; struct request_queue *q = req->q;
...@@ -910,6 +911,9 @@ void blk_zone_write_plug_attempt_merge(struct request *req) ...@@ -910,6 +911,9 @@ void blk_zone_write_plug_attempt_merge(struct request *req)
unsigned long flags; unsigned long flags;
struct bio *bio; struct bio *bio;
if (WARN_ON_ONCE(!zwplug))
return;
/* /*
* Indicate that completion of this request needs to be handled with * Indicate that completion of this request needs to be handled with
* blk_zone_write_plug_complete_request(), which will drop the reference * blk_zone_write_plug_complete_request(), which will drop the reference
...@@ -1269,7 +1273,7 @@ void blk_zone_write_plug_complete_request(struct request *req) ...@@ -1269,7 +1273,7 @@ void blk_zone_write_plug_complete_request(struct request *req)
/* /*
* Drop the reference we took when the request was initialized in * Drop the reference we took when the request was initialized in
* blk_zone_write_plug_attempt_merge(). * blk_zone_write_plug_init_request().
*/ */
disk_put_zone_wplug(zwplug); disk_put_zone_wplug(zwplug);
......
...@@ -427,7 +427,7 @@ static inline bool bio_is_zone_append(struct bio *bio) ...@@ -427,7 +427,7 @@ static inline bool bio_is_zone_append(struct bio *bio)
bio_flagged(bio, BIO_EMULATES_ZONE_APPEND); bio_flagged(bio, BIO_EMULATES_ZONE_APPEND);
} }
void blk_zone_write_plug_bio_merged(struct bio *bio); void blk_zone_write_plug_bio_merged(struct bio *bio);
void blk_zone_write_plug_attempt_merge(struct request *rq); void blk_zone_write_plug_init_request(struct request *rq);
static inline void blk_zone_update_request_bio(struct request *rq, static inline void blk_zone_update_request_bio(struct request *rq,
struct bio *bio) struct bio *bio)
{ {
...@@ -481,7 +481,7 @@ static inline bool bio_is_zone_append(struct bio *bio) ...@@ -481,7 +481,7 @@ static inline bool bio_is_zone_append(struct bio *bio)
static inline void blk_zone_write_plug_bio_merged(struct bio *bio) static inline void blk_zone_write_plug_bio_merged(struct bio *bio)
{ {
} }
static inline void blk_zone_write_plug_attempt_merge(struct request *rq) static inline void blk_zone_write_plug_init_request(struct request *rq)
{ {
} }
static inline void blk_zone_update_request_bio(struct request *rq, static inline void blk_zone_update_request_bio(struct request *rq,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment