diff options
author | Dan Williams <dan.j.williams@intel.com> | 2012-01-31 00:40:45 -0500 |
---|---|---|
committer | James Bottomley <JBottomley@Parallels.com> | 2012-02-29 16:42:51 -0500 |
commit | 26a2e68f816ebd736a0484ca293457b280af4ef1 (patch) | |
tree | 5fbae208c3bcda21dd1f9f918e811e07c2ce80e8 /drivers/scsi | |
parent | 77c309f3cdf9e217032dfe330f5881d352bb0436 (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.c | 2 | ||||
-rw-r--r-- | drivers/scsi/libsas/sas_ata.c | 5 | ||||
-rw-r--r-- | drivers/scsi/libsas/sas_init.c | 3 | ||||
-rw-r--r-- | drivers/scsi/libsas/sas_scsi_host.c | 3 |
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)); |