|
@@ -265,6 +265,22 @@ static void sas_eh_finish_cmd(struct scsi_cmnd *cmd)
|
|
|
scsi_eh_finish_cmd(cmd, &sas_ha->eh_done_q);
|
|
|
}
|
|
|
|
|
|
+static void sas_eh_defer_cmd(struct scsi_cmnd *cmd)
|
|
|
+{
|
|
|
+ struct sas_task *task = TO_SAS_TASK(cmd);
|
|
|
+ struct domain_device *dev = task->dev;
|
|
|
+ struct sas_ha_struct *ha = dev->port->ha;
|
|
|
+
|
|
|
+ if (!dev_is_sata(dev)) {
|
|
|
+ sas_eh_finish_cmd(cmd);
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* report the timeout to libata */
|
|
|
+ sas_end_task(cmd, task);
|
|
|
+ list_move_tail(&cmd->eh_entry, &ha->eh_ata_q);
|
|
|
+}
|
|
|
+
|
|
|
static void sas_scsi_clear_queue_lu(struct list_head *error_q, struct scsi_cmnd *my_cmd)
|
|
|
{
|
|
|
struct scsi_cmnd *cmd, *n;
|
|
@@ -562,12 +578,12 @@ Again:
|
|
|
case TASK_IS_DONE:
|
|
|
SAS_DPRINTK("%s: task 0x%p is done\n", __func__,
|
|
|
task);
|
|
|
- sas_eh_finish_cmd(cmd);
|
|
|
+ sas_eh_defer_cmd(cmd);
|
|
|
continue;
|
|
|
case TASK_IS_ABORTED:
|
|
|
SAS_DPRINTK("%s: task 0x%p is aborted\n",
|
|
|
__func__, task);
|
|
|
- sas_eh_finish_cmd(cmd);
|
|
|
+ sas_eh_defer_cmd(cmd);
|
|
|
continue;
|
|
|
case TASK_IS_AT_LU:
|
|
|
SAS_DPRINTK("task 0x%p is at LU: lu recover\n", task);
|
|
@@ -635,12 +651,14 @@ Again:
|
|
|
goto clear_q;
|
|
|
}
|
|
|
}
|
|
|
+ list_splice_tail_init(&ha->eh_ata_q, work_q);
|
|
|
return list_empty(work_q);
|
|
|
clear_q:
|
|
|
SAS_DPRINTK("--- Exit %s -- clear_q\n", __func__);
|
|
|
list_for_each_entry_safe(cmd, n, work_q, eh_entry)
|
|
|
sas_eh_finish_cmd(cmd);
|
|
|
|
|
|
+ list_splice_tail_init(&ha->eh_ata_q, work_q);
|
|
|
return list_empty(work_q);
|
|
|
}
|
|
|
|