Commit 15a36575 authored by Mike Anderson's avatar Mike Anderson Committed by James Bottomley

[PATCH] scsi_error handler update. (3/4)

This patch series is against scsi-misc-2.5.

02_serror-hndlr-1.diff:
	- Change to using eh_cmd_list.
	- Change scsi_unjam_host to get sense, abort cmds, ready
	  devices, and disposition cmds for retry or finish.
	- Moved retries outside of eh.

-andmike
--
Michael Anderson
andmike@us.ibm.com

 scsi_error.c |  477 +++++++++++++++++++++++++++++------------------------------
 1 files changed, 241 insertions(+), 236 deletions(-)
parent d6005ee1
...@@ -211,36 +211,36 @@ int scsi_block_when_processing_errors(Scsi_Device *sdev) ...@@ -211,36 +211,36 @@ int scsi_block_when_processing_errors(Scsi_Device *sdev)
* @sc_list: List for failed cmds. * @sc_list: List for failed cmds.
* @shost: scsi host being recovered. * @shost: scsi host being recovered.
**/ **/
static void scsi_eh_prt_fail_stats(Scsi_Cmnd *sc_list, struct Scsi_Host *shost) static inline void scsi_eh_prt_fail_stats(struct Scsi_Host *shost)
{ {
Scsi_Cmnd *scmd; struct scsi_cmnd *scmd;
Scsi_Device *sdev; struct scsi_device *sdev;
int total_failures = 0; int total_failures = 0;
int cmd_failed = 0; int cmd_failed = 0;
int cmd_timed_out = 0; int cmd_cancel = 0;
int devices_failed = 0; int devices_failed = 0;
list_for_each_entry(sdev, &shost->my_devices, siblings) { list_for_each_entry(sdev, &shost->my_devices, siblings) {
for (scmd = sc_list; scmd; scmd = scmd->bh_next) { list_for_each_entry(scmd, &shost->eh_cmd_list, eh_list) {
if (scmd->device == sdev) { if (scmd->device == sdev) {
++total_failures; ++total_failures;
if (scsi_eh_eflags_chk(scmd, if (scsi_eh_eflags_chk(scmd,
SCSI_EH_CMD_TIMEOUT)) SCSI_EH_CANCEL_CMD))
++cmd_timed_out; ++cmd_cancel;
else else
++cmd_failed; ++cmd_failed;
} }
} }
if (cmd_timed_out || cmd_failed) { if (cmd_cancel || cmd_failed) {
SCSI_LOG_ERROR_RECOVERY(3, SCSI_LOG_ERROR_RECOVERY(3,
printk("%s: %d:%d:%d:%d cmds failed: %d," printk("%s: %d:%d:%d:%d cmds failed: %d,"
" timedout: %d\n", " cancel: %d\n",
__FUNCTION__, shost->host_no, __FUNCTION__, shost->host_no,
sdev->channel, sdev->id, sdev->lun, sdev->channel, sdev->id, sdev->lun,
cmd_failed, cmd_timed_out)); cmd_failed, cmd_cancel));
cmd_timed_out = 0; cmd_cancel = 0;
cmd_failed = 0; cmd_failed = 0;
++devices_failed; ++devices_failed;
} }
...@@ -252,68 +252,6 @@ static void scsi_eh_prt_fail_stats(Scsi_Cmnd *sc_list, struct Scsi_Host *shost) ...@@ -252,68 +252,6 @@ static void scsi_eh_prt_fail_stats(Scsi_Cmnd *sc_list, struct Scsi_Host *shost)
} }
#endif #endif
/**
* scsi_eh_get_failed - Gather failed cmds.
* @sc_list: A pointer to a list for failed cmds.
* @shost: Scsi host being recovered.
*
* XXX Add opaque interator for device / shost. Investigate direct
* addition to per eh list on error allowing skipping of this step.
**/
static void scsi_eh_get_failed(Scsi_Cmnd **sc_list, struct Scsi_Host *shost)
{
int found;
Scsi_Device *sdev;
Scsi_Cmnd *scmd;
found = 0;
list_for_each_entry(sdev, &shost->my_devices, siblings) {
unsigned long flags;
spin_lock_irqsave(&sdev->list_lock, flags);
list_for_each_entry(scmd, &sdev->cmd_list, list) {
if (scsi_eh_eflags_chk(scmd, SCSI_EH_CMD_ERR)) {
scmd->bh_next = *sc_list;
*sc_list = scmd;
found++;
} else {
/*
* FIXME Verify how this can happen and if
* this is still needed??
*/
if (scmd->state != SCSI_STATE_INITIALIZING
&& scmd->state != SCSI_STATE_UNUSED) {
/*
* Rats. Something is still floating
* around out there This could be the
* result of the fact that the upper level
* drivers are still frobbing commands
* that might have succeeded. There are
* two outcomes. One is that the command
* block will eventually be freed, and the
* other one is that the command will be
* queued and will be finished along the
* way.
*/
SCSI_LOG_ERROR_RECOVERY(1, printk("Error hdlr"
" prematurely woken"
" cmds still active"
" (%p %x %d)\n",
scmd, scmd->state,
scmd->device->id));
}
}
}
spin_unlock_irqrestore(&sdev->list_lock, flags);
}
SCSI_LOG_ERROR_RECOVERY(1, scsi_eh_prt_fail_stats(*sc_list, shost));
if (shost->host_failed != found)
printk(KERN_ERR "%s: host_failed: %d != found: %d\n",
__FUNCTION__, shost->host_failed, found);
}
/** /**
* scsi_check_sense - Examine scsi cmd sense * scsi_check_sense - Examine scsi cmd sense
* @scmd: Cmd to have sense checked. * @scmd: Cmd to have sense checked.
...@@ -570,7 +508,8 @@ static int scsi_send_eh_cmnd(Scsi_Cmnd *scmd, int timeout) ...@@ -570,7 +508,8 @@ static int scsi_send_eh_cmnd(Scsi_Cmnd *scmd, int timeout)
spin_lock_irqsave(scmd->device->host->host_lock, flags); spin_lock_irqsave(scmd->device->host->host_lock, flags);
if (scmd->device->host->hostt->eh_abort_handler) if (scmd->device->host->hostt->eh_abort_handler)
scmd->device->host->hostt->eh_abort_handler(scmd); scmd->device->host->hostt->eh_abort_handler(scmd);
spin_unlock_irqrestore(scmd->device->host->host_lock, flags); spin_unlock_irqrestore(scmd->device->host->host_lock,
flags);
scmd->request->rq_status = RQ_SCSI_DONE; scmd->request->rq_status = RQ_SCSI_DONE;
scmd->owner = SCSI_OWNER_ERROR_HANDLER; scmd->owner = SCSI_OWNER_ERROR_HANDLER;
...@@ -712,6 +651,7 @@ static int scsi_eh_retry_cmd(Scsi_Cmnd *scmd) ...@@ -712,6 +651,7 @@ static int scsi_eh_retry_cmd(Scsi_Cmnd *scmd)
* scsi_eh_finish_cmd - Handle a cmd that eh is finished with. * scsi_eh_finish_cmd - Handle a cmd that eh is finished with.
* @scmd: Original SCSI cmd that eh has finished. * @scmd: Original SCSI cmd that eh has finished.
* @shost: SCSI host that cmd originally failed on. * @shost: SCSI host that cmd originally failed on.
* @done_list: list_head for processed commands.
* *
* Notes: * Notes:
* We don't want to use the normal command completion while we are are * We don't want to use the normal command completion while we are are
...@@ -720,7 +660,8 @@ static int scsi_eh_retry_cmd(Scsi_Cmnd *scmd) ...@@ -720,7 +660,8 @@ static int scsi_eh_retry_cmd(Scsi_Cmnd *scmd)
* keep a list of pending commands for final completion, and once we * keep a list of pending commands for final completion, and once we
* are ready to leave error handling we handle completion for real. * are ready to leave error handling we handle completion for real.
**/ **/
static void scsi_eh_finish_cmd(Scsi_Cmnd *scmd, struct Scsi_Host *shost) static void scsi_eh_finish_cmd(Scsi_Cmnd *scmd, struct Scsi_Host *shost,
struct list_head *done_list )
{ {
shost->host_failed--; shost->host_failed--;
scmd->state = SCSI_STATE_BHQUEUE; scmd->state = SCSI_STATE_BHQUEUE;
...@@ -731,12 +672,14 @@ static void scsi_eh_finish_cmd(Scsi_Cmnd *scmd, struct Scsi_Host *shost) ...@@ -731,12 +672,14 @@ static void scsi_eh_finish_cmd(Scsi_Cmnd *scmd, struct Scsi_Host *shost)
* things. * things.
*/ */
scsi_setup_cmd_retry(scmd); scsi_setup_cmd_retry(scmd);
list_move_tail(&scmd->eh_list, done_list);
} }
/** /**
* scsi_eh_get_sense - Get device sense data. * scsi_eh_get_sense - Get device sense data.
* @sc_todo: list of cmds that have failed.
* @shost: scsi host being recovered. * @shost: scsi host being recovered.
* @done_list: list_head for processed commands.
* *
* Description: * Description:
* See if we need to request sense information. if so, then get it * See if we need to request sense information. if so, then get it
...@@ -754,23 +697,23 @@ static void scsi_eh_finish_cmd(Scsi_Cmnd *scmd, struct Scsi_Host *shost) ...@@ -754,23 +697,23 @@ static void scsi_eh_finish_cmd(Scsi_Cmnd *scmd, struct Scsi_Host *shost)
* *
* In 2.5 this capability will be going away. * In 2.5 this capability will be going away.
**/ **/
static int scsi_eh_get_sense(Scsi_Cmnd *sc_todo, struct Scsi_Host *shost) static int scsi_eh_get_sense(struct Scsi_Host *shost,
struct list_head *done_list)
{ {
int rtn; int rtn;
struct list_head *lh, *lh_sf;
Scsi_Cmnd *scmd; Scsi_Cmnd *scmd;
SCSI_LOG_ERROR_RECOVERY(3, printk("%s: checking to see if we need" list_for_each_safe(lh, lh_sf, &shost->eh_cmd_list) {
" to request sense\n", scmd = list_entry(lh, struct scsi_cmnd, eh_list);
__FUNCTION__)); if (scsi_eh_eflags_chk(scmd, SCSI_EH_CANCEL_CMD) ||
for (scmd = sc_todo; scmd; scmd = scmd->bh_next) {
if (!scsi_eh_eflags_chk(scmd, SCSI_EH_CMD_FAILED) ||
SCSI_SENSE_VALID(scmd)) SCSI_SENSE_VALID(scmd))
continue; continue;
SCSI_LOG_ERROR_RECOVERY(2, printk("%s: requesting sense" SCSI_LOG_ERROR_RECOVERY(2, printk("%s: requesting sense"
" for tgt: %d\n", " for id: %d\n",
__FUNCTION__, scmd->device->id)); current->comm,
scmd->device->id));
rtn = scsi_request_sense(scmd); rtn = scsi_request_sense(scmd);
if (rtn != SUCCESS) if (rtn != SUCCESS)
continue; continue;
...@@ -787,7 +730,7 @@ static int scsi_eh_get_sense(Scsi_Cmnd *sc_todo, struct Scsi_Host *shost) ...@@ -787,7 +730,7 @@ static int scsi_eh_get_sense(Scsi_Cmnd *sc_todo, struct Scsi_Host *shost)
* upper level. * upper level.
*/ */
if (rtn == SUCCESS) if (rtn == SUCCESS)
scsi_eh_finish_cmd(scmd, shost); scsi_eh_finish_cmd(scmd, shost, done_list);
if (rtn != NEEDS_RETRY) if (rtn != NEEDS_RETRY)
continue; continue;
...@@ -806,10 +749,10 @@ static int scsi_eh_get_sense(Scsi_Cmnd *sc_todo, struct Scsi_Host *shost) ...@@ -806,10 +749,10 @@ static int scsi_eh_get_sense(Scsi_Cmnd *sc_todo, struct Scsi_Host *shost)
/* /*
* we eventually hand this one back to the top level. * we eventually hand this one back to the top level.
*/ */
scsi_eh_finish_cmd(scmd, shost); scsi_eh_finish_cmd(scmd, shost, done_list);
} }
return shost->host_failed; return list_empty(&shost->eh_cmd_list);
} }
/** /**
...@@ -899,9 +842,9 @@ static int scsi_eh_tur(Scsi_Cmnd *scmd) ...@@ -899,9 +842,9 @@ static int scsi_eh_tur(Scsi_Cmnd *scmd)
} }
/** /**
* scsi_eh_abort_cmd - abort a timed-out cmd. * scsi_eh_abort_cmds - abort canceled commands.
* @sc_todo: A list of cmds that have failed.
* @shost: scsi host being recovered. * @shost: scsi host being recovered.
* @done_list: list_head for processed commands.
* *
* Decription: * Decription:
* Try and see whether or not it makes sense to try and abort the * Try and see whether or not it makes sense to try and abort the
...@@ -910,29 +853,36 @@ static int scsi_eh_tur(Scsi_Cmnd *scmd) ...@@ -910,29 +853,36 @@ static int scsi_eh_tur(Scsi_Cmnd *scmd)
* no sense to try and abort the command, since as far as the shost * no sense to try and abort the command, since as far as the shost
* adapter is concerned, it isn't running. * adapter is concerned, it isn't running.
**/ **/
static int scsi_eh_abort_cmd(Scsi_Cmnd *sc_todo, struct Scsi_Host *shost) static int scsi_eh_abort_cmds(struct Scsi_Host *shost,
struct list_head *done_list)
{ {
int rtn; int rtn;
Scsi_Cmnd *scmd; struct list_head *lh, *lh_sf;
struct scsi_cmnd *scmd;
SCSI_LOG_ERROR_RECOVERY(3, printk("%s: checking to see if we need"
" to abort cmd\n", __FUNCTION__));
for (scmd = sc_todo; scmd; scmd = scmd->bh_next) { list_for_each_safe(lh, lh_sf, &shost->eh_cmd_list) {
if (!scsi_eh_eflags_chk(scmd, SCSI_EH_CMD_TIMEOUT)) scmd = list_entry(lh, struct scsi_cmnd, eh_list);
if (!scsi_eh_eflags_chk(scmd, SCSI_EH_CANCEL_CMD))
continue; continue;
SCSI_LOG_ERROR_RECOVERY(3, printk("%s: aborting cmd:"
"0x%p\n", current->comm,
scmd));
rtn = scsi_try_to_abort_cmd(scmd); rtn = scsi_try_to_abort_cmd(scmd);
if (rtn == SUCCESS) { if (rtn == SUCCESS) {
if (!scsi_eh_tur(scmd)) { scsi_eh_eflags_clr(scmd, SCSI_EH_CANCEL_CMD);
rtn = scsi_eh_retry_cmd(scmd); if (!scmd->device->online || !scsi_eh_tur(scmd)) {
if (rtn == SUCCESS) scsi_eh_finish_cmd(scmd, shost, done_list);
scsi_eh_finish_cmd(scmd, shost);
} }
}
} else
SCSI_LOG_ERROR_RECOVERY(3, printk("%s: aborting"
" cmd failed:"
"0x%p\n",
current->comm,
scmd));
} }
return shost->host_failed;
return list_empty(&shost->eh_cmd_list);
} }
/** /**
...@@ -968,9 +918,9 @@ static int scsi_try_bus_device_reset(Scsi_Cmnd *scmd) ...@@ -968,9 +918,9 @@ static int scsi_try_bus_device_reset(Scsi_Cmnd *scmd)
} }
/** /**
* scsi_eh_bus_device_reset - send bdr is needed * scsi_eh_bus_device_reset - send bdr if needed
* @sc_todo: a list of cmds that have failed.
* @shost: scsi host being recovered. * @shost: scsi host being recovered.
* @done_list: list_head for processed commands.
* *
* Notes: * Notes:
* Try a bus device reset. still, look to see whether we have multiple * Try a bus device reset. still, look to see whether we have multiple
...@@ -978,39 +928,52 @@ static int scsi_try_bus_device_reset(Scsi_Cmnd *scmd) ...@@ -978,39 +928,52 @@ static int scsi_try_bus_device_reset(Scsi_Cmnd *scmd)
* makes no sense to try bus_device_reset - we really would need to try * makes no sense to try bus_device_reset - we really would need to try
* a bus_reset instead. * a bus_reset instead.
**/ **/
static int scsi_eh_bus_device_reset(Scsi_Cmnd *sc_todo, struct Scsi_Host *shost) static int scsi_eh_bus_device_reset(struct Scsi_Host *shost,
struct list_head *done_list)
{ {
int rtn; int rtn;
Scsi_Cmnd *scmd; struct list_head *lh, *lh_sf;
Scsi_Device *sdev; struct scsi_cmnd *scmd, *bdr_scmd;
struct scsi_device *sdev;
SCSI_LOG_ERROR_RECOVERY(3, printk("%s: Trying BDR\n", __FUNCTION__));
list_for_each_entry(sdev, &shost->my_devices, siblings) { list_for_each_entry(sdev, &shost->my_devices, siblings) {
for (scmd = sc_todo; scmd; scmd = scmd->bh_next) bdr_scmd = NULL;
if ((scmd->device == sdev) && list_for_each_entry(scmd, &shost->eh_cmd_list, eh_list)
scsi_eh_eflags_chk(scmd, SCSI_EH_CMD_ERR)) if (scmd->device == sdev) {
bdr_scmd = scmd;
break; break;
}
if (!scmd) if (!bdr_scmd)
continue; continue;
/* SCSI_LOG_ERROR_RECOVERY(3, printk("%s: Sending BDR sdev:"
* ok, we have a device that is having problems. try and send " 0x%p\n", current->comm,
* a bus device reset to it. sdev));
*/ rtn = scsi_try_bus_device_reset(bdr_scmd);
rtn = scsi_try_bus_device_reset(scmd); if (rtn == SUCCESS) {
if ((rtn == SUCCESS) && (!scsi_eh_tur(scmd))) if (!sdev->online || !scsi_eh_tur(bdr_scmd)) {
for (scmd = sc_todo; scmd; scmd = scmd->bh_next) list_for_each_safe(lh, lh_sf,
if ((scmd->device == sdev) && &shost->eh_cmd_list) {
scsi_eh_eflags_chk(scmd, SCSI_EH_CMD_ERR)) { scmd = list_entry(lh, struct
rtn = scsi_eh_retry_cmd(scmd); scsi_cmnd,
if (rtn == SUCCESS) eh_list);
scsi_eh_finish_cmd(scmd, shost); if (scmd->device == sdev)
} scsi_eh_finish_cmd(scmd,
shost,
done_list);
}
}
} else {
SCSI_LOG_ERROR_RECOVERY(3, printk("%s: BDR"
" failed sdev:"
"0x%p\n",
current->comm,
sdev));
}
} }
return shost->host_failed; return list_empty(&shost->eh_cmd_list);
} }
/** /**
...@@ -1040,7 +1003,8 @@ static int scsi_try_bus_reset(Scsi_Cmnd *scmd) ...@@ -1040,7 +1003,8 @@ static int scsi_try_bus_reset(Scsi_Cmnd *scmd)
/* /*
* Mark all affected devices to expect a unit attention. * Mark all affected devices to expect a unit attention.
*/ */
list_for_each_entry(sdev, &scmd->device->host->my_devices, siblings) list_for_each_entry(sdev, &scmd->device->host->my_devices,
siblings)
if (scmd->device->channel == sdev->channel) { if (scmd->device->channel == sdev->channel) {
sdev->was_reset = 1; sdev->was_reset = 1;
sdev->expecting_cc_ua = 1; sdev->expecting_cc_ua = 1;
...@@ -1076,7 +1040,8 @@ static int scsi_try_host_reset(Scsi_Cmnd *scmd) ...@@ -1076,7 +1040,8 @@ static int scsi_try_host_reset(Scsi_Cmnd *scmd)
/* /*
* Mark all affected devices to expect a unit attention. * Mark all affected devices to expect a unit attention.
*/ */
list_for_each_entry(sdev, &scmd->device->host->my_devices, siblings) list_for_each_entry(sdev, &scmd->device->host->my_devices,
siblings)
if (scmd->device->channel == sdev->channel) { if (scmd->device->channel == sdev->channel) {
sdev->was_reset = 1; sdev->was_reset = 1;
sdev->expecting_cc_ua = 1; sdev->expecting_cc_ua = 1;
...@@ -1086,26 +1051,20 @@ static int scsi_try_host_reset(Scsi_Cmnd *scmd) ...@@ -1086,26 +1051,20 @@ static int scsi_try_host_reset(Scsi_Cmnd *scmd)
} }
/** /**
* scsi_eh_bus_host_reset - send a bus reset and on failure try host reset * scsi_eh_bus_reset - send a bus reset
* @sc_todo: a list of cmds that have failed.
* @shost: scsi host being recovered. * @shost: scsi host being recovered.
* @done_list: list_head for processed commands.
**/ **/
static int scsi_eh_bus_host_reset(Scsi_Cmnd *sc_todo, struct Scsi_Host *shost) static int scsi_eh_bus_reset(struct Scsi_Host *shost,
struct list_head *done_list)
{ {
int rtn; int rtn;
struct list_head *lh, *lh_sf;
Scsi_Cmnd *scmd; Scsi_Cmnd *scmd;
Scsi_Cmnd *chan_scmd; Scsi_Cmnd *chan_scmd;
unsigned int channel; unsigned int channel;
/* /*
* if we ended up here, we have serious problems. the only thing left
* to try is a full bus reset. if someone has grabbed the bus and isn't
* letting go, then perhaps this will help.
*/
SCSI_LOG_ERROR_RECOVERY(3, printk("%s: Try Bus/Host RST\n",
__FUNCTION__));
/*
* we really want to loop over the various channels, and do this on * we really want to loop over the various channels, and do this on
* a channel by channel basis. we should also check to see if any * a channel by channel basis. we should also check to see if any
* of the failed commands are on soft_reset devices, and if so, skip * of the failed commands are on soft_reset devices, and if so, skip
...@@ -1113,9 +1072,8 @@ static int scsi_eh_bus_host_reset(Scsi_Cmnd *sc_todo, struct Scsi_Host *shost) ...@@ -1113,9 +1072,8 @@ static int scsi_eh_bus_host_reset(Scsi_Cmnd *sc_todo, struct Scsi_Host *shost)
*/ */
for (channel = 0; channel <= shost->max_channel; channel++) { for (channel = 0; channel <= shost->max_channel; channel++) {
for (scmd = sc_todo; scmd; scmd = scmd->bh_next) { chan_scmd = NULL;
if (!scsi_eh_eflags_chk(scmd, SCSI_EH_CMD_ERR)) list_for_each_entry(scmd, &shost->eh_cmd_list, eh_list) {
continue;
if (channel == scmd->device->channel) { if (channel == scmd->device->channel) {
chan_scmd = scmd; chan_scmd = scmd;
break; break;
...@@ -1126,63 +1084,97 @@ static int scsi_eh_bus_host_reset(Scsi_Cmnd *sc_todo, struct Scsi_Host *shost) ...@@ -1126,63 +1084,97 @@ static int scsi_eh_bus_host_reset(Scsi_Cmnd *sc_todo, struct Scsi_Host *shost)
} }
} }
if (!scmd) if (!chan_scmd)
continue; continue;
SCSI_LOG_ERROR_RECOVERY(3, printk("%s: Sending BRST chan:"
" %d\n", current->comm,
channel));
rtn = scsi_try_bus_reset(chan_scmd);
if (rtn == SUCCESS) {
list_for_each_safe(lh, lh_sf, &shost->eh_cmd_list) {
scmd = list_entry(lh, struct scsi_cmnd,
eh_list);
if (channel == scmd->device->channel)
if (!scmd->device->online ||
!scsi_eh_tur(scmd))
scsi_eh_finish_cmd(scmd,
shost,
done_list);
}
} else {
SCSI_LOG_ERROR_RECOVERY(3, printk("%s: BRST"
" failed chan: %d\n",
current->comm,
channel));
}
}
return list_empty(&shost->eh_cmd_list);
}
/* /**
* we now know that we are able to perform a reset for the * scsi_eh_host_reset - send a host reset
* channel that scmd points to. * @shost: scsi host being recovered.
*/ * @done_list: list_head for processed commands.
rtn = scsi_try_bus_reset(scmd); **/
if (rtn != SUCCESS) static int scsi_eh_host_reset(struct Scsi_Host *shost,
rtn = scsi_try_host_reset(scmd); struct list_head *done_list)
{
int rtn;
struct list_head *lh, *lh_sf;
Scsi_Cmnd *scmd;
if (!list_empty(&shost->eh_cmd_list)) {
scmd = list_entry(shost->eh_cmd_list.next,
struct scsi_cmnd, eh_list);
SCSI_LOG_ERROR_RECOVERY(3, printk("%s: Sending HRST\n"
, current->comm));
rtn = scsi_try_host_reset(scmd);
if (rtn == SUCCESS) { if (rtn == SUCCESS) {
for (scmd = sc_todo; scmd; scmd = scmd->bh_next) { list_for_each_safe(lh, lh_sf, &shost->eh_cmd_list) {
if (!scsi_eh_eflags_chk(scmd, SCSI_EH_CMD_ERR) scmd = list_entry(lh, struct scsi_cmnd, eh_list);
|| channel != scmd->device->channel) if (!scmd->device->online || !scsi_eh_tur(scmd))
continue; scsi_eh_finish_cmd(scmd, shost,
if (!scsi_eh_tur(scmd)) { done_list);
rtn = scsi_eh_retry_cmd(scmd);
if (rtn == SUCCESS)
scsi_eh_finish_cmd(scmd, shost);
}
} }
} else {
SCSI_LOG_ERROR_RECOVERY(3, printk("%s: HRST"
" failed\n",
current->comm));
} }
} }
return shost->host_failed; return list_empty(&shost->eh_cmd_list);
} }
/** /**
* scsi_eh_offline_sdevs - offline scsi devices that fail to recover * scsi_eh_offline_sdevs - offline scsi devices that fail to recover
* @sc_todo: a list of cmds that have failed.
* @shost: scsi host being recovered. * @shost: scsi host being recovered.
* @done_list: list_head for processed commands.
* *
**/ **/
static void scsi_eh_offline_sdevs(Scsi_Cmnd *sc_todo, struct Scsi_Host *shost) static void scsi_eh_offline_sdevs(struct Scsi_Host *shost,
struct list_head *done_list)
{ {
struct list_head *lh, *lh_sf;
Scsi_Cmnd *scmd; Scsi_Cmnd *scmd;
for (scmd = sc_todo; scmd; scmd = scmd->bh_next) { list_for_each_safe(lh, lh_sf, &shost->eh_cmd_list) {
if (!scsi_eh_eflags_chk(scmd, SCSI_EH_CMD_ERR)) scmd = list_entry(lh, struct scsi_cmnd, eh_list);
continue;
printk(KERN_INFO "scsi: Device offlined - not" printk(KERN_INFO "scsi: Device offlined - not"
" ready or command retry failed" " ready after error recovery: host"
" after error recovery: host"
" %d channel %d id %d lun %d\n", " %d channel %d id %d lun %d\n",
shost->host_no, shost->host_no,
scmd->device->channel, scmd->device->channel,
scmd->device->id, scmd->device->id,
scmd->device->lun); scmd->device->lun);
scmd->device->online = FALSE;
if (scsi_eh_eflags_chk(scmd, SCSI_EH_CMD_TIMEOUT)) if (scsi_eh_eflags_chk(scmd, SCSI_EH_CANCEL_CMD)) {
scmd->result |= (DRIVER_TIMEOUT << 24); /*
* FIXME: Handle lost cmds.
scmd->device->online = 0; */
scsi_eh_finish_cmd(scmd, shost); }
scsi_eh_finish_cmd(scmd, shost, done_list);
} }
return; return;
} }
...@@ -1477,6 +1469,8 @@ static void scsi_restart_operations(struct Scsi_Host *shost) ...@@ -1477,6 +1469,8 @@ static void scsi_restart_operations(struct Scsi_Host *shost)
ASSERT_LOCK(shost->host_lock, 0); ASSERT_LOCK(shost->host_lock, 0);
shost->in_recovery = 0;
/* /*
* If the door was locked, we need to insert a door lock request * If the door was locked, we need to insert a door lock request
* onto the head of the SCSI request queue for the device. There * onto the head of the SCSI request queue for the device. There
...@@ -1516,6 +1510,56 @@ static void scsi_restart_operations(struct Scsi_Host *shost) ...@@ -1516,6 +1510,56 @@ static void scsi_restart_operations(struct Scsi_Host *shost)
spin_unlock_irqrestore(shost->host_lock, flags); spin_unlock_irqrestore(shost->host_lock, flags);
} }
/**
* scsi_eh_ready_devs - check device ready state and recover if not.
* @shost: host to be recovered.
* @done_list: list_head for processed commands.
*
**/
static void scsi_eh_ready_devs(struct Scsi_Host *shost,
struct list_head *done_list)
{
if (scsi_eh_bus_device_reset(shost, done_list))
if (scsi_eh_bus_reset(shost, done_list))
if (scsi_eh_host_reset(shost, done_list))
scsi_eh_offline_sdevs(shost, done_list);
}
/**
* scsi_eh_flush_done_list - finish processed commands or retry them.
* @shost: host to be recovered.
* @done_list: list_head of processed commands.
*
**/
static void scsi_eh_flush_done_list(struct Scsi_Host *shost,
struct list_head *done_list)
{
struct list_head *lh, *lh_sf;
Scsi_Cmnd *scmd;
list_for_each_safe(lh, lh_sf, done_list) {
scmd = list_entry(lh, struct scsi_cmnd, eh_list);
list_del_init(lh);
if (!scmd->device->online) {
scmd->result |= (DRIVER_TIMEOUT << 24);
} else {
if (++scmd->retries < scmd->allowed) {
SCSI_LOG_ERROR_RECOVERY(3,
printk("%s: flush retry"
" cmd: %p\n",
current->comm,
scmd));
scsi_retry_command(scmd);
continue;
}
}
SCSI_LOG_ERROR_RECOVERY(3, printk("%s: flush finish"
" cmd: %p\n",
current->comm, scmd));
scsi_finish_command(scmd);
}
}
/** /**
* scsi_unjam_host - Attempt to fix a host which has a cmd that failed. * scsi_unjam_host - Attempt to fix a host which has a cmd that failed.
* @shost: Host to unjam. * @shost: Host to unjam.
...@@ -1541,60 +1585,15 @@ static void scsi_restart_operations(struct Scsi_Host *shost) ...@@ -1541,60 +1585,15 @@ static void scsi_restart_operations(struct Scsi_Host *shost)
**/ **/
static void scsi_unjam_host(struct Scsi_Host *shost) static void scsi_unjam_host(struct Scsi_Host *shost)
{ {
Scsi_Cmnd *sc_todo = NULL; LIST_HEAD(done_list);
Scsi_Cmnd *scmd;
/*
* Is this assert really ok anymore (andmike). Should we at least
* be using spin_lock_unlocked.
*/
ASSERT_LOCK(shost->host_lock, 0);
scsi_eh_get_failed(&sc_todo, shost);
if (scsi_eh_get_sense(sc_todo, shost))
if (scsi_eh_abort_cmd(sc_todo, shost))
if (scsi_eh_bus_device_reset(sc_todo, shost))
if (scsi_eh_bus_host_reset(sc_todo, shost))
scsi_eh_offline_sdevs(sc_todo, shost);
BUG_ON(shost->host_failed); SCSI_LOG_ERROR_RECOVERY(1, scsi_eh_prt_fail_stats(shost));
/*
* We are currently holding these things in a linked list - we
* didn't put them in the bottom half queue because we wanted to
* keep things quiet while we were working on recovery, and
* passing them up to the top level could easily cause the top
* level to try and queue something else again.
*
* start by marking that the host is no longer in error recovery.
*/
shost->in_recovery = 0;
/* if (!scsi_eh_get_sense(shost, &done_list))
* take the list of commands, and stick them in the bottom half queue. if (!scsi_eh_abort_cmds(shost, &done_list))
* the current implementation of scsi_done will do this for us - if need scsi_eh_ready_devs(shost, &done_list);
* be we can create a special version of this function to do the
* same job for us.
*/
for (scmd = sc_todo; scmd; scmd = sc_todo) {
sc_todo = scmd->bh_next;
scmd->bh_next = NULL;
/*
* Oh, this is a vile hack. scsi_done() expects a timer
* to be running on the command. If there isn't, it assumes
* that the command has actually timed out, and a timer
* handler is running. That may well be how we got into
* this fix, but right now things are stable. We add
* a timer back again so that we can report completion.
* scsi_done() will immediately remove said timer from
* the command, and then process it.
*/
scsi_add_timer(scmd, 100, scsi_eh_times_out);
scsi_done(scmd);
}
scsi_eh_flush_done_list(shost, &done_list);
} }
/** /**
...@@ -1642,7 +1641,8 @@ void scsi_error_handler(void *data) ...@@ -1642,7 +1641,8 @@ void scsi_error_handler(void *data)
/* /*
* Wake up the thread that created us. * Wake up the thread that created us.
*/ */
SCSI_LOG_ERROR_RECOVERY(3, printk("Wake up parent of scsi_eh_%d\n",shost->host_no)); SCSI_LOG_ERROR_RECOVERY(3, printk("Wake up parent of"
" scsi_eh_%d\n",shost->host_no));
complete(shost->eh_notify); complete(shost->eh_notify);
...@@ -1652,7 +1652,9 @@ void scsi_error_handler(void *data) ...@@ -1652,7 +1652,9 @@ void scsi_error_handler(void *data)
* away and die. This typically happens if the user is * away and die. This typically happens if the user is
* trying to unload a module. * trying to unload a module.
*/ */
SCSI_LOG_ERROR_RECOVERY(1, printk("Error handler scsi_eh_%d sleeping\n",shost->host_no)); SCSI_LOG_ERROR_RECOVERY(1, printk("Error handler"
" scsi_eh_%d"
" sleeping\n",shost->host_no));
/* /*
* Note - we always use down_interruptible with the semaphore * Note - we always use down_interruptible with the semaphore
...@@ -1667,7 +1669,9 @@ void scsi_error_handler(void *data) ...@@ -1667,7 +1669,9 @@ void scsi_error_handler(void *data)
if (shost->eh_kill) if (shost->eh_kill)
break; break;
SCSI_LOG_ERROR_RECOVERY(1, printk("Error handler scsi_eh_%d waking up\n",shost->host_no)); SCSI_LOG_ERROR_RECOVERY(1, printk("Error handler"
" scsi_eh_%d waking"
" up\n",shost->host_no));
shost->eh_active = 1; shost->eh_active = 1;
...@@ -1695,7 +1699,8 @@ void scsi_error_handler(void *data) ...@@ -1695,7 +1699,8 @@ void scsi_error_handler(void *data)
} }
SCSI_LOG_ERROR_RECOVERY(1, printk("Error handler scsi_eh_%d exiting\n",shost->host_no)); SCSI_LOG_ERROR_RECOVERY(1, printk("Error handler scsi_eh_%d"
" exiting\n",shost->host_no));
/* /*
* Make sure that nobody tries to wake us up again. * Make sure that nobody tries to wake us up again.
......
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