diff options
author | Kashyap, Desai <kashyap.desai@lsi.com> | 2010-07-26 09:26:21 -0400 |
---|---|---|
committer | James Bottomley <James.Bottomley@suse.de> | 2010-07-28 10:07:46 -0400 |
commit | c9de7dc48307395fb71780b567ae8833b080d1c8 (patch) | |
tree | b034f8833d64b640d9ef9e3f61191d06b51e5027 /drivers/message/fusion/mptscsih.c | |
parent | b5833cbbd7c2bc3462e684feadd8e6a0ca8e5387 (diff) |
[SCSI] mptfusion: Block Error handling for deleting devices or Device in DMD
Issue description:
In multipath topology, when device deletion is in transient state,
multipath driver can call blk_flush_queue() as part of path failure.
Before device get deleted from OS, Device may go OFFLINE as part of error
handling kicked off triggered from multipathing driver. Above condition hits
more frequently if device missing delay timer (which is LSI specific firmware
parameter) is non zero value.
root cause of this issue is Error handling thread is getting kicked off for
device which is not really present(in transient state of deleting).
This patch has solution for this issue. driver is now using eh_timed_out
callback. See below.
mptsas_transport_template->eh_timed_out = mptsas_eh_timed_out
Using mptsas_eh_timed_out function, driver can decide weather vdevice is
under Device missing delay or deleting state.
for either of those cases, there is BLK_EH_RESET_TIMER return to scsi mid
and error handling thread will not be kicked off for that particular scsi
command.
Signed-off-by: Kashyap Desai <kashyap.desai@lsi.com>
Cc: Stable Tree <stable@kernel.org>
Signed-off-by: James Bottomley <James.Bottomley@suse.de>
Diffstat (limited to 'drivers/message/fusion/mptscsih.c')
-rw-r--r-- | drivers/message/fusion/mptscsih.c | 40 |
1 files changed, 31 insertions, 9 deletions
diff --git a/drivers/message/fusion/mptscsih.c b/drivers/message/fusion/mptscsih.c index dceb67a21825..59b8f53d1ece 100644 --- a/drivers/message/fusion/mptscsih.c +++ b/drivers/message/fusion/mptscsih.c | |||
@@ -664,6 +664,7 @@ mptscsih_io_done(MPT_ADAPTER *ioc, MPT_FRAME_HDR *mf, MPT_FRAME_HDR *mr) | |||
664 | u32 log_info; | 664 | u32 log_info; |
665 | 665 | ||
666 | status = le16_to_cpu(pScsiReply->IOCStatus) & MPI_IOCSTATUS_MASK; | 666 | status = le16_to_cpu(pScsiReply->IOCStatus) & MPI_IOCSTATUS_MASK; |
667 | |||
667 | scsi_state = pScsiReply->SCSIState; | 668 | scsi_state = pScsiReply->SCSIState; |
668 | scsi_status = pScsiReply->SCSIStatus; | 669 | scsi_status = pScsiReply->SCSIStatus; |
669 | xfer_cnt = le32_to_cpu(pScsiReply->TransferCount); | 670 | xfer_cnt = le32_to_cpu(pScsiReply->TransferCount); |
@@ -738,15 +739,36 @@ mptscsih_io_done(MPT_ADAPTER *ioc, MPT_FRAME_HDR *mf, MPT_FRAME_HDR *mr) | |||
738 | 739 | ||
739 | case MPI_IOCSTATUS_SCSI_IOC_TERMINATED: /* 0x004B */ | 740 | case MPI_IOCSTATUS_SCSI_IOC_TERMINATED: /* 0x004B */ |
740 | if ( ioc->bus_type == SAS ) { | 741 | if ( ioc->bus_type == SAS ) { |
741 | u16 ioc_status = le16_to_cpu(pScsiReply->IOCStatus); | 742 | u16 ioc_status = |
742 | if (ioc_status & MPI_IOCSTATUS_FLAG_LOG_INFO_AVAILABLE) { | 743 | le16_to_cpu(pScsiReply->IOCStatus); |
743 | if ((log_info & SAS_LOGINFO_MASK) | 744 | if ((ioc_status & |
744 | == SAS_LOGINFO_NEXUS_LOSS) { | 745 | MPI_IOCSTATUS_FLAG_LOG_INFO_AVAILABLE) |
745 | sc->result = | 746 | && |
746 | (DID_TRANSPORT_DISRUPTED | 747 | ((log_info & SAS_LOGINFO_MASK) == |
747 | << 16); | 748 | SAS_LOGINFO_NEXUS_LOSS)) { |
748 | break; | 749 | VirtDevice *vdevice = |
749 | } | 750 | sc->device->hostdata; |
751 | |||
752 | /* flag the device as being in | ||
753 | * device removal delay so we can | ||
754 | * notify the midlayer to hold off | ||
755 | * on timeout eh */ | ||
756 | if (vdevice && vdevice-> | ||
757 | vtarget && | ||
758 | vdevice->vtarget-> | ||
759 | raidVolume) | ||
760 | printk(KERN_INFO | ||
761 | "Skipping Raid Volume" | ||
762 | "for inDMD\n"); | ||
763 | else if (vdevice && | ||
764 | vdevice->vtarget) | ||
765 | vdevice->vtarget-> | ||
766 | inDMD = 1; | ||
767 | |||
768 | sc->result = | ||
769 | (DID_TRANSPORT_DISRUPTED | ||
770 | << 16); | ||
771 | break; | ||
750 | } | 772 | } |
751 | } else if (ioc->bus_type == FC) { | 773 | } else if (ioc->bus_type == FC) { |
752 | /* | 774 | /* |