aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/scsi
diff options
context:
space:
mode:
authorDan Williams <dan.j.williams@intel.com>2012-01-31 00:40:45 -0500
committerJames Bottomley <JBottomley@Parallels.com>2012-02-29 16:42:51 -0500
commit26a2e68f816ebd736a0484ca293457b280af4ef1 (patch)
tree5fbae208c3bcda21dd1f9f918e811e07c2ce80e8 /drivers/scsi
parent77c309f3cdf9e217032dfe330f5881d352bb0436 (diff)
[SCSI] libsas: don't recover end devices attached to disabled phys
If userspace has decided to disable a phy the kernel should honor that and not inadvertantly re-enable the phy via error recovery. This is more straightforward in the sata case where link recovery (via libata-eh) is separate from sas_task cancelling in libsas-eh. Teach libsas to accept -ENODEV as a successful response from I_T_nexus_reset ('successful' in terms of not escalating further). This is a more comprehensive fix then "libsas: don't recover 'gone' devices in sas_ata_hard_reset()", as it is no longer sata-specific. aic94xx does check the return value from sas_phy_reset() so if the phy is disabled we proceed with clearing the I_T_nexus. Signed-off-by: Dan Williams <dan.j.williams@intel.com> Signed-off-by: James Bottomley <JBottomley@Parallels.com>
Diffstat (limited to 'drivers/scsi')
-rw-r--r--drivers/scsi/aic94xx/aic94xx_tmf.c2
-rw-r--r--drivers/scsi/libsas/sas_ata.c5
-rw-r--r--drivers/scsi/libsas/sas_init.c3
-rw-r--r--drivers/scsi/libsas/sas_scsi_host.c3
4 files changed, 8 insertions, 5 deletions
diff --git a/drivers/scsi/aic94xx/aic94xx_tmf.c b/drivers/scsi/aic94xx/aic94xx_tmf.c
index 50b914ffab94..cf9040933da6 100644
--- a/drivers/scsi/aic94xx/aic94xx_tmf.c
+++ b/drivers/scsi/aic94xx/aic94xx_tmf.c
@@ -192,7 +192,7 @@ int asd_I_T_nexus_reset(struct domain_device *dev)
192 ASD_DPRINTK("sending %s reset to %s\n", 192 ASD_DPRINTK("sending %s reset to %s\n",
193 reset_type ? "hard" : "soft", dev_name(&phy->dev)); 193 reset_type ? "hard" : "soft", dev_name(&phy->dev));
194 res = sas_phy_reset(phy, reset_type); 194 res = sas_phy_reset(phy, reset_type);
195 if (res == TMF_RESP_FUNC_COMPLETE) { 195 if (res == TMF_RESP_FUNC_COMPLETE || res == -ENODEV) {
196 /* wait for the maximum settle time */ 196 /* wait for the maximum settle time */
197 msleep(500); 197 msleep(500);
198 /* clear all outstanding commands (keep nexus suspended) */ 198 /* clear all outstanding commands (keep nexus suspended) */
diff --git a/drivers/scsi/libsas/sas_ata.c b/drivers/scsi/libsas/sas_ata.c
index 08d2103a45b7..bc0cecc6ad62 100644
--- a/drivers/scsi/libsas/sas_ata.c
+++ b/drivers/scsi/libsas/sas_ata.c
@@ -407,10 +407,9 @@ static int sas_ata_hard_reset(struct ata_link *link, unsigned int *class,
407 struct domain_device *dev = ap->private_data; 407 struct domain_device *dev = ap->private_data;
408 struct sas_internal *i = dev_to_sas_internal(dev); 408 struct sas_internal *i = dev_to_sas_internal(dev);
409 409
410 if (test_bit(SAS_DEV_GONE, &dev->state))
411 return -ENODEV;
412
413 res = i->dft->lldd_I_T_nexus_reset(dev); 410 res = i->dft->lldd_I_T_nexus_reset(dev);
411 if (res == -ENODEV)
412 return res;
414 413
415 if (res != TMF_RESP_FUNC_COMPLETE) 414 if (res != TMF_RESP_FUNC_COMPLETE)
416 sas_ata_printk(KERN_DEBUG, dev, "Unable to reset ata device?\n"); 415 sas_ata_printk(KERN_DEBUG, dev, "Unable to reset ata device?\n");
diff --git a/drivers/scsi/libsas/sas_init.c b/drivers/scsi/libsas/sas_init.c
index 09c14ca3fbd5..120bff64be30 100644
--- a/drivers/scsi/libsas/sas_init.c
+++ b/drivers/scsi/libsas/sas_init.c
@@ -298,6 +298,9 @@ int sas_phy_reset(struct sas_phy *phy, int hard_reset)
298 int ret; 298 int ret;
299 enum phy_func reset_type; 299 enum phy_func reset_type;
300 300
301 if (!phy->enabled)
302 return -ENODEV;
303
301 if (hard_reset) 304 if (hard_reset)
302 reset_type = PHY_FUNC_HARD_RESET; 305 reset_type = PHY_FUNC_HARD_RESET;
303 else 306 else
diff --git a/drivers/scsi/libsas/sas_scsi_host.c b/drivers/scsi/libsas/sas_scsi_host.c
index fd3291337c1b..f0b9b7bf1882 100644
--- a/drivers/scsi/libsas/sas_scsi_host.c
+++ b/drivers/scsi/libsas/sas_scsi_host.c
@@ -607,7 +607,8 @@ static void sas_eh_handle_sas_errors(struct Scsi_Host *shost, struct list_head *
607 SAS_DPRINTK("task 0x%p is not at LU: I_T recover\n", 607 SAS_DPRINTK("task 0x%p is not at LU: I_T recover\n",
608 task); 608 task);
609 tmf_resp = sas_recover_I_T(task->dev); 609 tmf_resp = sas_recover_I_T(task->dev);
610 if (tmf_resp == TMF_RESP_FUNC_COMPLETE) { 610 if (tmf_resp == TMF_RESP_FUNC_COMPLETE ||
611 tmf_resp == -ENODEV) {
611 struct domain_device *dev = task->dev; 612 struct domain_device *dev = task->dev;
612 SAS_DPRINTK("I_T %016llx recovered\n", 613 SAS_DPRINTK("I_T %016llx recovered\n",
613 SAS_ADDR(task->dev->sas_addr)); 614 SAS_ADDR(task->dev->sas_addr));