Commit 39af7a98 authored by nagalakshmi.nandigama@lsi.com's avatar nagalakshmi.nandigama@lsi.com Committed by James Bottomley

[SCSI] mpt2sas: Fix for hard drive going OFFLINE when hard reset issued and...

[SCSI] mpt2sas: Fix for hard drive going OFFLINE when hard reset issued and simultaneously another hard drive is hot unplugged

Following the host reset, the firmware discovery is reassigning another hard
drive in the topology to the same device handle as that device is getting hot
removed. Until the driver device removal routine is called, there will be two
hard drive with the matching device handle in the internal device link
list. In the device removal routine, a separate function which moves the
device from BLOCKED into OFFLINE state.  Since this routine is passed with the
device handle passed as input parameter, the routine will be traversing the
internal device link list searching for matching device handle. This results
in two devices with matching device handle, therefore both devices goes
OFFLINE.

To fix this issue,the input parameter is changed from device handle to SAS
address, therefore only the device that is hot unplugged will be placed in
OFFLINE state.
Signed-off-by: default avatarNagalakshmi Nandigama <nagalakshmi.nandigama@lsi.com>
Cc: stable@vger.kernel.org
Signed-off-by: default avatarJames Bottomley <JBottomley@Parallels.com>
parent 64bb8138
...@@ -2915,7 +2915,7 @@ _scsih_ublock_io_all_device(struct MPT2SAS_ADAPTER *ioc) ...@@ -2915,7 +2915,7 @@ _scsih_ublock_io_all_device(struct MPT2SAS_ADAPTER *ioc)
* During device pull we need to appropiately set the sdev state. * During device pull we need to appropiately set the sdev state.
*/ */
static void static void
_scsih_ublock_io_device(struct MPT2SAS_ADAPTER *ioc, u16 handle) _scsih_ublock_io_device(struct MPT2SAS_ADAPTER *ioc, u64 sas_address)
{ {
struct MPT2SAS_DEVICE *sas_device_priv_data; struct MPT2SAS_DEVICE *sas_device_priv_data;
struct scsi_device *sdev; struct scsi_device *sdev;
...@@ -2926,10 +2926,12 @@ _scsih_ublock_io_device(struct MPT2SAS_ADAPTER *ioc, u16 handle) ...@@ -2926,10 +2926,12 @@ _scsih_ublock_io_device(struct MPT2SAS_ADAPTER *ioc, u16 handle)
continue; continue;
if (!sas_device_priv_data->block) if (!sas_device_priv_data->block)
continue; continue;
if (sas_device_priv_data->sas_target->handle == handle) { if (sas_device_priv_data->sas_target->sas_address ==
sas_address) {
dewtprintk(ioc, sdev_printk(KERN_INFO, sdev, dewtprintk(ioc, sdev_printk(KERN_INFO, sdev,
MPT2SAS_INFO_FMT "SDEV_RUNNING: " MPT2SAS_INFO_FMT "SDEV_RUNNING: "
"handle(0x%04x)\n", ioc->name, handle)); "sas address(0x%016llx)\n", ioc->name,
(unsigned long long)sas_address));
sas_device_priv_data->block = 0; sas_device_priv_data->block = 0;
scsi_internal_device_unblock(sdev); scsi_internal_device_unblock(sdev);
} }
...@@ -3137,7 +3139,7 @@ _scsih_tm_tr_send(struct MPT2SAS_ADAPTER *ioc, u16 handle) ...@@ -3137,7 +3139,7 @@ _scsih_tm_tr_send(struct MPT2SAS_ADAPTER *ioc, u16 handle)
dewtprintk(ioc, printk(MPT2SAS_INFO_FMT "setting delete flag: " dewtprintk(ioc, printk(MPT2SAS_INFO_FMT "setting delete flag: "
"handle(0x%04x), sas_addr(0x%016llx)\n", ioc->name, handle, "handle(0x%04x), sas_addr(0x%016llx)\n", ioc->name, handle,
(unsigned long long)sas_address)); (unsigned long long)sas_address));
_scsih_ublock_io_device(ioc, handle); _scsih_ublock_io_device(ioc, sas_address);
sas_target_priv_data->handle = MPT2SAS_INVALID_DEVICE_HANDLE; sas_target_priv_data->handle = MPT2SAS_INVALID_DEVICE_HANDLE;
} }
...@@ -5185,7 +5187,7 @@ _scsih_check_device(struct MPT2SAS_ADAPTER *ioc, u16 handle) ...@@ -5185,7 +5187,7 @@ _scsih_check_device(struct MPT2SAS_ADAPTER *ioc, u16 handle)
return; return;
} }
spin_unlock_irqrestore(&ioc->sas_device_lock, flags); spin_unlock_irqrestore(&ioc->sas_device_lock, flags);
_scsih_ublock_io_device(ioc, handle); _scsih_ublock_io_device(ioc, sas_address);
} }
...@@ -5322,7 +5324,7 @@ _scsih_remove_device(struct MPT2SAS_ADAPTER *ioc, ...@@ -5322,7 +5324,7 @@ _scsih_remove_device(struct MPT2SAS_ADAPTER *ioc,
if (sas_device->starget && sas_device->starget->hostdata) { if (sas_device->starget && sas_device->starget->hostdata) {
sas_target_priv_data = sas_device->starget->hostdata; sas_target_priv_data = sas_device->starget->hostdata;
sas_target_priv_data->deleted = 1; sas_target_priv_data->deleted = 1;
_scsih_ublock_io_device(ioc, sas_device->handle); _scsih_ublock_io_device(ioc, sas_device->sas_address);
sas_target_priv_data->handle = sas_target_priv_data->handle =
MPT2SAS_INVALID_DEVICE_HANDLE; MPT2SAS_INVALID_DEVICE_HANDLE;
} }
...@@ -6998,8 +7000,8 @@ _scsih_remove_unresponding_sas_devices(struct MPT2SAS_ADAPTER *ioc) ...@@ -6998,8 +7000,8 @@ _scsih_remove_unresponding_sas_devices(struct MPT2SAS_ADAPTER *ioc)
list_for_each_entry_safe(sas_device, sas_device_next, list_for_each_entry_safe(sas_device, sas_device_next,
&ioc->sas_device_list, list) { &ioc->sas_device_list, list) {
if (!sas_device->responding) if (!sas_device->responding)
_scsih_device_remove_by_handle(ioc, mpt2sas_device_remove_by_sas_address(ioc,
sas_device->handle); sas_device->sas_address);
else else
sas_device->responding = 0; sas_device->responding = 0;
} }
......
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