|
@@ -3899,9 +3899,18 @@ _scsih_temp_threshold_events(struct MPT3SAS_ADAPTER *ioc,
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-static inline bool ata_12_16_cmd(struct scsi_cmnd *scmd)
|
|
|
+static int _scsih_set_satl_pending(struct scsi_cmnd *scmd, bool pending)
|
|
|
{
|
|
|
- return (scmd->cmnd[0] == ATA_12 || scmd->cmnd[0] == ATA_16);
|
|
|
+ struct MPT3SAS_DEVICE *priv = scmd->device->hostdata;
|
|
|
+
|
|
|
+ if (scmd->cmnd[0] != ATA_12 && scmd->cmnd[0] != ATA_16)
|
|
|
+ return 0;
|
|
|
+
|
|
|
+ if (pending)
|
|
|
+ return test_and_set_bit(0, &priv->ata_command_pending);
|
|
|
+
|
|
|
+ clear_bit(0, &priv->ata_command_pending);
|
|
|
+ return 0;
|
|
|
}
|
|
|
|
|
|
/**
|
|
@@ -3925,9 +3934,7 @@ _scsih_flush_running_cmds(struct MPT3SAS_ADAPTER *ioc)
|
|
|
if (!scmd)
|
|
|
continue;
|
|
|
count++;
|
|
|
- if (ata_12_16_cmd(scmd))
|
|
|
- scsi_internal_device_unblock(scmd->device,
|
|
|
- SDEV_RUNNING);
|
|
|
+ _scsih_set_satl_pending(scmd, false);
|
|
|
mpt3sas_base_free_smid(ioc, smid);
|
|
|
scsi_dma_unmap(scmd);
|
|
|
if (ioc->pci_error_recovery)
|
|
@@ -4063,13 +4070,6 @@ scsih_qcmd(struct Scsi_Host *shost, struct scsi_cmnd *scmd)
|
|
|
if (ioc->logging_level & MPT_DEBUG_SCSI)
|
|
|
scsi_print_command(scmd);
|
|
|
|
|
|
- /*
|
|
|
- * Lock the device for any subsequent command until command is
|
|
|
- * done.
|
|
|
- */
|
|
|
- if (ata_12_16_cmd(scmd))
|
|
|
- scsi_internal_device_block(scmd->device);
|
|
|
-
|
|
|
sas_device_priv_data = scmd->device->hostdata;
|
|
|
if (!sas_device_priv_data || !sas_device_priv_data->sas_target) {
|
|
|
scmd->result = DID_NO_CONNECT << 16;
|
|
@@ -4083,6 +4083,19 @@ scsih_qcmd(struct Scsi_Host *shost, struct scsi_cmnd *scmd)
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
|
+ /*
|
|
|
+ * Bug work around for firmware SATL handling. The loop
|
|
|
+ * is based on atomic operations and ensures consistency
|
|
|
+ * since we're lockless at this point
|
|
|
+ */
|
|
|
+ do {
|
|
|
+ if (test_bit(0, &sas_device_priv_data->ata_command_pending)) {
|
|
|
+ scmd->result = SAM_STAT_BUSY;
|
|
|
+ scmd->scsi_done(scmd);
|
|
|
+ return 0;
|
|
|
+ }
|
|
|
+ } while (_scsih_set_satl_pending(scmd, true));
|
|
|
+
|
|
|
sas_target_priv_data = sas_device_priv_data->sas_target;
|
|
|
|
|
|
/* invalid device handle */
|
|
@@ -4650,8 +4663,7 @@ _scsih_io_done(struct MPT3SAS_ADAPTER *ioc, u16 smid, u8 msix_index, u32 reply)
|
|
|
if (scmd == NULL)
|
|
|
return 1;
|
|
|
|
|
|
- if (ata_12_16_cmd(scmd))
|
|
|
- scsi_internal_device_unblock(scmd->device, SDEV_RUNNING);
|
|
|
+ _scsih_set_satl_pending(scmd, false);
|
|
|
|
|
|
mpi_request = mpt3sas_base_get_msg_frame(ioc, smid);
|
|
|
|