diff options
author | Christopher Pascoe <pascoe@cvs.openbsd.org> | 2007-03-20 12:24:03 +0000 |
---|---|---|
committer | Christopher Pascoe <pascoe@cvs.openbsd.org> | 2007-03-20 12:24:03 +0000 |
commit | 45ba8af3a396979405c850de786e9480f952778a (patch) | |
tree | f5dc6d57392d86fbba70aed08eaa75fc7c0a898d /sys | |
parent | e89cff3a5d99bf0bcdb6fc1b38737c07643aaa8e (diff) |
Move common completion handling into ata_exec itself. Removes a double free
in the inquiry error paths, as the HBA will have already completed the
ata_xfer with an error.
Reminder and ok dlg@
Diffstat (limited to 'sys')
-rw-r--r-- | sys/dev/ata/atascsi.c | 148 |
1 files changed, 73 insertions, 75 deletions
diff --git a/sys/dev/ata/atascsi.c b/sys/dev/ata/atascsi.c index c0757cedd8b..19e655a9f4c 100644 --- a/sys/dev/ata/atascsi.c +++ b/sys/dev/ata/atascsi.c @@ -1,4 +1,4 @@ -/* $OpenBSD: atascsi.c,v 1.23 2007/03/20 12:01:18 pascoe Exp $ */ +/* $OpenBSD: atascsi.c,v 1.24 2007/03/20 12:24:02 pascoe Exp $ */ /* * Copyright (c) 2007 David Gwynne <dlg@openbsd.org> @@ -347,15 +347,7 @@ atascsi_disk_cmd(struct scsi_xfer *xs) xa->timeout = xs->timeout; xa->atascsi_private = xs; - switch (ata_exec(as, xa)) { - case ATA_COMPLETE: - case ATA_ERROR: - return (COMPLETE); - case ATA_QUEUED: - return (SUCCESSFULLY_QUEUED); - default: - panic("unexpected return from ata_exec"); - } + return (ata_exec(as, xa)); } void @@ -403,19 +395,7 @@ atascsi_disk_inq(struct scsi_xfer *xs) if (xs->flags & SCSI_POLL) xa->flags |= ATA_F_POLL; - switch (ata_exec(as, xa)) { - case ATA_COMPLETE: - return (COMPLETE); - case ATA_QUEUED: - return (SUCCESSFULLY_QUEUED); - case ATA_ERROR: - s = splbio(); - ata_free_identify(xa); - splx(s); - return (atascsi_stuffup(xs)); - default: - panic("unexpected return from ata_exec"); - } + return (ata_exec(as, xa)); } void @@ -425,22 +405,37 @@ atascsi_disk_inq_done(struct ata_xfer *xa) struct ata_identify id; struct scsi_inquiry_data inq; - ata_complete_identify(xa, &id); + switch (xa->state) { + case ATA_S_COMPLETE: + ata_complete_identify(xa, &id); - bzero(&inq, sizeof(inq)); + bzero(&inq, sizeof(inq)); - inq.device = T_DIRECT; - inq.version = 2; - inq.response_format = 2; - inq.additional_length = 32; - bcopy("ATA ", inq.vendor, sizeof(inq.vendor)); - bcopy(id.model, inq.product, sizeof(inq.product)); - bcopy(id.firmware, inq.revision, sizeof(inq.revision)); + inq.device = T_DIRECT; + inq.version = 2; + inq.response_format = 2; + inq.additional_length = 32; + bcopy("ATA ", inq.vendor, sizeof(inq.vendor)); + bcopy(id.model, inq.product, sizeof(inq.product)); + bcopy(id.firmware, inq.revision, sizeof(inq.revision)); - bcopy(&inq, xs->data, MIN(sizeof(inq), xs->datalen)); - xs->error = XS_NOERROR; - xs->flags |= ITSDONE; + bcopy(&inq, xs->data, MIN(sizeof(inq), xs->datalen)); + xs->error = XS_NOERROR; + break; + + case ATA_S_ERROR: + case ATA_S_TIMEOUT: + ata_free_identify(xa); + xs->error = (xa->state == ATA_S_TIMEOUT ? XS_TIMEOUT : + XS_DRIVER_STUFFUP); + break; + default: + panic("atascsi_disk_inq_done: unexpected ata_xfer state (%d)", + xa->state); + } + + xs->flags |= ITSDONE; scsi_done(xs); } @@ -469,19 +464,7 @@ atascsi_disk_capacity(struct scsi_xfer *xs) if (xs->flags & SCSI_POLL) xa->flags |= ATA_F_POLL; - switch (ata_exec(as, xa)) { - case ATA_COMPLETE: - return (COMPLETE); - case ATA_QUEUED: - return (SUCCESSFULLY_QUEUED); - case ATA_ERROR: - s = splbio(); - ata_free_identify(xa); - splx(s); - return (atascsi_stuffup(xs)); - default: - panic("unexpected return from ata_exec"); - } + return (ata_exec(as, xa)); } void @@ -493,27 +476,42 @@ atascsi_disk_capacity_done(struct ata_xfer *xa) u_int32_t capacity; int i; - ata_complete_identify(xa, &id); - - bzero(&rcd, sizeof(rcd)); - if (id.cmdset83 & 0x0400) { - for (i = 3; i >= 0; --i) { + switch (xa->state) { + case ATA_S_COMPLETE: + ata_complete_identify(xa, &id); + + bzero(&rcd, sizeof(rcd)); + if (id.cmdset83 & 0x0400) { /* LBA48 feature set supported */ + for (i = 3; i >= 0; --i) { + capacity <<= 16; + capacity += id.addrsecxt[i]; + } + } else { + capacity = id.addrsec[1]; capacity <<= 16; - capacity += id.addrsecxt[i]; + capacity += id.addrsec[0]; } - } else { - capacity = id.addrsec[1]; - capacity <<= 16; - capacity += id.addrsec[0]; - } - _lto4b(capacity - 1, rcd.addr); - _lto4b(512, rcd.length); + _lto4b(capacity - 1, rcd.addr); + _lto4b(512, rcd.length); - bcopy(&rcd, xs->data, MIN(sizeof(rcd), xs->datalen)); - xs->error = XS_NOERROR; - xs->flags |= ITSDONE; + bcopy(&rcd, xs->data, MIN(sizeof(rcd), xs->datalen)); + xs->error = XS_NOERROR; + break; + + case ATA_S_ERROR: + case ATA_S_TIMEOUT: + ata_free_identify(xa); + xs->error = (xa->state == ATA_S_TIMEOUT ? XS_TIMEOUT : + XS_DRIVER_STUFFUP); + break; + default: + panic("atascsi_disk_capacity_done: " + "unexpected ata_xfer state (%d)", xa->state); + } + + xs->flags |= ITSDONE; scsi_done(xs); } @@ -582,15 +580,7 @@ atascsi_atapi_cmd(struct scsi_xfer *xs) /* Copy SCSI command into ATAPI packet. */ memcpy(xa->cmd.packetcmd, xs->cmd, xs->cmdlen); - switch (ata_exec(as, xa)) { - case ATA_COMPLETE: - case ATA_ERROR: - return (COMPLETE); - case ATA_QUEUED: - return (SUCCESSFULLY_QUEUED); - default: - panic("unexpected return from ata_exec"); - } + return (ata_exec(as, xa)); } void @@ -648,10 +638,18 @@ atascsi_stuffup(struct scsi_xfer *xs) int ata_exec(struct atascsi *as, struct ata_xfer *xa) { - return (as->as_methods->ata_cmd(xa)); + switch (as->as_methods->ata_issue_cmd(xa)) { + case ATA_COMPLETE: + case ATA_ERROR: + return (COMPLETE); + case ATA_QUEUED: + if (!(xa->flags & ATA_F_POLL)) + return (SUCCESSFULLY_QUEUED); + default: + panic("unexpected return from ata_exec"); + } } - struct ata_xfer * ata_get_xfer(struct ata_port *ap, int nosleep /* XXX unused */) { |