diff --git a/block/blk-exec.c b/block/blk-exec.c index fb2cbd55162..8b6dc5bd4dd 100644 --- a/block/blk-exec.c +++ b/block/blk-exec.c @@ -43,6 +43,9 @@ static void blk_end_sync_rq(struct request *rq, int error) * Description: * Insert a fully prepared request at the back of the I/O scheduler queue * for execution. Don't wait for completion. + * + * Note: + * This function will invoke @done directly if the queue is dead. */ void blk_execute_rq_nowait(struct request_queue *q, struct gendisk *bd_disk, struct request *rq, int at_head, @@ -51,18 +54,20 @@ void blk_execute_rq_nowait(struct request_queue *q, struct gendisk *bd_disk, int where = at_head ? ELEVATOR_INSERT_FRONT : ELEVATOR_INSERT_BACK; WARN_ON(irqs_disabled()); - spin_lock_irq(q->queue_lock); - - if (unlikely(blk_queue_dead(q))) { - spin_unlock_irq(q->queue_lock); - rq->errors = -ENXIO; - if (rq->end_io) - rq->end_io(rq, rq->errors); - return; - } rq->rq_disk = bd_disk; rq->end_io = done; + + spin_lock_irq(q->queue_lock); + + if (unlikely(blk_queue_dead(q))) { + rq->errors = -ENXIO; + if (rq->end_io) + rq->end_io(rq, rq->errors); + spin_unlock_irq(q->queue_lock); + return; + } + __elv_add_request(q, rq, where); __blk_run_queue(q); /* the queue is stopped so it won't be run */