In atapi_wdc.c, issue a 'REQUEST SENSE' command when appropriate.

Return XS_SENSE when the full sense info has been retrieved, or
XS_SHORTSENSE if only the sense key was available (from the error register)
Make atapi_interpret_sense() deal with this, and call scsipi_interpret_sense()
for XS_SENSE. (XXX sd_interpret_sense() and the ioctl code needs to be made
aware of XS_SHORTSENSE too ! sense hanlding for these is now less broken for
devices that support 'REQUEST SENSE')
All the ATAPI devices I have access to seems to honnor the SENSE_REQUEST
command, but I suspect some ATAPI devices will not (althouh it's mandatory).
The code should be able to deal with this, but is untested ...
This commit is contained in:
bouyer 1998-11-17 14:45:39 +00:00
parent 00d93f776b
commit 0d0ff884e5
2 changed files with 67 additions and 24 deletions

View File

@ -1,4 +1,4 @@
/* $NetBSD: atapi_base.c,v 1.10 1998/10/12 16:09:23 bouyer Exp $ */
/* $NetBSD: atapi_base.c,v 1.11 1998/11/17 14:45:39 bouyer Exp $ */
/*-
* Copyright (c) 1998 The NetBSD Foundation, Inc.
@ -67,7 +67,6 @@ atapi_interpret_sense(xs)
struct scsipi_link *sc_link = xs->sc_link;
char *msg = NULL;
key = (xs->sense.atapi_sense & 0xf0) >> 4;
/*
* If the device has it's own error handler, call it first.
* If it returns a legit error value, return that, otherwise
@ -80,7 +79,14 @@ atapi_interpret_sense(xs)
if (error != SCSIRET_CONTINUE)
return (error); /* error >= 0 better ? */
}
/* otherwise use the default */
/*
* otherwise use the default, call the generic sense handler if we have
* more than the sense key
*/
if (xs->error == XS_SENSE)
return (scsipi_interpret_sense(xs));
key = (xs->sense.atapi_sense & 0xf0) >> 4;
switch (key) {
case SKEY_RECOVERED_ERROR:
msg = "soft error (corrected)";

View File

@ -1,4 +1,4 @@
/* $NetBSD: atapi_wdc.c,v 1.7 1998/10/19 12:28:03 bouyer Exp $ */
/* $NetBSD: atapi_wdc.c,v 1.8 1998/11/17 14:45:39 bouyer Exp $ */
/*
* Copyright (c) 1998 Manuel Bouyer.
@ -310,6 +310,10 @@ wdc_atapi_intr(chp, xfer)
int len, phase, i, retries=0;
int ire, dma_err = 0;
int dma_flags = 0;
struct scsipi_generic _cmd_reqsense;
struct scsipi_sense *cmd_reqsense =
(struct scsipi_sense *)&_cmd_reqsense;
void *cmd;
WDCDEBUG_PRINT(("wdc_atapi_intr %s:%d:%d\n",
chp->wdc->sc_dev.dv_xname, chp->channel, drvp->drive), DEBUG_INTR);
@ -332,13 +336,17 @@ wdc_atapi_intr(chp, xfer)
wdc_atapi_reset(chp, xfer);
return 1;
}
/* Error here only if the command is aborted */
if ((chp->ch_status & WDCS_ERR) != 0 &&
/*
* if the request sense command was aborted, report the short sense
* previously recorded, else continue normal processing
*/
if ((xfer->c_flags & C_SENSE) != 0 &&
(chp->ch_status & WDCS_ERR) != 0 &&
(chp->ch_error & WDCE_ABRT) != 0) {
sc_xfer->error = XS_SENSE;
sc_xfer->sense.atapi_sense = chp->ch_error;
WDCDEBUG_PRINT(("wdc_atapi_intr: wdc_atapi_done(), "
"sense 0x%x\n", sc_xfer->sense.atapi_sense), DEBUG_INTR);
WDCDEBUG_PRINT(("wdc_atapi_intr: request_sense aborted, "
"calling wdc_atapi_done(), sense 0x%x\n",
sc_xfer->sense.atapi_sense), DEBUG_INTR);
wdc_atapi_done(chp, xfer);
return 1;
}
@ -359,12 +367,20 @@ again:
switch (phase) {
case PHASE_CMDOUT:
if (xfer->c_flags & C_SENSE) {
memset(cmd_reqsense, 0, sizeof(struct scsipi_generic));
cmd_reqsense->opcode = REQUEST_SENSE;
cmd_reqsense->length = xfer->c_bcount;
cmd = cmd_reqsense;
} else {
cmd = sc_xfer->cmd;
}
WDCDEBUG_PRINT(("PHASE_CMDOUT\n"), DEBUG_INTR);
/* Init the DMA channel if necessary */
if (xfer->c_flags & C_DMA) {
if ((*chp->wdc->dma_init)(chp->wdc->dma_arg,
chp->channel, xfer->drive,
xfer->databuf, sc_xfer->datalen, dma_flags) != 0) {
xfer->databuf, xfer->c_bcount, dma_flags) != 0) {
sc_xfer->error = XS_DRIVER_STUFFUP;
break;
}
@ -375,24 +391,24 @@ again:
if (drvp->drive_flags & DRIVE_CAP32) {
bus_space_write_multi_4(chp->data32iot,
chp->data32ioh, 0,
(u_int32_t *)sc_xfer->cmd,
(u_int32_t *)cmd,
sc_xfer->cmdlen >> 2);
} else {
bus_space_write_multi_2(chp->cmd_iot,
chp->cmd_ioh, wd_data,
(u_int16_t *)sc_xfer->cmd,
(u_int16_t *)cmd,
sc_xfer->cmdlen >> 1);
}
} else {
if (drvp->drive_flags & DRIVE_CAP32) {
bus_space_write_multi_stream_4(chp->data32iot,
chp->data32ioh, 0,
(u_int32_t *)sc_xfer->cmd,
(u_int32_t *)cmd,
sc_xfer->cmdlen >> 2);
} else {
bus_space_write_multi_stream_2(chp->cmd_iot,
chp->cmd_ioh, wd_data,
(u_int16_t *)sc_xfer->cmd,
(u_int16_t *)cmd,
sc_xfer->cmdlen >> 1);
}
}
@ -478,7 +494,8 @@ again:
case PHASE_DATAIN:
/* Read data */
WDCDEBUG_PRINT(("PHASE_DATAIN\n"), DEBUG_INTR);
if ((sc_xfer->flags & SCSI_DATA_IN) == 0 ||
if (((sc_xfer->flags & SCSI_DATA_IN) == 0 &&
(xfer->c_flags & C_SENSE) == 0) ||
(xfer->c_flags & C_DMA) != 0) {
printf("wdc_atapi_intr: bad data phase DATAIN");
if (xfer->c_flags & C_DMA) {
@ -547,13 +564,33 @@ again:
chp->channel, xfer->drive, dma_flags);
xfer->c_bcount -= sc_xfer->datalen;
}
if (chp->ch_status & WDCS_ERR) {
if (xfer->c_flags & C_SENSE) {
if ((chp->ch_status & WDCS_ERR) || dma_err < 0) {
/*
* request sense failed ! it's not suppossed
* to be possible
*/
sc_xfer->error = XS_DRIVER_STUFFUP;
} else {
/* use the sense we just read */
sc_xfer->error = XS_SENSE;
}
} else {
if (chp->ch_status & WDCS_ERR) {
/* save the short sense */
sc_xfer->error = XS_SHORTSENSE;
sc_xfer->sense.atapi_sense = chp->ch_error;
/* let the driver issue a 'request sense' */
xfer->databuf = &sc_xfer->sense;
xfer->c_bcount =
sizeof(sc_xfer->sense.scsi_sense);
xfer->c_flags |= C_SENSE;
wdc_atapi_start(chp, xfer);
return 1;
} else if (dma_err < 0) {
sc_xfer->error = XS_DRIVER_STUFFUP;
}
}
if (xfer->c_bcount != 0) {
WDCDEBUG_PRINT(("wdc_atapi_intr: bcount value is "
"%d after io\n", xfer->c_bcount), DEBUG_XFERS);
@ -577,7 +614,7 @@ again:
}
printf("wdc_atapi_intr: unknown phase 0x%x\n", phase);
if (chp->ch_status & WDCS_ERR) {
sc_xfer->error = XS_SENSE;
sc_xfer->error = XS_SHORTSENSE;
sc_xfer->sense.atapi_sense = chp->ch_error;
} else {
sc_xfer->error = XS_DRIVER_STUFFUP;
@ -676,7 +713,7 @@ error:
chp->wdc->sc_dev.dv_xname, chp->channel, xfer->drive,
errstring);
printf("error (0x%x)\n", chp->ch_error);
sc_xfer->error = XS_SENSE;
sc_xfer->error = XS_SHORTSENSE;
sc_xfer->sense.atapi_sense = chp->ch_error;
wdc_atapi_reset(chp, xfer);
return 1;