Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

[SCSI] libsas: check dev->gone before submitting sata i/o

Head off doomed-to-fail i/o in sas_queuecommand before sending it down
the ata path.

Before:
sd 7:0:0:0: [sdd] Synchronizing SCSI cache
ata8: no sense translation for status: 0x00
ata8: translated ATA stat/err 0x00/00 to SCSI SK/ASC/ASCQ 0xb/00/00
ata8.00: device reported invalid CHS sector 0
ata8: status=0x00 { }
ata8: no sense translation for status: 0x00
ata8: translated ATA stat/err 0x00/00 to SCSI SK/ASC/ASCQ 0xb/00/00
ata8.00: device reported invalid CHS sector 0
ata8: status=0x00 { }
ata8: no sense translation for status: 0x00
ata8: translated ATA stat/err 0x00/00 to SCSI SK/ASC/ASCQ 0xb/00/00
ata8.00: device reported invalid CHS sector 0
ata8: status=0x00 { }
sd 7:0:0:0: [sdd] Result: hostbyte=DID_OK driverbyte=DRIVER_SENSE
sd 7:0:0:0: [sdd] Sense Key : Aborted Command [current] [descriptor]
sd 7:0:0:0: [sdd] Add. Sense: No additional sense information
sd 7:0:0:0: [sdd] Stopping disk

After:
sd 9:0:0:0: [sdd] Synchronizing SCSI cache
sd 9:0:0:0: [sdd] Result: hostbyte=DID_BAD_TARGET driverbyte=DRIVER_OK
sd 9:0:0:0: [sdd] Stopping disk
sd 9:0:0:0: [sdd] START_STOP FAILED
sd 9:0:0:0: [sdd] Result: hostbyte=DID_BAD_TARGET driverbyte=DRIVER_OK

This is a cosmetic change as sata i/o can still leak to a gone device,
but this addresses the nominal hotplug case when releasing the target.

Acked-by: Jack Wang <jack_wang@usish.com>
Signed-off-by: Dan Williams <dan.j.williams@intel.com>
Signed-off-by: James Bottomley <jbottomley@parallels.com>

authored by

Dan Williams and committed by
James Bottomley
3673f4bf 90f1e10d

+7 -7
+7 -7
drivers/scsi/libsas/sas_scsi_host.c
··· 207 207 struct sas_ha_struct *sas_ha = dev->port->ha; 208 208 struct sas_task *task; 209 209 210 + /* If the device fell off, no sense in issuing commands */ 211 + if (dev->gone) { 212 + cmd->result = DID_BAD_TARGET << 16; 213 + scsi_done(cmd); 214 + goto out; 215 + } 216 + 210 217 if (dev_is_sata(dev)) { 211 218 unsigned long flags; 212 219 213 220 spin_lock_irqsave(dev->sata_dev.ap->lock, flags); 214 221 res = ata_sas_queuecmd(cmd, dev->sata_dev.ap); 215 222 spin_unlock_irqrestore(dev->sata_dev.ap->lock, flags); 216 - goto out; 217 - } 218 - 219 - /* If the device fell off, no sense in issuing commands */ 220 - if (dev->gone) { 221 - cmd->result = DID_BAD_TARGET << 16; 222 - scsi_done(cmd); 223 223 goto out; 224 224 } 225 225