diff options
author | Mark Kettenis <kettenis@cvs.openbsd.org> | 2010-07-03 00:41:59 +0000 |
---|---|---|
committer | Mark Kettenis <kettenis@cvs.openbsd.org> | 2010-07-03 00:41:59 +0000 |
commit | c7fb6fb68e43859610679081e1851b4cb1a25e2a (patch) | |
tree | 263ec6441d16685f037675e30200f730e619c3a8 /sys/dev/ata/atascsi.c | |
parent | e0fdf94d22d7176ed6143f070d3180fa098ab37d (diff) |
Implement translation of the SCSI START STOP UNIT command.
ok dlg@, marco@
Diffstat (limited to 'sys/dev/ata/atascsi.c')
-rw-r--r-- | sys/dev/ata/atascsi.c | 87 |
1 files changed, 85 insertions, 2 deletions
diff --git a/sys/dev/ata/atascsi.c b/sys/dev/ata/atascsi.c index c2e57140e3c..7da8495fb8d 100644 --- a/sys/dev/ata/atascsi.c +++ b/sys/dev/ata/atascsi.c @@ -1,4 +1,4 @@ -/* $OpenBSD: atascsi.c,v 1.87 2010/06/28 18:31:01 krw Exp $ */ +/* $OpenBSD: atascsi.c,v 1.88 2010/07/03 00:41:58 kettenis Exp $ */ /* * Copyright (c) 2007 David Gwynne <dlg@openbsd.org> @@ -89,6 +89,8 @@ void atascsi_disk_capacity16(struct scsi_xfer *); void atascsi_disk_sync(struct scsi_xfer *); void atascsi_disk_sync_done(struct ata_xfer *); void atascsi_disk_sense(struct scsi_xfer *); +void atascsi_disk_start_stop(struct scsi_xfer *); +void atascsi_disk_start_stop_done(struct ata_xfer *); void atascsi_atapi_cmd(struct scsi_xfer *); void atascsi_atapi_cmd_done(struct ata_xfer *); @@ -404,8 +406,11 @@ atascsi_disk_cmd(struct scsi_xfer *xs) atascsi_passthru_16(xs); return; - case TEST_UNIT_READY: case START_STOP: + atascsi_disk_start_stop(xs); + return; + + case TEST_UNIT_READY: case PREVENT_ALLOW: atascsi_done(xs, XS_NOERROR); return; @@ -1043,6 +1048,84 @@ atascsi_disk_sense(struct scsi_xfer *xs) } void +atascsi_disk_start_stop(struct scsi_xfer *xs) +{ + struct scsi_link *link = xs->sc_link; + struct atascsi *as = link->adapter_softc; + struct ata_xfer *xa = xs->io; + struct scsi_start_stop *ss = (struct scsi_start_stop *)xs->cmd; + + if (ss->how != SSS_STOP) { + atascsi_done(xs, XS_NOERROR); + return; + } + + /* + * A SCSI START_STOP UNIT command with the START bit set to + * zero gets translated into an ATA FLUSH CACHE command + * followed by an ATA STANDBY IMMEDIATE command. + */ + xa->datalen = 0; + xa->flags = ATA_F_READ; + xa->complete = atascsi_disk_start_stop_done; + /* Spec says flush cache can take >30 sec, so give it at least 45. */ + xa->timeout = (xs->timeout < 45000) ? 45000 : xs->timeout; + xa->atascsi_private = xs; + if (xs->flags & SCSI_POLL) + xa->flags |= ATA_F_POLL; + + xa->fis->flags = ATA_H2D_FLAGS_CMD; + xa->fis->command = ATA_C_FLUSH_CACHE; + xa->fis->device = 0; + + ata_exec(as, xa); +} + +void +atascsi_disk_start_stop_done(struct ata_xfer *xa) +{ + struct scsi_xfer *xs = xa->atascsi_private; + struct scsi_link *link = xs->sc_link; + struct atascsi *as = link->adapter_softc; + + switch (xa->state) { + case ATA_S_COMPLETE: + break; + + case ATA_S_ERROR: + case ATA_S_TIMEOUT: + xs->error = (xa->state == ATA_S_TIMEOUT ? XS_TIMEOUT : + XS_DRIVER_STUFFUP); + xs->resid = xa->resid; + scsi_done(xs); + return; + + default: + panic("atascsi_disk_start_stop_done: unexpected ata_xfer state (%d)", + xa->state); + } + + /* + * The FLUSH CACHE command completed succesfully; now issue + * the STANDBY IMMEDATE command. + */ + xa->datalen = 0; + xa->flags = ATA_F_READ; + xa->complete = atascsi_disk_cmd_done; + /* Spec says flush cache can take >30 sec, so give it at least 45. */ + xa->timeout = (xs->timeout < 45000) ? 45000 : xs->timeout; + xa->atascsi_private = xs; + if (xs->flags & SCSI_POLL) + xa->flags |= ATA_F_POLL; + + xa->fis->flags = ATA_H2D_FLAGS_CMD; + xa->fis->command = ATA_C_STANDBY_IMMED; + xa->fis->device = 0; + + ata_exec(as, xa); +} + +void atascsi_atapi_cmd(struct scsi_xfer *xs) { struct scsi_link *link = xs->sc_link; |