aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/scsi/libsas
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/scsi/libsas')
-rw-r--r--drivers/scsi/libsas/sas_init.c2
-rw-r--r--drivers/scsi/libsas/sas_scsi_host.c29
2 files changed, 27 insertions, 4 deletions
diff --git a/drivers/scsi/libsas/sas_init.c b/drivers/scsi/libsas/sas_init.c
index c836a237fb79..a2b479a65908 100644
--- a/drivers/scsi/libsas/sas_init.c
+++ b/drivers/scsi/libsas/sas_init.c
@@ -112,6 +112,8 @@ int sas_register_ha(struct sas_ha_struct *sas_ha)
112 } 112 }
113 } 113 }
114 114
115 INIT_LIST_HEAD(&sas_ha->eh_done_q);
116
115 return 0; 117 return 0;
116 118
117Undo_ports: 119Undo_ports:
diff --git a/drivers/scsi/libsas/sas_scsi_host.c b/drivers/scsi/libsas/sas_scsi_host.c
index e46e79355b77..6a97b07849b4 100644
--- a/drivers/scsi/libsas/sas_scsi_host.c
+++ b/drivers/scsi/libsas/sas_scsi_host.c
@@ -29,6 +29,7 @@
29#include <scsi/scsi_device.h> 29#include <scsi/scsi_device.h>
30#include <scsi/scsi_tcq.h> 30#include <scsi/scsi_tcq.h>
31#include <scsi/scsi.h> 31#include <scsi/scsi.h>
32#include <scsi/scsi_eh.h>
32#include <scsi/scsi_transport.h> 33#include <scsi/scsi_transport.h>
33#include <scsi/scsi_transport_sas.h> 34#include <scsi/scsi_transport_sas.h>
34#include "../scsi_sas_internal.h" 35#include "../scsi_sas_internal.h"
@@ -46,6 +47,7 @@ static void sas_scsi_task_done(struct sas_task *task)
46{ 47{
47 struct task_status_struct *ts = &task->task_status; 48 struct task_status_struct *ts = &task->task_status;
48 struct scsi_cmnd *sc = task->uldd_task; 49 struct scsi_cmnd *sc = task->uldd_task;
50 struct sas_ha_struct *sas_ha = SHOST_TO_SAS_HA(sc->device->host);
49 unsigned ts_flags = task->task_state_flags; 51 unsigned ts_flags = task->task_state_flags;
50 int hs = 0, stat = 0; 52 int hs = 0, stat = 0;
51 53
@@ -116,7 +118,7 @@ static void sas_scsi_task_done(struct sas_task *task)
116 sas_free_task(task); 118 sas_free_task(task);
117 /* This is very ugly but this is how SCSI Core works. */ 119 /* This is very ugly but this is how SCSI Core works. */
118 if (ts_flags & SAS_TASK_STATE_ABORTED) 120 if (ts_flags & SAS_TASK_STATE_ABORTED)
119 scsi_finish_command(sc); 121 scsi_eh_finish_cmd(sc, &sas_ha->eh_done_q);
120 else 122 else
121 sc->scsi_done(sc); 123 sc->scsi_done(sc);
122} 124}
@@ -307,6 +309,15 @@ static enum task_disposition sas_scsi_find_task(struct sas_task *task)
307 spin_unlock_irqrestore(&core->task_queue_lock, flags); 309 spin_unlock_irqrestore(&core->task_queue_lock, flags);
308 } 310 }
309 311
312 spin_lock_irqsave(&task->task_state_lock, flags);
313 if (task->task_state_flags & SAS_TASK_INITIATOR_ABORTED) {
314 spin_unlock_irqrestore(&task->task_state_lock, flags);
315 SAS_DPRINTK("%s: task 0x%p already aborted\n",
316 __FUNCTION__, task);
317 return TASK_IS_ABORTED;
318 }
319 spin_unlock_irqrestore(&task->task_state_lock, flags);
320
310 for (i = 0; i < 5; i++) { 321 for (i = 0; i < 5; i++) {
311 SAS_DPRINTK("%s: aborting task 0x%p\n", __FUNCTION__, task); 322 SAS_DPRINTK("%s: aborting task 0x%p\n", __FUNCTION__, task);
312 res = si->dft->lldd_abort_task(task); 323 res = si->dft->lldd_abort_task(task);
@@ -409,13 +420,16 @@ Again:
409 SAS_DPRINTK("going over list...\n"); 420 SAS_DPRINTK("going over list...\n");
410 list_for_each_entry_safe(cmd, n, &error_q, eh_entry) { 421 list_for_each_entry_safe(cmd, n, &error_q, eh_entry) {
411 struct sas_task *task = TO_SAS_TASK(cmd); 422 struct sas_task *task = TO_SAS_TASK(cmd);
423 list_del_init(&cmd->eh_entry);
412 424
425 if (!task) {
426 SAS_DPRINTK("%s: taskless cmd?!\n", __FUNCTION__);
427 continue;
428 }
413 SAS_DPRINTK("trying to find task 0x%p\n", task); 429 SAS_DPRINTK("trying to find task 0x%p\n", task);
414 list_del_init(&cmd->eh_entry);
415 res = sas_scsi_find_task(task); 430 res = sas_scsi_find_task(task);
416 431
417 cmd->eh_eflags = 0; 432 cmd->eh_eflags = 0;
418 shost->host_failed--;
419 433
420 switch (res) { 434 switch (res) {
421 case TASK_IS_DONE: 435 case TASK_IS_DONE:
@@ -491,6 +505,7 @@ Again:
491 } 505 }
492 } 506 }
493out: 507out:
508 scsi_eh_flush_done_q(&ha->eh_done_q);
494 SAS_DPRINTK("--- Exit %s\n", __FUNCTION__); 509 SAS_DPRINTK("--- Exit %s\n", __FUNCTION__);
495 return; 510 return;
496clear_q: 511clear_q:
@@ -508,12 +523,18 @@ enum scsi_eh_timer_return sas_scsi_timed_out(struct scsi_cmnd *cmd)
508 unsigned long flags; 523 unsigned long flags;
509 524
510 if (!task) { 525 if (!task) {
511 SAS_DPRINTK("command 0x%p, task 0x%p, timed out: EH_HANDLED\n", 526 SAS_DPRINTK("command 0x%p, task 0x%p, gone: EH_HANDLED\n",
512 cmd, task); 527 cmd, task);
513 return EH_HANDLED; 528 return EH_HANDLED;
514 } 529 }
515 530
516 spin_lock_irqsave(&task->task_state_lock, flags); 531 spin_lock_irqsave(&task->task_state_lock, flags);
532 if (task->task_state_flags & SAS_TASK_INITIATOR_ABORTED) {
533 spin_unlock_irqrestore(&task->task_state_lock, flags);
534 SAS_DPRINTK("command 0x%p, task 0x%p, aborted by initiator: "
535 "EH_NOT_HANDLED\n", cmd, task);
536 return EH_NOT_HANDLED;
537 }
517 if (task->task_state_flags & SAS_TASK_STATE_DONE) { 538 if (task->task_state_flags & SAS_TASK_STATE_DONE) {
518 spin_unlock_irqrestore(&task->task_state_lock, flags); 539 spin_unlock_irqrestore(&task->task_state_lock, flags);
519 SAS_DPRINTK("command 0x%p, task 0x%p, timed out: EH_HANDLED\n", 540 SAS_DPRINTK("command 0x%p, task 0x%p, timed out: EH_HANDLED\n",