Commit 50824d6c authored by Dan Williams's avatar Dan Williams Committed by James Bottomley

[SCSI] libsas: async ata-eh

Once sas_ata_hard_reset() starts honoring the 'deadline' parameter a
pathological configuration could take 25 seconds per ata device
(serialized) to recover.  Run per-port recoveries in parallel.
Signed-off-by: default avatarDan Williams <dan.j.williams@intel.com>
Signed-off-by: default avatarJames Bottomley <JBottomley@Parallels.com>
parent 89d3cf6a
...@@ -23,6 +23,7 @@ ...@@ -23,6 +23,7 @@
#include <linux/scatterlist.h> #include <linux/scatterlist.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/async.h>
#include <scsi/sas_ata.h> #include <scsi/sas_ata.h>
#include "sas_internal.h" #include "sas_internal.h"
...@@ -605,10 +606,21 @@ int sas_discover_sata(struct domain_device *dev) ...@@ -605,10 +606,21 @@ int sas_discover_sata(struct domain_device *dev)
return 0; return 0;
} }
static void async_sas_ata_eh(void *data, async_cookie_t cookie)
{
struct domain_device *dev = data;
struct ata_port *ap = dev->sata_dev.ap;
struct sas_ha_struct *ha = dev->port->ha;
ata_port_printk(ap, KERN_DEBUG, "sas eh calling libata port error handler");
ata_scsi_port_error_handler(ha->core.shost, ap);
}
void sas_ata_strategy_handler(struct Scsi_Host *shost) void sas_ata_strategy_handler(struct Scsi_Host *shost)
{ {
struct scsi_device *sdev; struct scsi_device *sdev;
struct sas_ha_struct *sas_ha = SHOST_TO_SAS_HA(shost); struct sas_ha_struct *sas_ha = SHOST_TO_SAS_HA(shost);
LIST_HEAD(async);
/* it's ok to defer revalidation events during ata eh, these /* it's ok to defer revalidation events during ata eh, these
* disks are in one of three states: * disks are in one of three states:
...@@ -622,14 +634,13 @@ void sas_ata_strategy_handler(struct Scsi_Host *shost) ...@@ -622,14 +634,13 @@ void sas_ata_strategy_handler(struct Scsi_Host *shost)
shost_for_each_device(sdev, shost) { shost_for_each_device(sdev, shost) {
struct domain_device *ddev = sdev_to_domain_dev(sdev); struct domain_device *ddev = sdev_to_domain_dev(sdev);
struct ata_port *ap = ddev->sata_dev.ap;
if (!dev_is_sata(ddev)) if (!dev_is_sata(ddev))
continue; continue;
ata_port_printk(ap, KERN_DEBUG, "sas eh calling libata port error handler"); async_schedule_domain(async_sas_ata_eh, ddev, &async);
ata_scsi_port_error_handler(shost, ap);
} }
async_synchronize_full_domain(&async);
sas_enable_revalidation(sas_ha); sas_enable_revalidation(sas_ha);
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment