diff options
-rw-r--r-- | drivers/block/nvme-core.c | 21 | ||||
-rw-r--r-- | include/linux/nvme.h | 1 |
2 files changed, 20 insertions, 2 deletions
diff --git a/drivers/block/nvme-core.c b/drivers/block/nvme-core.c index 000bca43c23b..2f5b9f5f5a21 100644 --- a/drivers/block/nvme-core.c +++ b/drivers/block/nvme-core.c | |||
@@ -60,6 +60,8 @@ static LIST_HEAD(dev_list); | |||
60 | static struct task_struct *nvme_thread; | 60 | static struct task_struct *nvme_thread; |
61 | static struct workqueue_struct *nvme_workq; | 61 | static struct workqueue_struct *nvme_workq; |
62 | 62 | ||
63 | static void nvme_reset_failed_dev(struct work_struct *ws); | ||
64 | |||
63 | /* | 65 | /* |
64 | * An NVM Express queue. Each device has at least two (one for admin | 66 | * An NVM Express queue. Each device has at least two (one for admin |
65 | * commands and one for I/O commands). | 67 | * commands and one for I/O commands). |
@@ -1612,13 +1614,25 @@ static void nvme_resubmit_bios(struct nvme_queue *nvmeq) | |||
1612 | 1614 | ||
1613 | static int nvme_kthread(void *data) | 1615 | static int nvme_kthread(void *data) |
1614 | { | 1616 | { |
1615 | struct nvme_dev *dev; | 1617 | struct nvme_dev *dev, *next; |
1616 | 1618 | ||
1617 | while (!kthread_should_stop()) { | 1619 | while (!kthread_should_stop()) { |
1618 | set_current_state(TASK_INTERRUPTIBLE); | 1620 | set_current_state(TASK_INTERRUPTIBLE); |
1619 | spin_lock(&dev_list_lock); | 1621 | spin_lock(&dev_list_lock); |
1620 | list_for_each_entry(dev, &dev_list, node) { | 1622 | list_for_each_entry_safe(dev, next, &dev_list, node) { |
1621 | int i; | 1623 | int i; |
1624 | if (readl(&dev->bar->csts) & NVME_CSTS_CFS && | ||
1625 | dev->initialized) { | ||
1626 | if (work_busy(&dev->reset_work)) | ||
1627 | continue; | ||
1628 | list_del_init(&dev->node); | ||
1629 | dev_warn(&dev->pci_dev->dev, | ||
1630 | "Failed status, reset controller\n"); | ||
1631 | INIT_WORK(&dev->reset_work, | ||
1632 | nvme_reset_failed_dev); | ||
1633 | queue_work(nvme_workq, &dev->reset_work); | ||
1634 | continue; | ||
1635 | } | ||
1622 | for (i = 0; i < dev->queue_count; i++) { | 1636 | for (i = 0; i < dev->queue_count; i++) { |
1623 | struct nvme_queue *nvmeq = dev->queues[i]; | 1637 | struct nvme_queue *nvmeq = dev->queues[i]; |
1624 | if (!nvmeq) | 1638 | if (!nvmeq) |
@@ -2006,6 +2020,7 @@ static void nvme_dev_shutdown(struct nvme_dev *dev) | |||
2006 | { | 2020 | { |
2007 | int i; | 2021 | int i; |
2008 | 2022 | ||
2023 | dev->initialized = 0; | ||
2009 | for (i = dev->queue_count - 1; i >= 0; i--) | 2024 | for (i = dev->queue_count - 1; i >= 0; i--) |
2010 | nvme_disable_queue(dev, i); | 2025 | nvme_disable_queue(dev, i); |
2011 | 2026 | ||
@@ -2196,6 +2211,7 @@ static int nvme_dev_resume(struct nvme_dev *dev) | |||
2196 | queue_work(nvme_workq, &dev->reset_work); | 2211 | queue_work(nvme_workq, &dev->reset_work); |
2197 | spin_unlock(&dev_list_lock); | 2212 | spin_unlock(&dev_list_lock); |
2198 | } | 2213 | } |
2214 | dev->initialized = 1; | ||
2199 | return 0; | 2215 | return 0; |
2200 | } | 2216 | } |
2201 | 2217 | ||
@@ -2269,6 +2285,7 @@ static int nvme_probe(struct pci_dev *pdev, const struct pci_device_id *id) | |||
2269 | if (result) | 2285 | if (result) |
2270 | goto remove; | 2286 | goto remove; |
2271 | 2287 | ||
2288 | dev->initialized = 1; | ||
2272 | kref_init(&dev->kref); | 2289 | kref_init(&dev->kref); |
2273 | return 0; | 2290 | return 0; |
2274 | 2291 | ||
diff --git a/include/linux/nvme.h b/include/linux/nvme.h index eed81cc56d7e..117d877e8be5 100644 --- a/include/linux/nvme.h +++ b/include/linux/nvme.h | |||
@@ -95,6 +95,7 @@ struct nvme_dev { | |||
95 | u32 max_hw_sectors; | 95 | u32 max_hw_sectors; |
96 | u32 stripe_size; | 96 | u32 stripe_size; |
97 | u16 oncs; | 97 | u16 oncs; |
98 | u8 initialized; | ||
98 | }; | 99 | }; |
99 | 100 | ||
100 | /* | 101 | /* |