]> www.infradead.org Git - users/jedix/linux-maple.git/commitdiff
xen-blkfront: fix mq start/stop race
authorJunxiao Bi <junxiao.bi@oracle.com>
Mon, 19 Jun 2017 07:56:36 +0000 (15:56 +0800)
committerChuck Anderson <chuck.anderson@oracle.com>
Sun, 9 Jul 2017 01:21:01 +0000 (18:21 -0700)
Orabug: 26397415

When ring buf full, hw queue will be stopped. While blkif interrupt consume
request and make free space in ring buf, hw queue will be started again.
But since start queue is protected by spin lock while stop not, that will
cause a race.

interrupt:                                      process:
blkif_interrupt()                               blkif_queue_rq()
 kick_pending_request_queues_locked()
  blk_mq_start_stopped_hw_queues()
   clear_bit(BLK_MQ_S_STOPPED, &hctx->state)
                                                 blk_mq_stop_hw_queue(hctx)
   blk_mq_run_hw_queue(hctx, async)

If ring buf is made empty in this case, interrupt will never come, then the
hw queue will be stopped forever, all processes waiting for the pending io
in the queue will hung.

Signed-off-by: Junxiao Bi <junxiao.bi@oracle.com>
Reviewed-by: Ankur Arora <ankur.a.arora@oracle.com>
Reviewed-by: Boris Ostrovsky <boris.ostrovsky@oracle.com>
Acked-by: Roger Pau Monné <roger.pau@citrix.com>
drivers/block/xen-blkfront.c

index 2edf895d3d0f8203826b4d8a15b2558694e91152..2320f230c5302df3b6134d508bbd6629c36a80b5 100644 (file)
@@ -912,8 +912,8 @@ out_err:
        return BLK_MQ_RQ_QUEUE_ERROR;
 
 out_busy:
-       spin_unlock_irqrestore(&rinfo->ring_lock, flags);
        blk_mq_stop_hw_queue(hctx);
+       spin_unlock_irqrestore(&rinfo->ring_lock, flags);
        return BLK_MQ_RQ_QUEUE_BUSY;
 }