diff options
author | Jianchao Wang <jianchao.w.wang@oracle.com> | 2018-06-20 01:42:22 -0400 |
---|---|---|
committer | Christoph Hellwig <hch@lst.de> | 2018-06-21 10:59:42 -0400 |
commit | 9f9cafc14016f23f982d3ce18f9057923bd3037a (patch) | |
tree | 15b08d09c35beacdd80cd36c25f8d0151540bde9 | |
parent | 02d62a8bc48e92171c46540722e2d52ce77d87af (diff) |
nvme-pci: move nvme_kill_queues to nvme_remove_dead_ctrl
There is race between nvme_remove and nvme_reset_work that can
lead to io hang.
nvme_remove nvme_reset_work
-> nvme_remove_dead_ctrl
-> nvme_dev_disable
-> quiesce request_queue
-> queue remove_work
-> cancel_work_sync reset_work
-> nvme_remove_namespaces
-> splice ctrl->namespaces
nvme_remove_dead_ctrl_work
-> nvme_kill_queues
-> nvme_ns_remove do nothing
-> blk_cleanup_queue
-> blk_freeze_queue
Finally, the request_queue is quiesced state when wait freeze,
we will get io hang here. To fix it, move the nvme_kill_queues
from nvme_remove_dead_ctrl_work to nvme_remove_dead_ctrl.
Suggested-by: Keith Busch <keith.busch@linux.intel.com>
Signed-off-by: Jianchao Wang <jianchao.w.wang@oracle.com>
Reviewed-by: Keith Busch <keith.busch@intel.com>
Signed-off-by: Christoph Hellwig <hch@lst.de>
-rw-r--r-- | drivers/nvme/host/pci.c | 2 |
1 files changed, 1 insertions, 1 deletions
diff --git a/drivers/nvme/host/pci.c b/drivers/nvme/host/pci.c index fc33804662e7..73a97fcea364 100644 --- a/drivers/nvme/host/pci.c +++ b/drivers/nvme/host/pci.c | |||
@@ -2289,6 +2289,7 @@ static void nvme_remove_dead_ctrl(struct nvme_dev *dev, int status) | |||
2289 | 2289 | ||
2290 | nvme_get_ctrl(&dev->ctrl); | 2290 | nvme_get_ctrl(&dev->ctrl); |
2291 | nvme_dev_disable(dev, false); | 2291 | nvme_dev_disable(dev, false); |
2292 | nvme_kill_queues(&dev->ctrl); | ||
2292 | if (!queue_work(nvme_wq, &dev->remove_work)) | 2293 | if (!queue_work(nvme_wq, &dev->remove_work)) |
2293 | nvme_put_ctrl(&dev->ctrl); | 2294 | nvme_put_ctrl(&dev->ctrl); |
2294 | } | 2295 | } |
@@ -2405,7 +2406,6 @@ static void nvme_remove_dead_ctrl_work(struct work_struct *work) | |||
2405 | struct nvme_dev *dev = container_of(work, struct nvme_dev, remove_work); | 2406 | struct nvme_dev *dev = container_of(work, struct nvme_dev, remove_work); |
2406 | struct pci_dev *pdev = to_pci_dev(dev->dev); | 2407 | struct pci_dev *pdev = to_pci_dev(dev->dev); |
2407 | 2408 | ||
2408 | nvme_kill_queues(&dev->ctrl); | ||
2409 | if (pci_get_drvdata(pdev)) | 2409 | if (pci_get_drvdata(pdev)) |
2410 | device_release_driver(&pdev->dev); | 2410 | device_release_driver(&pdev->dev); |
2411 | nvme_put_ctrl(&dev->ctrl); | 2411 | nvme_put_ctrl(&dev->ctrl); |