Skip to content

Commit f299b7c

Browse files
committed
blk-mq: provide internal in-flight variant
We don't have to inc/dec some counter, since we can just iterate the tags. That makes inc/dec a noop, but means we have to iterate busy tags to get an in-flight count. Reviewed-by: Bart Van Assche <bart.vanassche@wdc.com> Reviewed-by: Omar Sandoval <osandov@fb.com> Signed-off-by: Jens Axboe <axboe@kernel.dk>
1 parent 0609e0e commit f299b7c

File tree

4 files changed

+77
-29
lines changed

4 files changed

+77
-29
lines changed

block/blk-mq.c

Lines changed: 31 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -83,6 +83,37 @@ static void blk_mq_hctx_clear_pending(struct blk_mq_hw_ctx *hctx,
8383
sbitmap_clear_bit(&hctx->ctx_map, ctx->index_hw);
8484
}
8585

86+
struct mq_inflight {
87+
struct hd_struct *part;
88+
unsigned int *inflight;
89+
};
90+
91+
static void blk_mq_check_inflight(struct blk_mq_hw_ctx *hctx,
92+
struct request *rq, void *priv,
93+
bool reserved)
94+
{
95+
struct mq_inflight *mi = priv;
96+
97+
if (test_bit(REQ_ATOM_STARTED, &rq->atomic_flags) &&
98+
!test_bit(REQ_ATOM_COMPLETE, &rq->atomic_flags)) {
99+
/*
100+
* Count as inflight if it either matches the partition we
101+
* asked for, or if it's the root
102+
*/
103+
if (rq->part == mi->part || mi->part->partno)
104+
mi->inflight[0]++;
105+
}
106+
}
107+
108+
void blk_mq_in_flight(struct request_queue *q, struct hd_struct *part,
109+
unsigned int inflight[2])
110+
{
111+
struct mq_inflight mi = { .part = part, .inflight = inflight, };
112+
113+
inflight[0] = 0;
114+
blk_mq_queue_tag_busy_iter(q, blk_mq_check_inflight, &mi);
115+
}
116+
86117
void blk_freeze_queue_start(struct request_queue *q)
87118
{
88119
int freeze_depth;

block/blk-mq.h

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -133,4 +133,7 @@ static inline bool blk_mq_hw_queue_mapped(struct blk_mq_hw_ctx *hctx)
133133
return hctx->nr_ctx && hctx->tags;
134134
}
135135

136+
void blk_mq_in_flight(struct request_queue *q, struct hd_struct *part,
137+
unsigned int inflight[2]);
138+
136139
#endif

block/genhd.c

Lines changed: 37 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,43 @@ static void disk_add_events(struct gendisk *disk);
4545
static void disk_del_events(struct gendisk *disk);
4646
static void disk_release_events(struct gendisk *disk);
4747

48+
void part_inc_in_flight(struct request_queue *q, struct hd_struct *part, int rw)
49+
{
50+
if (q->mq_ops)
51+
return;
52+
53+
atomic_inc(&part->in_flight[rw]);
54+
if (part->partno)
55+
atomic_inc(&part_to_disk(part)->part0.in_flight[rw]);
56+
}
57+
58+
void part_dec_in_flight(struct request_queue *q, struct hd_struct *part, int rw)
59+
{
60+
if (q->mq_ops)
61+
return;
62+
63+
atomic_dec(&part->in_flight[rw]);
64+
if (part->partno)
65+
atomic_dec(&part_to_disk(part)->part0.in_flight[rw]);
66+
}
67+
68+
void part_in_flight(struct request_queue *q, struct hd_struct *part,
69+
unsigned int inflight[2])
70+
{
71+
if (q->mq_ops) {
72+
blk_mq_in_flight(q, part, inflight);
73+
return;
74+
}
75+
76+
inflight[0] = atomic_read(&part->in_flight[0]) +
77+
atomic_read(&part->in_flight[1]);
78+
if (part->partno) {
79+
part = &part_to_disk(part)->part0;
80+
inflight[1] = atomic_read(&part->in_flight[0]) +
81+
atomic_read(&part->in_flight[1]);
82+
}
83+
}
84+
4885
/**
4986
* disk_get_part - get partition
5087
* @disk: disk to look partition from

include/linux/genhd.h

Lines changed: 6 additions & 29 deletions
Original file line numberDiff line numberDiff line change
@@ -362,35 +362,12 @@ static inline void free_part_stats(struct hd_struct *part)
362362
#define part_stat_sub(cpu, gendiskp, field, subnd) \
363363
part_stat_add(cpu, gendiskp, field, -subnd)
364364

365-
static inline void part_inc_in_flight(struct request_queue *q,
366-
struct hd_struct *part, int rw)
367-
{
368-
atomic_inc(&part->in_flight[rw]);
369-
if (part->partno)
370-
atomic_inc(&part_to_disk(part)->part0.in_flight[rw]);
371-
}
372-
373-
static inline void part_dec_in_flight(struct request_queue *q,
374-
struct hd_struct *part, int rw)
375-
{
376-
atomic_dec(&part->in_flight[rw]);
377-
if (part->partno)
378-
atomic_dec(&part_to_disk(part)->part0.in_flight[rw]);
379-
}
380-
381-
static inline void part_in_flight(struct request_queue *q,
382-
struct hd_struct *part,
383-
unsigned int inflight[2])
384-
{
385-
inflight[0] = atomic_read(&part->in_flight[0]) +
386-
atomic_read(&part->in_flight[1]);
387-
if (part->partno) {
388-
part = &part_to_disk(part)->part0;
389-
inflight[1] = atomic_read(&part->in_flight[0]) +
390-
atomic_read(&part->in_flight[1]);
391-
} else
392-
inflight[1] = 0;
393-
}
365+
void part_in_flight(struct request_queue *q, struct hd_struct *part,
366+
unsigned int inflight[2]);
367+
void part_dec_in_flight(struct request_queue *q, struct hd_struct *part,
368+
int rw);
369+
void part_inc_in_flight(struct request_queue *q, struct hd_struct *part,
370+
int rw);
394371

395372
static inline struct partition_meta_info *alloc_part_info(struct gendisk *disk)
396373
{

0 commit comments

Comments
 (0)