* [PATCHSET 0/3] blk-wbt fixes
@ 2018-08-21 15:05 Jens Axboe
2018-08-21 15:05 ` [PATCH 1/3] blk-wbt: move disable check into get_limit() Jens Axboe
` (2 more replies)
0 siblings, 3 replies; 4+ messages in thread
From: Jens Axboe @ 2018-08-21 15:05 UTC (permalink / raw)
To: linux-block
One cleanup, and two fixes:
Patch 2 - Ensure that the wait queue active check includes the necessary
memory barrier.
Patch 3 - Ensure that new tasks don't supersede already waiting tasks.
--
Jens Axboe
^ permalink raw reply [flat|nested] 4+ messages in thread
* [PATCH 1/3] blk-wbt: move disable check into get_limit()
2018-08-21 15:05 [PATCHSET 0/3] blk-wbt fixes Jens Axboe
@ 2018-08-21 15:05 ` Jens Axboe
2018-08-21 15:05 ` [PATCH 2/3] blk-wbt: use wq_has_sleeper() for wq active check Jens Axboe
2018-08-21 15:05 ` [PATCH 3/3] blk-wbt: fix has-sleeper queueing check Jens Axboe
2 siblings, 0 replies; 4+ messages in thread
From: Jens Axboe @ 2018-08-21 15:05 UTC (permalink / raw)
To: linux-block; +Cc: Jens Axboe
Check it in one place, instead of in multiple places.
Signed-off-by: Jens Axboe <axboe@kernel.dk>
---
block/blk-wbt.c | 22 +++++++---------------
1 file changed, 7 insertions(+), 15 deletions(-)
diff --git a/block/blk-wbt.c b/block/blk-wbt.c
index bb93c7c2b182..1dd7edce7a99 100644
--- a/block/blk-wbt.c
+++ b/block/blk-wbt.c
@@ -449,6 +449,13 @@ static inline unsigned int get_limit(struct rq_wb *rwb, unsigned long rw)
{
unsigned int limit;
+ /*
+ * If we got disabled, just return UINT_MAX. This ensures that
+ * we'll properly inc a new IO, and dec+wakeup at the end.
+ */
+ if (!rwb_enabled(rwb))
+ return UINT_MAX;
+
if ((rw & REQ_OP_MASK) == REQ_OP_DISCARD)
return rwb->wb_background;
@@ -486,16 +493,6 @@ static void __wbt_wait(struct rq_wb *rwb, enum wbt_flags wb_acct,
struct rq_wait *rqw = get_rq_wait(rwb, wb_acct);
DECLARE_WAITQUEUE(wait, current);
- /*
- * inc it here even if disabled, since we'll dec it at completion.
- * this only happens if the task was sleeping in __wbt_wait(),
- * and someone turned it off at the same time.
- */
- if (!rwb_enabled(rwb)) {
- atomic_inc(&rqw->inflight);
- return;
- }
-
if (!waitqueue_active(&rqw->wait)
&& rq_wait_inc_below(rqw, get_limit(rwb, rw)))
return;
@@ -504,11 +501,6 @@ static void __wbt_wait(struct rq_wb *rwb, enum wbt_flags wb_acct,
do {
set_current_state(TASK_UNINTERRUPTIBLE);
- if (!rwb_enabled(rwb)) {
- atomic_inc(&rqw->inflight);
- break;
- }
-
if (rq_wait_inc_below(rqw, get_limit(rwb, rw)))
break;
--
2.17.1
^ permalink raw reply related [flat|nested] 4+ messages in thread
* [PATCH 2/3] blk-wbt: use wq_has_sleeper() for wq active check
2018-08-21 15:05 [PATCHSET 0/3] blk-wbt fixes Jens Axboe
2018-08-21 15:05 ` [PATCH 1/3] blk-wbt: move disable check into get_limit() Jens Axboe
@ 2018-08-21 15:05 ` Jens Axboe
2018-08-21 15:05 ` [PATCH 3/3] blk-wbt: fix has-sleeper queueing check Jens Axboe
2 siblings, 0 replies; 4+ messages in thread
From: Jens Axboe @ 2018-08-21 15:05 UTC (permalink / raw)
To: linux-block; +Cc: Jens Axboe
We need the memory barrier before checking the list head,
use the appropriate helper for this. The matching queue
side memory barrier is provided by set_current_state().
Signed-off-by: Jens Axboe <axboe@kernel.dk>
---
block/blk-wbt.c | 8 ++++----
1 file changed, 4 insertions(+), 4 deletions(-)
diff --git a/block/blk-wbt.c b/block/blk-wbt.c
index 1dd7edce7a99..853e22492f4e 100644
--- a/block/blk-wbt.c
+++ b/block/blk-wbt.c
@@ -118,7 +118,7 @@ static void rwb_wake_all(struct rq_wb *rwb)
for (i = 0; i < WBT_NUM_RWQ; i++) {
struct rq_wait *rqw = &rwb->rq_wait[i];
- if (waitqueue_active(&rqw->wait))
+ if (wq_has_sleeper(&rqw->wait))
wake_up_all(&rqw->wait);
}
}
@@ -162,7 +162,7 @@ static void __wbt_done(struct rq_qos *rqos, enum wbt_flags wb_acct)
if (inflight && inflight >= limit)
return;
- if (waitqueue_active(&rqw->wait)) {
+ if (wq_has_sleeper(&rqw->wait)) {
int diff = limit - inflight;
if (!inflight || diff >= rwb->wb_background / 2)
@@ -493,8 +493,8 @@ static void __wbt_wait(struct rq_wb *rwb, enum wbt_flags wb_acct,
struct rq_wait *rqw = get_rq_wait(rwb, wb_acct);
DECLARE_WAITQUEUE(wait, current);
- if (!waitqueue_active(&rqw->wait)
- && rq_wait_inc_below(rqw, get_limit(rwb, rw)))
+ if (!wq_has_sleeper(&rqw->wait) &&
+ rq_wait_inc_below(rqw, get_limit(rwb, rw)))
return;
add_wait_queue_exclusive(&rqw->wait, &wait);
--
2.17.1
^ permalink raw reply related [flat|nested] 4+ messages in thread
* [PATCH 3/3] blk-wbt: fix has-sleeper queueing check
2018-08-21 15:05 [PATCHSET 0/3] blk-wbt fixes Jens Axboe
2018-08-21 15:05 ` [PATCH 1/3] blk-wbt: move disable check into get_limit() Jens Axboe
2018-08-21 15:05 ` [PATCH 2/3] blk-wbt: use wq_has_sleeper() for wq active check Jens Axboe
@ 2018-08-21 15:05 ` Jens Axboe
2 siblings, 0 replies; 4+ messages in thread
From: Jens Axboe @ 2018-08-21 15:05 UTC (permalink / raw)
To: linux-block; +Cc: Jens Axboe
We need to do this inside the loop as well, or we can allow new
IO to supersede previous IO.
Signed-off-by: Jens Axboe <axboe@kernel.dk>
---
block/blk-wbt.c | 8 +++++---
1 file changed, 5 insertions(+), 3 deletions(-)
diff --git a/block/blk-wbt.c b/block/blk-wbt.c
index 853e22492f4e..c9358f1981fb 100644
--- a/block/blk-wbt.c
+++ b/block/blk-wbt.c
@@ -492,16 +492,17 @@ static void __wbt_wait(struct rq_wb *rwb, enum wbt_flags wb_acct,
{
struct rq_wait *rqw = get_rq_wait(rwb, wb_acct);
DECLARE_WAITQUEUE(wait, current);
+ bool has_sleeper;
- if (!wq_has_sleeper(&rqw->wait) &&
- rq_wait_inc_below(rqw, get_limit(rwb, rw)))
+ has_sleeper = wq_has_sleeper(&rqw->wait);
+ if (!has_sleeper && rq_wait_inc_below(rqw, get_limit(rwb, rw)))
return;
add_wait_queue_exclusive(&rqw->wait, &wait);
do {
set_current_state(TASK_UNINTERRUPTIBLE);
- if (rq_wait_inc_below(rqw, get_limit(rwb, rw)))
+ if (!has_sleeper && rq_wait_inc_below(rqw, get_limit(rwb, rw)))
break;
if (lock) {
@@ -510,6 +511,7 @@ static void __wbt_wait(struct rq_wb *rwb, enum wbt_flags wb_acct,
spin_lock_irq(lock);
} else
io_schedule();
+ has_sleeper = false;
} while (1);
__set_current_state(TASK_RUNNING);
--
2.17.1
^ permalink raw reply related [flat|nested] 4+ messages in thread
end of thread, other threads:[~2018-08-21 15:05 UTC | newest]
Thread overview: 4+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2018-08-21 15:05 [PATCHSET 0/3] blk-wbt fixes Jens Axboe
2018-08-21 15:05 ` [PATCH 1/3] blk-wbt: move disable check into get_limit() Jens Axboe
2018-08-21 15:05 ` [PATCH 2/3] blk-wbt: use wq_has_sleeper() for wq active check Jens Axboe
2018-08-21 15:05 ` [PATCH 3/3] blk-wbt: fix has-sleeper queueing check Jens Axboe
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).