Commit cddd5d17 authored by Tejun Heo's avatar Tejun Heo Committed by Jens Axboe

blk-mq: blk_mq_freeze_queue() should allow nesting

While converting to percpu_ref for freezing, add703fd ("blk-mq:
use percpu_ref for mq usage count") incorrectly made
blk_mq_freeze_queue() misbehave when freezing is nested due to
percpu_ref_kill() being invoked on an already killed ref.

Fix it by making blk_mq_freeze_queue() kill and kick the queue only
for the outermost freeze attempt.  All the nested ones can simply wait
for the ref to reach zero.

While at it, remove unnecessary @wake initialization from
blk_mq_unfreeze_queue().
Signed-off-by: default avatarTejun Heo <tj@kernel.org>
Reported-by: default avatarMing Lei <ming.lei@canonical.com>
Signed-off-by: default avatarJens Axboe <axboe@fb.com>
parent a68aafa5
...@@ -112,18 +112,22 @@ static void blk_mq_usage_counter_release(struct percpu_ref *ref) ...@@ -112,18 +112,22 @@ static void blk_mq_usage_counter_release(struct percpu_ref *ref)
*/ */
void blk_mq_freeze_queue(struct request_queue *q) void blk_mq_freeze_queue(struct request_queue *q)
{ {
bool freeze;
spin_lock_irq(q->queue_lock); spin_lock_irq(q->queue_lock);
q->mq_freeze_depth++; freeze = !q->mq_freeze_depth++;
spin_unlock_irq(q->queue_lock); spin_unlock_irq(q->queue_lock);
percpu_ref_kill(&q->mq_usage_counter); if (freeze) {
blk_mq_run_queues(q, false); percpu_ref_kill(&q->mq_usage_counter);
blk_mq_run_queues(q, false);
}
wait_event(q->mq_freeze_wq, percpu_ref_is_zero(&q->mq_usage_counter)); wait_event(q->mq_freeze_wq, percpu_ref_is_zero(&q->mq_usage_counter));
} }
static void blk_mq_unfreeze_queue(struct request_queue *q) static void blk_mq_unfreeze_queue(struct request_queue *q)
{ {
bool wake = false; bool wake;
spin_lock_irq(q->queue_lock); spin_lock_irq(q->queue_lock);
wake = !--q->mq_freeze_depth; wake = !--q->mq_freeze_depth;
......
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