mirror of
https://github.com/freebsd/freebsd-src
synced 2024-09-23 18:25:19 +00:00
Added "scsi target" device that can act as a target for scsi transfers
from an initiator Added Julian's support for residuals. Added Julian's fixes to the tape driver Made compile cleanly with -Wall Reduce boot up output
This commit is contained in:
parent
c98fda6ec8
commit
81e01d44ce
Notes:
svn2git
2020-12-20 02:59:44 +00:00
svn path=/head/; revision=7816
|
@ -14,7 +14,7 @@
|
|||
*
|
||||
* Ported to run under 386BSD by Julian Elischer (julian@tfs.com) Sept 1992
|
||||
*
|
||||
* $Id: cd.c,v 1.36 1995/03/15 14:22:03 dufault Exp $
|
||||
* $Id: cd.c,v 1.37 1995/03/21 11:21:00 dufault Exp $
|
||||
*/
|
||||
|
||||
#define SPLCD splbio
|
||||
|
@ -183,6 +183,12 @@ cdattach(struct scsi_link *sc_link)
|
|||
* Use the subdriver to request information regarding
|
||||
* the drive. We cannot use interrupts yet, so the
|
||||
* request must specify this.
|
||||
*
|
||||
* XXX dufault@hda.com:
|
||||
* Need to handle this better in the case of no record. Rather than
|
||||
* a state driven sense handler I think we should make it so that
|
||||
* the command can get the sense back so that it can selectively log
|
||||
* errors.
|
||||
*/
|
||||
cd_get_parms(unit, SCSI_NOSLEEP | SCSI_NOMASK);
|
||||
if (dp->disksize) {
|
||||
|
@ -190,8 +196,9 @@ cdattach(struct scsi_link *sc_link)
|
|||
cd->params.disksize,
|
||||
cd->params.blksize);
|
||||
} else {
|
||||
printf("drive empty");
|
||||
printf("can't get the size\n");
|
||||
}
|
||||
|
||||
cd->flags |= CDINIT;
|
||||
cd_registerdev(unit);
|
||||
|
||||
|
@ -220,7 +227,7 @@ struct scsi_link *sc_link)
|
|||
return (ENXIO);
|
||||
|
||||
SC_DEBUG(sc_link, SDEV_DB1,
|
||||
("cd_open: dev=0x%x (unit %d,partition %d)\n",
|
||||
("cd_open: dev=0x%lx (unit %ld,partition %ld)\n",
|
||||
dev, unit, part));
|
||||
/*
|
||||
* Check that it is still responding and ok.
|
||||
|
@ -278,7 +285,7 @@ struct scsi_link *sc_link)
|
|||
* Check that the partition CAN exist
|
||||
*/
|
||||
if (part >= cd->disklabel.d_npartitions) {
|
||||
SC_DEBUG(sc_link, SDEV_DB3, ("partition %d > %d\n", part
|
||||
SC_DEBUG(sc_link, SDEV_DB3, ("partition %ld > %d\n", part
|
||||
,cd->disklabel.d_npartitions));
|
||||
errcode = ENXIO;
|
||||
goto bad;
|
||||
|
@ -288,7 +295,7 @@ struct scsi_link *sc_link)
|
|||
*/
|
||||
if (cd->disklabel.d_partitions[part].p_fstype == FS_UNUSED) {
|
||||
SC_DEBUG(sc_link, SDEV_DB3,
|
||||
("part %d type UNUSED\n", part));
|
||||
("part %ld type UNUSED\n", part));
|
||||
errcode = ENXIO;
|
||||
goto bad;
|
||||
}
|
||||
|
@ -326,7 +333,7 @@ cd_close(dev_t dev, int flag, int fmt, struct proc *p,
|
|||
part = PARTITION(dev);
|
||||
cd = sc_link->sd;
|
||||
|
||||
SC_DEBUG(sc_link, SDEV_DB2, ("cd%ld: closing part %d\n", unit, part));
|
||||
SC_DEBUG(sc_link, SDEV_DB2, ("cd%d: closing part %d\n", unit, part));
|
||||
cd->partflags[part] &= ~CDOPEN;
|
||||
cd->openparts &= ~(1 << part);
|
||||
|
||||
|
@ -449,7 +456,7 @@ cdstart(unit)
|
|||
struct scsi_link *sc_link = SCSI_LINK(&cd_switch, unit);
|
||||
struct scsi_data *cd = sc_link->sd;
|
||||
|
||||
SC_DEBUG(sc_link, SDEV_DB2, ("cdstart%d ", unit));
|
||||
SC_DEBUG(sc_link, SDEV_DB2, ("cdstart%ld ", unit));
|
||||
/*
|
||||
* See if there is a buf to do and we are not already
|
||||
* doing one
|
||||
|
@ -961,7 +968,6 @@ cd_size(unit, flags)
|
|||
20000, /* might be a disk-changer */
|
||||
NULL,
|
||||
SCSI_DATA_IN | flags) != 0) {
|
||||
printf("cd%d: could not get size\n", unit);
|
||||
return (0);
|
||||
} else {
|
||||
size = rdcap.addr_0 + 1;
|
||||
|
@ -977,7 +983,7 @@ cd_size(unit, flags)
|
|||
blksize = 2048; /* some drives lie ! */
|
||||
if (size < 100)
|
||||
size = 400000; /* ditto */
|
||||
SC_DEBUG(sc_link, SDEV_DB3, ("cd%ld: %d %d byte blocks\n"
|
||||
SC_DEBUG(sc_link, SDEV_DB3, ("cd%d: %ld %ld byte blocks\n"
|
||||
,unit, size, blksize));
|
||||
cd->params.disksize = size;
|
||||
cd->params.blksize = blksize;
|
||||
|
@ -1183,15 +1189,7 @@ errval
|
|||
cd_reset(unit)
|
||||
u_int32 unit;
|
||||
{
|
||||
return (scsi_scsi_cmd(SCSI_LINK(&cd_switch, unit),
|
||||
0,
|
||||
0,
|
||||
0,
|
||||
0,
|
||||
CDRETRIES,
|
||||
2000,
|
||||
NULL,
|
||||
SCSI_RESET));
|
||||
return scsi_reset_target(SCSI_LINK(&cd_switch, unit));
|
||||
}
|
||||
|
||||
/*
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
* Written by grefen@?????
|
||||
* Based on scsi drivers by Julian Elischer (julian@tfs.com)
|
||||
*
|
||||
* $Id: ch.c,v 1.16 1995/03/15 14:22:04 dufault Exp $
|
||||
* $Id: ch.c,v 1.17 1995/03/28 07:57:22 bde Exp $
|
||||
*/
|
||||
|
||||
#include <sys/types.h>
|
||||
|
@ -128,8 +128,7 @@ ch_registerdev(int unit)
|
|||
errval
|
||||
chattach(struct scsi_link *sc_link)
|
||||
{
|
||||
u_int32 unit, i;
|
||||
unsigned char *tbl;
|
||||
u_int32 unit;
|
||||
|
||||
struct scsi_data *ch = sc_link->sd;
|
||||
|
||||
|
@ -170,7 +169,7 @@ struct scsi_link *sc_link)
|
|||
* Only allow one at a time
|
||||
*/
|
||||
if (cd->flags & CH_OPEN) {
|
||||
printf("ch%d: already open\n", unit);
|
||||
printf("ch%ld: already open\n", unit);
|
||||
return EBUSY;
|
||||
}
|
||||
/*
|
||||
|
@ -182,15 +181,15 @@ struct scsi_link *sc_link)
|
|||
/*
|
||||
* Check that it is still responding and ok.
|
||||
*/
|
||||
if (errcode = (scsi_test_unit_ready(sc_link, 0))) {
|
||||
printf("ch%d: not ready\n", unit);
|
||||
if ( (errcode = (scsi_test_unit_ready(sc_link, 0))) ) {
|
||||
printf("ch%ld: not ready\n", unit);
|
||||
sc_link->flags &= ~SDEV_OPEN;
|
||||
return errcode;
|
||||
}
|
||||
/*
|
||||
* Make sure data is loaded
|
||||
*/
|
||||
if (errcode = (ch_mode_sense(unit, SCSI_NOSLEEP | SCSI_NOMASK))) {
|
||||
if ( (errcode = (ch_mode_sense(unit, SCSI_NOSLEEP | SCSI_NOMASK))) ) {
|
||||
printf("ch%d: scsi changer :- offline\n", unit);
|
||||
sc_link->flags &= ~SDEV_OPEN;
|
||||
return (errcode);
|
||||
|
@ -221,12 +220,8 @@ ch_ioctl(dev_t dev, int cmd, caddr_t arg, int mode,
|
|||
struct proc *p, struct scsi_link *sc_link)
|
||||
{
|
||||
/* struct ch_cmd_buf *args; */
|
||||
union scsi_cmd *scsi_cmd;
|
||||
register i, j;
|
||||
u_int32 opri;
|
||||
errval errcode = 0;
|
||||
unsigned char unit;
|
||||
u_int32 number, flags;
|
||||
u_int32 flags;
|
||||
errval ret;
|
||||
struct scsi_data *cd;
|
||||
|
||||
|
@ -438,7 +433,7 @@ ch_mode_sense(unit, flags)
|
|||
/*
|
||||
* Read in the pages
|
||||
*/
|
||||
if (errcode = scsi_scsi_cmd(sc_link,
|
||||
if ( (errcode = scsi_scsi_cmd(sc_link,
|
||||
(struct scsi_generic *) &scsi_cmd,
|
||||
sizeof(struct scsi_mode_sense),
|
||||
(u_char *) & scsi_sense,
|
||||
|
@ -446,9 +441,9 @@ ch_mode_sense(unit, flags)
|
|||
CHRETRIES,
|
||||
5000,
|
||||
NULL,
|
||||
flags | SCSI_DATA_IN) != 0) {
|
||||
flags | SCSI_DATA_IN) != 0) ) {
|
||||
if (!(flags & SCSI_SILENT))
|
||||
printf("ch%d: could not mode sense\n", unit);
|
||||
printf("ch%ld: could not mode sense\n", unit);
|
||||
return (errcode);
|
||||
}
|
||||
sc_link->flags |= SDEV_MEDIA_LOADED;
|
||||
|
|
|
@ -37,7 +37,7 @@
|
|||
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||||
* SUCH DAMAGE.
|
||||
*
|
||||
* $Id:$
|
||||
* $Id: pt.c,v 1.1 1995/03/04 20:50:46 dufault Exp $
|
||||
*/
|
||||
|
||||
/*
|
||||
|
@ -49,6 +49,7 @@
|
|||
#include <sys/param.h>
|
||||
#include <sys/systm.h>
|
||||
#include <sys/buf.h>
|
||||
#include <sys/proc.h>
|
||||
#include <scsi/scsi_all.h>
|
||||
#include <scsi/scsiconf.h>
|
||||
|
||||
|
@ -58,12 +59,13 @@ struct scsi_data {
|
|||
|
||||
void ptstart(u_int32 unit);
|
||||
void pt_strategy(struct buf *bp, struct scsi_link *sc_link);
|
||||
int pt_sense(struct scsi_xfer *scsi_xfer);
|
||||
|
||||
SCSI_DEVICE_ENTRIES(pt)
|
||||
|
||||
struct scsi_device pt_switch =
|
||||
{
|
||||
NULL,
|
||||
pt_sense,
|
||||
ptstart, /* we have a queue, and this is how we service it */
|
||||
NULL,
|
||||
NULL,
|
||||
|
@ -155,13 +157,12 @@ ptstart(unit)
|
|||
sizeof(cmd),
|
||||
(u_char *) bp->b_un.b_addr,
|
||||
bp->b_bcount,
|
||||
0, /* can't retry a read on a tape really */
|
||||
100000,
|
||||
0,
|
||||
10000,
|
||||
bp,
|
||||
flags | SCSI_NOSLEEP) == SUCCESSFULLY_QUEUED) {
|
||||
} else {
|
||||
badnews:
|
||||
printf("pt%d: oops not queued\n", unit);
|
||||
printf("pt%ld: oops not queued\n", unit);
|
||||
bp->b_flags |= B_ERROR;
|
||||
bp->b_error = EIO;
|
||||
biodone(bp);
|
||||
|
@ -212,3 +213,42 @@ pt_strategy(struct buf *bp, struct scsi_link *sc_link)
|
|||
splx(opri);
|
||||
return;
|
||||
}
|
||||
|
||||
/*
|
||||
* sense handler: Called to determine what to do when the
|
||||
* device returns a CHECK CONDITION.
|
||||
*
|
||||
* For the processor type devices we try to handle the "info" field.
|
||||
*/
|
||||
|
||||
int pt_sense(struct scsi_xfer *xs)
|
||||
{
|
||||
struct scsi_sense_data *sense = &(xs->sense);
|
||||
struct buf *bp;
|
||||
|
||||
long resid;
|
||||
|
||||
if ((sense->error_code & SSD_ERRCODE_VALID) == 0 ||
|
||||
(sense->ext.extended.flags & SSD_ILI) == 0) {
|
||||
return SCSIRET_CONTINUE; /* let the default handler handle it */
|
||||
}
|
||||
|
||||
resid = ntohl(*((int32 *) sense->ext.extended.info));
|
||||
|
||||
bp = xs->bp;
|
||||
|
||||
if (resid < 0) {
|
||||
/* It synthesized data in order to fill our request.
|
||||
* Move resid back to cover this.
|
||||
*/
|
||||
xs->resid = -resid;
|
||||
xs->flags |= SCSI_RESID_VALID;
|
||||
return 0;
|
||||
}
|
||||
else {
|
||||
/* It wanted to send more data. We can't really do anything
|
||||
* about this.
|
||||
*/
|
||||
return SCSIRET_CONTINUE;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -14,7 +14,7 @@
|
|||
*
|
||||
* Ported to run under 386BSD by Julian Elischer (julian@tfs.com) Sept 1992
|
||||
*
|
||||
* $Id: scsi_all.h,v 1.10 1995/03/01 22:24:41 dufault Exp $
|
||||
* $Id: scsi_all.h,v 1.11 1995/03/04 12:36:55 bde Exp $
|
||||
*/
|
||||
|
||||
/*
|
||||
|
@ -207,8 +207,9 @@ struct scsi_changedef
|
|||
#define T_COMM 9
|
||||
#define T_ASC0 10
|
||||
#define T_ASC1 11
|
||||
#define T_UNKNOWN 12
|
||||
#define T_NTYPES 13
|
||||
#define T_TARGET 12
|
||||
#define T_UNKNOWN 13
|
||||
#define T_NTYPES 14
|
||||
|
||||
#define T_NODEVICE 0x1F
|
||||
|
||||
|
|
|
@ -17,6 +17,7 @@
|
|||
#include <sys/param.h>
|
||||
#include <sys/systm.h>
|
||||
#include <sys/proc.h>
|
||||
#include <sys/kernel.h>
|
||||
#include <sys/buf.h>
|
||||
#include <sys/uio.h>
|
||||
#include <sys/malloc.h>
|
||||
|
@ -61,8 +62,9 @@ get_xs(sc_link, flags)
|
|||
sc_link->flags |= SDEV_WAITING;
|
||||
tsleep((caddr_t)sc_link, PRIBIO, "scsiget", 0);
|
||||
}
|
||||
sc_link->active++;
|
||||
sc_link->opennings--;
|
||||
if (xs = next_free_xs) {
|
||||
if ( (xs = next_free_xs) ) {
|
||||
next_free_xs = xs->next;
|
||||
splx(s);
|
||||
} else {
|
||||
|
@ -97,6 +99,7 @@ free_xs(xs, sc_link, flags)
|
|||
|
||||
SC_DEBUG(sc_link, SDEV_DB3, ("free_xs\n"));
|
||||
/* if was 0 and someone waits, wake them up */
|
||||
sc_link->active--;
|
||||
if ((!sc_link->opennings++) && (sc_link->flags & SDEV_WAITING)) {
|
||||
sc_link->flags &= ~SDEV_WAITING;
|
||||
wakeup((caddr_t)sc_link); /* remember, it wakes them ALL up */
|
||||
|
@ -156,6 +159,42 @@ scsi_read_capacity(sc_link, blk_size, flags)
|
|||
return (size);
|
||||
}
|
||||
|
||||
errval
|
||||
scsi_reset_target(sc_link)
|
||||
struct scsi_link *sc_link;
|
||||
{
|
||||
return (scsi_scsi_cmd(sc_link,
|
||||
0,
|
||||
0,
|
||||
0,
|
||||
0,
|
||||
1,
|
||||
2000,
|
||||
NULL,
|
||||
SCSI_RESET));
|
||||
}
|
||||
|
||||
errval
|
||||
scsi_target_mode(sc_link, on_off)
|
||||
struct scsi_link *sc_link;
|
||||
int on_off;
|
||||
{
|
||||
struct scsi_generic scsi_cmd;
|
||||
bzero(&scsi_cmd, sizeof(scsi_cmd));
|
||||
scsi_cmd.opcode = SCSI_OP_TARGET;
|
||||
scsi_cmd.bytes[0] = (on_off) ? 1 : 0;
|
||||
|
||||
return (scsi_scsi_cmd(sc_link,
|
||||
&scsi_cmd,
|
||||
sizeof(scsi_cmd),
|
||||
0,
|
||||
0,
|
||||
1,
|
||||
2000,
|
||||
NULL,
|
||||
SCSI_ESCAPE));
|
||||
}
|
||||
|
||||
/*
|
||||
* Get scsi driver to send a "are you ready?" command
|
||||
*/
|
||||
|
@ -354,8 +393,11 @@ scsi_done(xs)
|
|||
free_xs(xs, sc_link, SCSI_NOSLEEP); /*XXX */
|
||||
return; /* it did it all, finish up */
|
||||
}
|
||||
/* BUG: This isn't used anywhere. Do you have plans for it,
|
||||
/* XXX: This isn't used anywhere. Do you have plans for it,
|
||||
* Julian? (dufault@hda.com).
|
||||
* This allows a private 'done' handler to
|
||||
* resubmit the command if it wants to retry,
|
||||
* In this case the xs must NOT be freed. (julian)
|
||||
*/
|
||||
if (retval == -2) {
|
||||
return; /* it did it all, finish up */
|
||||
|
@ -373,15 +415,16 @@ scsi_done(xs)
|
|||
}
|
||||
/*
|
||||
* Go and handle errors now.
|
||||
* If it returns -1 then we should RETRY
|
||||
* If it returns SCSIRET_DO_RETRY then we should RETRY
|
||||
*/
|
||||
if ((retval = sc_err1(xs)) == -1) {
|
||||
if ((retval = sc_err1(xs)) == SCSIRET_DO_RETRY) {
|
||||
if ((*(sc_link->adapter->scsi_cmd)) (xs)
|
||||
== SUCCESSFULLY_QUEUED) { /* don't wake the job, ok? */
|
||||
return;
|
||||
}
|
||||
xs->flags |= ITSDONE;
|
||||
}
|
||||
|
||||
free_xs(xs, sc_link, SCSI_NOSLEEP); /* does a start if needed */
|
||||
biodone(bp);
|
||||
}
|
||||
|
@ -414,20 +457,23 @@ scsi_scsi_cmd(sc_link, scsi_cmd, cmdlen, data_addr, datalen,
|
|||
* Reject zero length commands and assert all defined commands
|
||||
* are the correct length.
|
||||
*/
|
||||
if (cmdlen == 0)
|
||||
return EFAULT;
|
||||
else
|
||||
if ((flags & (SCSI_RESET | SCSI_ESCAPE)) == 0)
|
||||
{
|
||||
static u_int8 sizes[] = {6, 10, 10, 0, 0, 12, 0, 0 };
|
||||
u_int8 size = sizes[((scsi_cmd->opcode) >> 5)];
|
||||
if (size && (size != cmdlen))
|
||||
return EIO;
|
||||
if (cmdlen == 0)
|
||||
return EFAULT;
|
||||
else
|
||||
{
|
||||
static u_int8 sizes[] = {6, 10, 10, 0, 0, 12, 0, 0 };
|
||||
u_int8 size = sizes[((scsi_cmd->opcode) >> 5)];
|
||||
if (size && (size != cmdlen))
|
||||
return EIO;
|
||||
}
|
||||
}
|
||||
|
||||
if (bp && !(flags & SCSI_USER)) flags |= SCSI_NOSLEEP;
|
||||
SC_DEBUG(sc_link, SDEV_DB2, ("scsi_cmd\n"));
|
||||
|
||||
xs = get_xs(sc_link, flags); /* should wait unless booting */
|
||||
xs = get_xs(sc_link, flags);
|
||||
if (!xs) return (ENOMEM);
|
||||
/*
|
||||
* Fill out the scsi_xfer structure. We don't know whose context
|
||||
|
@ -442,7 +488,7 @@ scsi_scsi_cmd(sc_link, scsi_cmd, cmdlen, data_addr, datalen,
|
|||
xs->cmdlen = cmdlen;
|
||||
xs->data = data_addr;
|
||||
xs->datalen = datalen;
|
||||
xs->resid = datalen;
|
||||
xs->resid = 0;
|
||||
xs->bp = bp;
|
||||
/*XXX*/ /*use constant not magic number */
|
||||
if (datalen && ((caddr_t) data_addr < (caddr_t) KERNBASE)) {
|
||||
|
@ -503,8 +549,9 @@ scsi_scsi_cmd(sc_link, scsi_cmd, cmdlen, data_addr, datalen,
|
|||
|
||||
switch (retval) {
|
||||
case SUCCESSFULLY_QUEUED:
|
||||
if (bp)
|
||||
return retval; /* will sleep (or not) elsewhere */
|
||||
if (bp) {
|
||||
return 0; /* will sleep (or not) elsewhere */
|
||||
}
|
||||
s = splbio();
|
||||
while (!(xs->flags & ITSDONE)) {
|
||||
tsleep((caddr_t)xs, PRIBIO + 1, "scsicmd", 0);
|
||||
|
@ -514,7 +561,7 @@ scsi_scsi_cmd(sc_link, scsi_cmd, cmdlen, data_addr, datalen,
|
|||
case COMPLETE: /* Polling command completed ok */
|
||||
/*XXX*/ case HAD_ERROR: /* Polling command completed with error */
|
||||
SC_DEBUG(sc_link, SDEV_DB3, ("back in cmd()\n"));
|
||||
if ((retval = sc_err1(xs)) == -1)
|
||||
if ((retval = sc_err1(xs)) == SCSIRET_DO_RETRY)
|
||||
goto retry;
|
||||
break;
|
||||
|
||||
|
@ -562,85 +609,168 @@ scsi_scsi_cmd(sc_link, scsi_cmd, cmdlen, data_addr, datalen,
|
|||
return (retval);
|
||||
}
|
||||
|
||||
static errval
|
||||
sc_err1(xs)
|
||||
struct scsi_xfer *xs;
|
||||
static errval
|
||||
sc_done(struct scsi_xfer *xs, int code)
|
||||
{
|
||||
struct buf *bp = xs->bp;
|
||||
errval retval;
|
||||
|
||||
SC_DEBUG(xs->sc_link, SDEV_DB3, ("sc_err1,err = 0x%x \n", xs->error));
|
||||
/*
|
||||
* If it has a buf, we might be working with
|
||||
* a request from the buffer cache or some other
|
||||
* piece of code that requires us to process
|
||||
* errors at inetrrupt time. We have probably
|
||||
* errors at interrupt time. We have probably
|
||||
* been called by scsi_done()
|
||||
*/
|
||||
switch ((int)xs->error) {
|
||||
case XS_NOERROR: /* nearly always hit this one */
|
||||
retval = ESUCCESS;
|
||||
if (bp) {
|
||||
bp->b_error = 0;
|
||||
bp->b_resid = 0;
|
||||
}
|
||||
break;
|
||||
struct buf *bp;
|
||||
|
||||
case XS_SENSE:
|
||||
retval = scsi_interpret_sense(xs);
|
||||
if (retval == SCSIRET_DO_RETRY) {
|
||||
if (xs->retries--) {
|
||||
xs->error = XS_NOERROR;
|
||||
xs->flags &= ~ITSDONE;
|
||||
goto retry;
|
||||
}
|
||||
retval = EIO; /* Too many retries */
|
||||
}
|
||||
|
||||
if (bp) {
|
||||
bp->b_error = 0;
|
||||
if (retval) {
|
||||
bp->b_flags |= B_ERROR;
|
||||
bp->b_error = retval;
|
||||
bp->b_resid = bp->b_bcount;
|
||||
}
|
||||
SC_DEBUG(xs->sc_link, SDEV_DB3,
|
||||
("scsi_interpret_sense (bp) returned %d\n", retval));
|
||||
} else {
|
||||
SC_DEBUG(xs->sc_link, SDEV_DB3,
|
||||
("scsi_interpret_sense (no bp) returned %d\n", retval));
|
||||
}
|
||||
break;
|
||||
|
||||
case XS_BUSY:
|
||||
/*should somehow arange for a 1 sec delay here (how?) */
|
||||
/* XXX tsleep(&localvar, priority, "foo", hz);
|
||||
that's how! */
|
||||
case XS_TIMEOUT:
|
||||
/*
|
||||
* If we can, resubmit it to the adapter.
|
||||
*/
|
||||
if (code == SCSIRET_DO_RETRY) {
|
||||
if (xs->retries--) {
|
||||
xs->error = XS_NOERROR;
|
||||
xs->flags &= ~ITSDONE;
|
||||
goto retry;
|
||||
return SCSIRET_DO_RETRY;
|
||||
}
|
||||
code = EIO; /* Too many retries */
|
||||
}
|
||||
|
||||
/*
|
||||
* an EOF condition results in a VALID resid..
|
||||
*/
|
||||
if(xs->flags & SCSI_EOF) {
|
||||
xs->resid = xs->datalen;
|
||||
xs->flags |= SCSI_RESID_VALID;
|
||||
}
|
||||
|
||||
bp = xs->bp;
|
||||
if (code != ESUCCESS) {
|
||||
if (bp) {
|
||||
bp->b_error = 0;
|
||||
bp->b_flags |= B_ERROR;
|
||||
bp->b_error = code;
|
||||
bp->b_resid = bp->b_bcount;
|
||||
SC_DEBUG(xs->sc_link, SDEV_DB3,
|
||||
("scsi_interpret_sense (bp) returned %d\n", code));
|
||||
} else {
|
||||
SC_DEBUG(xs->sc_link, SDEV_DB3,
|
||||
("scsi_interpret_sense (no bp) returned %d\n", code));
|
||||
}
|
||||
}
|
||||
else {
|
||||
if (bp) {
|
||||
|
||||
bp->b_error = 0;
|
||||
|
||||
/* XXX: We really shouldn't need this SCSI_RESID_VALID flag.
|
||||
* If we initialize it to 0 and only touch it if we have
|
||||
* a value then we can leave out the test.
|
||||
*/
|
||||
|
||||
if (xs->flags & SCSI_RESID_VALID) {
|
||||
bp->b_resid = xs->resid;
|
||||
bp->b_flags |= B_ERROR;
|
||||
} else {
|
||||
bp->b_resid = 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return code;
|
||||
}
|
||||
|
||||
/*
|
||||
* submit a scsi command, given the command.. used for retries
|
||||
* and callable from timeout()
|
||||
*/
|
||||
#ifdef NOTYET
|
||||
errval scsi_submit(xs)
|
||||
struct scsi_xfer *xs;
|
||||
{
|
||||
struct scsi_link *sc_link = xs->sc_link;
|
||||
int retval;
|
||||
|
||||
retval = (*(sc_link->adapter->scsi_cmd)) (xs);
|
||||
|
||||
return retval;
|
||||
}
|
||||
|
||||
/*
|
||||
* Retry a scsi command, given the command, and a delay.
|
||||
*/
|
||||
errval scsi_retry(xs,delay)
|
||||
struct scsi_xfer *xs;
|
||||
int delay;
|
||||
{
|
||||
if(delay)
|
||||
{
|
||||
timeout(((void())*)scsi_submit,xs,hz*delay);
|
||||
return(0);
|
||||
}
|
||||
else
|
||||
{
|
||||
return(scsi_submit(xs));
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
/*
|
||||
* handle checking for errors..
|
||||
* called at interrupt time from scsi_done() and
|
||||
* at user time from scsi_scsi_cmd(), depending on whether
|
||||
* there was a bp (basically, if there is a bp, there may be no
|
||||
* associated process at the time. (it could be an async operation))
|
||||
* lower level routines shouldn't know about xs->bp.. we are the lowest.
|
||||
*/
|
||||
static errval
|
||||
sc_err1(xs)
|
||||
struct scsi_xfer *xs;
|
||||
{
|
||||
SC_DEBUG(xs->sc_link, SDEV_DB3, ("sc_err1,err = 0x%lx \n", xs->error));
|
||||
|
||||
switch ((int)xs->error) {
|
||||
case XS_SENSE:
|
||||
return sc_done(xs, scsi_interpret_sense(xs));
|
||||
|
||||
case XS_NOERROR:
|
||||
return sc_done(xs, ESUCCESS);
|
||||
|
||||
case XS_BUSY:
|
||||
/* should somehow arange for a 1 sec delay here (how?)[jre]
|
||||
* tsleep(&localvar, priority, "foo", hz);
|
||||
* that's how! [unknown]
|
||||
* no, we could be at interrupt context.. use
|
||||
* timeout(scsi_resubmit,xs,hz); [jre] (not implimenteed yet)
|
||||
*/
|
||||
case XS_TIMEOUT:
|
||||
return sc_done(xs, SCSIRET_DO_RETRY);
|
||||
|
||||
/* fall through */
|
||||
case XS_DRIVER_STUFFUP:
|
||||
if (bp) {
|
||||
bp->b_flags |= B_ERROR;
|
||||
bp->b_error = EIO;
|
||||
}
|
||||
retval = EIO;
|
||||
break;
|
||||
return sc_done(xs, EIO);
|
||||
|
||||
default:
|
||||
retval = EIO;
|
||||
sc_print_addr(xs->sc_link);
|
||||
printf("unknown error category from scsi driver\n");
|
||||
return sc_done(xs, EIO);
|
||||
}
|
||||
return retval;
|
||||
retry:
|
||||
return (-1);
|
||||
}
|
||||
|
||||
int
|
||||
scsi_sense_qualifiers(xs, asc, ascq)
|
||||
struct scsi_xfer *xs;
|
||||
int *asc;
|
||||
int *ascq;
|
||||
{
|
||||
struct scsi_sense_data_new *sense;
|
||||
struct scsi_sense_extended *ext;
|
||||
|
||||
sense = (struct scsi_sense_data_new *)&(xs->sense);
|
||||
|
||||
ext = &(sense->ext.extended);
|
||||
|
||||
if (ext->extra_len < 5)
|
||||
return 0;
|
||||
|
||||
*asc = (ext->extra_len >= 5) ? ext->add_sense_code : 0;
|
||||
*ascq = (ext->extra_len >= 6) ? ext->add_sense_code_qual : 0;
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
/*
|
||||
|
@ -655,7 +785,6 @@ void scsi_sense_print(xs)
|
|||
struct scsi_sense_extended *ext;
|
||||
u_int32 key;
|
||||
u_int32 info;
|
||||
errval errcode;
|
||||
int asc, ascq;
|
||||
|
||||
/* This sense key text now matches what is in the SCSI spec
|
||||
|
@ -675,7 +804,7 @@ void scsi_sense_print(xs)
|
|||
"MISCOMPARE", "RESERVED"
|
||||
};
|
||||
|
||||
sc_print_addr(xs->sc_link);
|
||||
sc_print_start(xs->sc_link);
|
||||
|
||||
sense = (struct scsi_sense_data_new *)&(xs->sense);
|
||||
ext = &(sense->ext.extended);
|
||||
|
@ -706,8 +835,14 @@ void scsi_sense_print(xs)
|
|||
info);
|
||||
break;
|
||||
default:
|
||||
if (info)
|
||||
printf(" info:%lx", info);
|
||||
if (info) {
|
||||
if (sense->ext.extended.flags & SSD_ILI) {
|
||||
printf(" ILI (length mismatch): %ld", info);
|
||||
}
|
||||
else {
|
||||
printf(" info:%lx", info);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
else if (info)
|
||||
|
@ -734,11 +869,11 @@ void scsi_sense_print(xs)
|
|||
if (strlen(desc) > 40)
|
||||
sc_print_addr(xs->sc_link);;
|
||||
|
||||
printf("%s", desc);
|
||||
printf(" %s", desc);
|
||||
}
|
||||
|
||||
if (ext->extra_len >= 7 && ext->fru) {
|
||||
printf(" fru:%x", ext->fru);
|
||||
printf(" field replaceable unit: %x", ext->fru);
|
||||
}
|
||||
|
||||
if (ext->extra_len >= 10 &&
|
||||
|
@ -764,6 +899,7 @@ void scsi_sense_print(xs)
|
|||
}
|
||||
|
||||
printf("\n");
|
||||
sc_print_finish();
|
||||
}
|
||||
|
||||
/*
|
||||
|
@ -781,10 +917,11 @@ scsi_interpret_sense(xs)
|
|||
u_int32 key;
|
||||
u_int32 silent;
|
||||
errval errcode;
|
||||
int error_code;
|
||||
|
||||
/*
|
||||
* If the flags say errs are ok, then always return ok.
|
||||
* BUG: What if it is a deferred error?
|
||||
* XXX: What if it is a deferred error?
|
||||
*/
|
||||
if (xs->flags & SCSI_ERR_OK)
|
||||
return (ESUCCESS);
|
||||
|
@ -823,20 +960,34 @@ scsi_interpret_sense(xs)
|
|||
* request a retry or continue with default sense handling.
|
||||
*/
|
||||
if (sc_link->device->err_handler) {
|
||||
SC_DEBUG(sc_link, SDEV_DB2, ("calling private err_handler()\n"));
|
||||
SC_DEBUG(sc_link, SDEV_DB2,
|
||||
("calling private err_handler()\n"));
|
||||
errcode = (*sc_link->device->err_handler) (xs);
|
||||
|
||||
if (errcode >= 0)
|
||||
SC_DEBUG(sc_link, SDEV_DB2,
|
||||
("private err_handler() returned %d\n",errcode));
|
||||
if (errcode >= 0) {
|
||||
SC_DEBUG(sc_link, SDEV_DB2,
|
||||
("SCSI_EOF = %d\n",(xs->flags & SCSI_EOF)?1:0));
|
||||
SC_DEBUG(sc_link, SDEV_DB2,
|
||||
("SCSI_RESID_VALID = %d\n",
|
||||
(xs->flags & SCSI_RESID_VALID)?1:0));
|
||||
|
||||
if(xs->flags & SCSI_EOF) {
|
||||
xs->resid = xs->datalen;
|
||||
xs->flags |= SCSI_RESID_VALID;
|
||||
}
|
||||
return errcode; /* valid errno value */
|
||||
}
|
||||
|
||||
switch(errcode) {
|
||||
case SCSIRET_DO_RETRY: /* Requested a retry */
|
||||
case SCSIRET_DO_RETRY: /* Requested a retry */
|
||||
return errcode;
|
||||
|
||||
case SCSIRET_CONTINUE: /* Continue with default sense processing */
|
||||
case SCSIRET_CONTINUE: /* Continue with default sense processing */
|
||||
break;
|
||||
|
||||
default:
|
||||
default:
|
||||
sc_print_addr(xs->sc_link);
|
||||
printf("unknown return code %d from sense handler.\n",
|
||||
errcode);
|
||||
|
@ -844,15 +995,17 @@ scsi_interpret_sense(xs)
|
|||
return errcode;
|
||||
}
|
||||
}
|
||||
|
||||
/* otherwise use the default */
|
||||
silent = (xs->flags & SCSI_SILENT);
|
||||
key = sense->ext.extended.flags & SSD_KEY;
|
||||
error_code = sense->error_code & SSD_ERRCODE;
|
||||
|
||||
if (!silent) {
|
||||
scsi_sense_print(xs);
|
||||
}
|
||||
|
||||
switch (sense->error_code & SSD_ERRCODE) {
|
||||
switch (error_code) {
|
||||
case 0x71: /* deferred error */
|
||||
/* Print even if silent (not silent was already done)
|
||||
*/
|
||||
|
@ -860,7 +1013,7 @@ scsi_interpret_sense(xs)
|
|||
scsi_sense_print(xs);
|
||||
}
|
||||
|
||||
/* BUG:
|
||||
/* XXX:
|
||||
* This error doesn't relate to the command associated
|
||||
* with this request sense. A deferred error is an error
|
||||
* for a command that has already returned GOOD status (see 7.2.14.2).
|
||||
|
@ -890,9 +1043,11 @@ scsi_interpret_sense(xs)
|
|||
switch ((int)key) {
|
||||
case 0x0: /* NO SENSE */
|
||||
case 0x1: /* RECOVERED ERROR */
|
||||
if (xs->resid == xs->datalen)
|
||||
xs->resid = 0; /* not short read */
|
||||
case 0xc: /* EQUAL */
|
||||
if(xs->flags & SCSI_EOF) {
|
||||
xs->resid = xs->datalen;
|
||||
xs->flags |= SCSI_RESID_VALID;
|
||||
}
|
||||
return (ESUCCESS);
|
||||
case 0x2: /* NOT READY */
|
||||
sc_link->flags &= ~SDEV_MEDIA_LOADED;
|
||||
|
@ -911,6 +1066,7 @@ scsi_interpret_sense(xs)
|
|||
case 0xd: /* VOLUME OVERFLOW */
|
||||
return (ENOSPC);
|
||||
case 0x8: /* BLANK CHECK */
|
||||
xs->flags |= SCSI_EOF; /* force EOF on tape read */
|
||||
return (ESUCCESS);
|
||||
default:
|
||||
return (EIO);
|
||||
|
@ -1014,16 +1170,38 @@ sc_print_start(sc_link)
|
|||
struct scsi_link *sc_link;
|
||||
{
|
||||
sc_print_addr(sc_link);
|
||||
sc_printing = 1;
|
||||
sc_printing++;
|
||||
}
|
||||
void
|
||||
sc_print_finish()
|
||||
{
|
||||
sc_printing = 0;
|
||||
sc_printing--;
|
||||
}
|
||||
|
||||
static void
|
||||
id_put(int id, char *after)
|
||||
{
|
||||
switch(id)
|
||||
{
|
||||
case SCCONF_UNSPEC:
|
||||
break;
|
||||
|
||||
case SCCONF_ANY:
|
||||
printf("?");
|
||||
break;
|
||||
|
||||
default:
|
||||
printf("%d", id);
|
||||
break;
|
||||
}
|
||||
|
||||
printf("%s", after);
|
||||
}
|
||||
|
||||
/*
|
||||
* Print out the scsi_link structure's address info.
|
||||
* sc_print_addr: Print out the scsi_link structure's address info.
|
||||
* This should handle any circumstance, even the transitory ones
|
||||
* during system configuration.
|
||||
*/
|
||||
|
||||
void
|
||||
|
@ -1033,12 +1211,26 @@ sc_print_addr(sc_link)
|
|||
if (sc_printing)
|
||||
printf("\n");
|
||||
|
||||
if (strcmp(sc_link->device->name, "probe") != 0)
|
||||
printf("%s%d", sc_link->device->name, sc_link->dev_unit);
|
||||
if (sc_link->device == 0) {
|
||||
printf("nodevice");
|
||||
}
|
||||
else if (strcmp(sc_link->device->name, "probe") != 0) {
|
||||
printf("%s", sc_link->device->name);
|
||||
id_put(sc_link->dev_unit, "");
|
||||
}
|
||||
|
||||
printf("(%s%d:%d:%d): ", sc_link->adapter->name, sc_link->adapter_unit,
|
||||
sc_link->target, sc_link->lun);
|
||||
if (sc_link->adapter == 0) {
|
||||
printf("(noadapter:");
|
||||
}
|
||||
else {
|
||||
printf("(%s", sc_link->adapter->name);
|
||||
id_put(sc_link->adapter_unit, ":");
|
||||
}
|
||||
|
||||
id_put(sc_link->target, ":");
|
||||
id_put(sc_link->lun, "): ");
|
||||
}
|
||||
|
||||
#ifdef SCSIDEBUG
|
||||
/*
|
||||
* Given a scsi_xfer, dump the request, in all it's glory
|
||||
|
@ -1047,18 +1239,18 @@ void
|
|||
show_scsi_xs(xs)
|
||||
struct scsi_xfer *xs;
|
||||
{
|
||||
printf("xs(0x%x): ", xs);
|
||||
printf("flg(0x%x)", xs->flags);
|
||||
printf("sc_link(0x%x)", xs->sc_link);
|
||||
printf("xs(%p): ", xs);
|
||||
printf("flg(0x%lx)", xs->flags);
|
||||
printf("sc_link(%p)", xs->sc_link);
|
||||
printf("retr(0x%x)", xs->retries);
|
||||
printf("timo(0x%x)", xs->timeout);
|
||||
printf("cmd(0x%x)", xs->cmd);
|
||||
printf("len(0x%x)", xs->cmdlen);
|
||||
printf("data(0x%x)", xs->data);
|
||||
printf("len(0x%x)", xs->datalen);
|
||||
printf("res(0x%x)", xs->resid);
|
||||
printf("err(0x%x)", xs->error);
|
||||
printf("bp(0x%x)", xs->bp);
|
||||
printf("timo(0x%lx)", xs->timeout);
|
||||
printf("cmd(%p)", xs->cmd);
|
||||
printf("len(0x%lx)", xs->cmdlen);
|
||||
printf("data(%p)", xs->data);
|
||||
printf("len(0x%lx)", xs->datalen);
|
||||
printf("res(0x%lx)", xs->resid);
|
||||
printf("err(0x%lx)", xs->error);
|
||||
printf("bp(%p)", xs->bp);
|
||||
show_scsi_cmd(xs);
|
||||
}
|
||||
|
||||
|
@ -1077,7 +1269,7 @@ show_scsi_cmd(struct scsi_xfer *xs)
|
|||
printf(",");
|
||||
printf("%x", b[i++]);
|
||||
}
|
||||
printf("-[%d bytes]\n", xs->datalen);
|
||||
printf("-[%ld bytes]\n", xs->datalen);
|
||||
if (xs->datalen)
|
||||
show_mem(xs->data, min(64, xs->datalen));
|
||||
} else {
|
||||
|
@ -1090,11 +1282,11 @@ show_mem(address, num)
|
|||
unsigned char *address;
|
||||
u_int32 num;
|
||||
{
|
||||
u_int32 x, y;
|
||||
u_int32 y;
|
||||
printf("------------------------------");
|
||||
for (y = 0; y < num; y += 1) {
|
||||
if (!(y % 16))
|
||||
printf("\n%03d: ", y);
|
||||
printf("\n%03ld: ", y);
|
||||
printf("%02x ", *address++);
|
||||
}
|
||||
printf("\n------------------------------\n");
|
||||
|
|
|
@ -35,7 +35,7 @@
|
|||
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||||
* SUCH DAMAGE.
|
||||
*
|
||||
* $Id: scsi_driver.c,v 1.6 1995/03/16 18:15:48 bde Exp $
|
||||
* $Id: scsi_driver.c,v 1.7 1995/03/21 11:21:02 dufault Exp $
|
||||
*
|
||||
*/
|
||||
#include <sys/types.h>
|
||||
|
@ -125,18 +125,19 @@ struct scsi_device *device)
|
|||
errcode = (device->dev_open) ?
|
||||
(*device->dev_open)(dev, flags, fmt, p, sc_link) : 0;
|
||||
|
||||
if (sc_link->flags & SDEV_ONCE_ONLY) {
|
||||
if (errcode == 0 && (sc_link->flags & SDEV_ONCE_ONLY)) {
|
||||
/*
|
||||
* Only allow one at a time
|
||||
*/
|
||||
if (sc_link->flags & SDEV_OPEN) {
|
||||
return EBUSY;
|
||||
errcode = EBUSY;
|
||||
}
|
||||
else {
|
||||
sc_link->flags |= SDEV_OPEN;
|
||||
}
|
||||
|
||||
sc_link->flags |= SDEV_OPEN;
|
||||
}
|
||||
|
||||
SC_DEBUG(sc_link, SDEV_DB1, ("%sopen: dev=0x%x (unit %d) result %d\n",
|
||||
SC_DEBUG(sc_link, SDEV_DB1, ("%sopen: dev=0x%lx (unit %ld) result %d\n",
|
||||
device->name, dev, unit, errcode));
|
||||
|
||||
return errcode;
|
||||
|
@ -192,12 +193,14 @@ scsi_strategy(struct buf *bp, struct scsi_device *device)
|
|||
struct scsi_link *sc_link = SCSI_LINK(device, unit);
|
||||
|
||||
SC_DEBUG(sc_link, SDEV_DB2, ("\n%sstrategy ", device->name));
|
||||
SC_DEBUG(sc_link, SDEV_DB1, ("%s%ld: %d bytes @ blk%d\n",
|
||||
device->name, unit, bp->b_bcount, bp->b_blkno));
|
||||
SC_DEBUG(sc_link, SDEV_DB1, ("%ld bytes @ blk%ld\n",
|
||||
bp->b_bcount, bp->b_blkno));
|
||||
|
||||
bp->b_resid = 0;
|
||||
bp->b_error = 0;
|
||||
|
||||
if (bp->b_bcount == 0)
|
||||
{
|
||||
bp->b_resid = 0;
|
||||
biodone(bp);
|
||||
}
|
||||
else if (device->dev_strategy)
|
||||
|
|
|
@ -35,12 +35,14 @@
|
|||
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||||
* SUCH DAMAGE.
|
||||
*
|
||||
* $Id: scsi_driver.h,v 1.1 1995/03/01 22:24:42 dufault Exp $
|
||||
* $Id: scsi_driver.h,v 1.2 1995/03/04 20:50:52 dufault Exp $
|
||||
*
|
||||
*/
|
||||
#ifndef _SCSI__DRIVER_H_
|
||||
#define _SCSI__DRIVER_H_
|
||||
|
||||
#ifdef KERNEL
|
||||
|
||||
#include <sys/conf.h>
|
||||
|
||||
struct kern_devconf;
|
||||
|
@ -59,4 +61,16 @@ int scsi_ioctl __P((dev_t, u_int32, caddr_t, int, struct proc *,
|
|||
void scsi_strategy __P((struct buf *, struct scsi_device *));
|
||||
void scsi_minphys __P((struct buf *, struct scsi_device *));
|
||||
|
||||
int suopen __P((dev_t, int, int, struct proc *));
|
||||
int suclose __P((dev_t, int, int, struct proc *));
|
||||
void sustrategy __P((struct buf *));
|
||||
int suioctl __P((dev_t, int, caddr_t, int, struct proc *));
|
||||
int sudump __P((dev_t));
|
||||
int supsize __P((dev_t));
|
||||
int suread __P((dev_t, struct uio *, int));
|
||||
int suwrite __P((dev_t, struct uio *, int));
|
||||
int suselect __P((dev_t, int, struct proc *));
|
||||
|
||||
#endif /* KERNEL */
|
||||
|
||||
#endif /* _SCSI__DRIVER_H_ */
|
||||
|
|
|
@ -38,7 +38,7 @@
|
|||
* SUCH DAMAGE.
|
||||
*End copyright
|
||||
*
|
||||
* $Id: scsi_ioctl.c,v 1.11 1995/03/01 22:24:42 dufault Exp $
|
||||
* $Id: scsi_ioctl.c,v 1.12 1995/03/04 20:50:55 dufault Exp $
|
||||
*
|
||||
*
|
||||
*/
|
||||
|
@ -97,7 +97,10 @@ struct scsi_xfer *xs;
|
|||
switch((int)xs->error) {
|
||||
case XS_NOERROR:
|
||||
SC_DEBUG(xs->sc_link,SDEV_DB3,("no error\n"));
|
||||
screq->datalen_used = xs->datalen - xs->resid; /* probably rubbish */
|
||||
if (xs->flags & SCSI_RESID_VALID)
|
||||
screq->datalen_used = xs->datalen - xs->resid;
|
||||
else
|
||||
screq->datalen_used = xs->datalen;
|
||||
screq->retsts = SCCMD_OK;
|
||||
break;
|
||||
|
||||
|
@ -283,7 +286,7 @@ struct proc *p, struct scsi_link *sc_link)
|
|||
if (len) {
|
||||
struct uio auio;
|
||||
struct iovec aiov;
|
||||
long cnt, error = 0;
|
||||
long cnt;
|
||||
|
||||
aiov.iov_base = d_addr;
|
||||
aiov.iov_len = len;
|
||||
|
|
|
@ -16,7 +16,7 @@
|
|||
*
|
||||
* New configuration setup: dufault@hda.com
|
||||
*
|
||||
* $Id: scsiconf.c,v 1.25 1995/03/19 14:29:06 davidg Exp $
|
||||
* $Id: scsiconf.c,v 1.26 1995/03/21 11:21:04 dufault Exp $
|
||||
*/
|
||||
|
||||
#include <sys/types.h>
|
||||
|
@ -37,6 +37,7 @@
|
|||
#include "ch.h"
|
||||
|
||||
#include "su.h"
|
||||
#include "sctarg.h"
|
||||
|
||||
#include <scsi/scsi_all.h>
|
||||
#include <scsi/scsiconf.h>
|
||||
|
@ -312,7 +313,6 @@ static struct scsidevs knowndevs[] =
|
|||
*/
|
||||
struct scsidevs *scsi_probedev();
|
||||
struct scsidevs *scsi_selectdev();
|
||||
errval scsi_probe_bus(int bus, int targ, int lun);
|
||||
|
||||
/* XXX dufault@hda.com
|
||||
* This scsi_device doesn't have the scsi_data_size.
|
||||
|
@ -364,11 +364,14 @@ static int next_free_type = T_NTYPES;
|
|||
* more than once - the list links are part of the structure. That is
|
||||
* prevented.
|
||||
*
|
||||
* Unusual devices should always be registered as type "-1". Then
|
||||
* Custom devices should always be registered as type "-1". Then
|
||||
* the next available type number will be allocated for it.
|
||||
*
|
||||
* Be careful not to register a type as 0 unless you really mean to
|
||||
* replace the disk driver.
|
||||
*
|
||||
* This is usually called only by the "device_init" function generated
|
||||
* automatically in the SCSI_DEVICE_ENTRIES macro.
|
||||
*/
|
||||
|
||||
void
|
||||
|
@ -438,7 +441,7 @@ scsi_init(void)
|
|||
|
||||
/* First call all type initialization functions.
|
||||
*/
|
||||
ukinit();
|
||||
ukinit(); /* We always have the unknown device. */
|
||||
|
||||
for (i = 0; scsi_tinit[i]; i++)
|
||||
(*scsi_tinit[i])();
|
||||
|
@ -472,12 +475,6 @@ scsi_init(void)
|
|||
}
|
||||
}
|
||||
|
||||
/* Feel free to take this out when everyone is sure this config
|
||||
* code works well. For now it lets us tell new configurations from
|
||||
* old ones.
|
||||
*/
|
||||
#define CONFIG_NOISE
|
||||
|
||||
/* scsi_bus_conf: Figure out which bus this is. If it is wired in config
|
||||
* use that. Otherwise use the next free one.
|
||||
*/
|
||||
|
@ -501,9 +498,9 @@ scsi_bus_conf(sc_link_proto)
|
|||
(sc_link_proto->adapter_unit == scsi_cinit[i].unit) )
|
||||
{
|
||||
bus = scsi_cinit[i].bus;
|
||||
#ifdef CONFIG_NOISE
|
||||
printf("Choosing drivers for scbus configured at %d\n", bus);
|
||||
#endif
|
||||
if (bootverbose)
|
||||
printf("Choosing drivers for scbus configured at %d\n",
|
||||
bus);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
@ -535,10 +532,9 @@ scsi_assign_unit(struct scsi_link *sc_link)
|
|||
sc_link->scsibus == scsi_dinit[i].cunit) {
|
||||
sc_link->dev_unit = scsi_dinit[i].unit;
|
||||
found = 1;
|
||||
#ifdef CONFIG_NOISE
|
||||
printf("%s is configured at %d\n",
|
||||
sc_link->device->name, sc_link->dev_unit);
|
||||
#endif
|
||||
if (bootverbose)
|
||||
printf("%s is configured at %d\n",
|
||||
sc_link->device->name, sc_link->dev_unit);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
@ -549,6 +545,47 @@ scsi_assign_unit(struct scsi_link *sc_link)
|
|||
return sc_link->dev_unit;
|
||||
}
|
||||
|
||||
|
||||
#if NSCTARG > 0
|
||||
/* The SCSI target configuration is simpler. If an entry is present
|
||||
* we just return the bus, target and lun for that unit.
|
||||
*/
|
||||
static void
|
||||
scsi_sctarg_lookup(char *name, int unit, int *target, int *lun, int *bus)
|
||||
{
|
||||
int i;
|
||||
|
||||
*bus = SCCONF_UNSPEC;
|
||||
*target = SCCONF_UNSPEC;
|
||||
*lun = SCCONF_UNSPEC;
|
||||
|
||||
for (i = 0; scsi_dinit[i].name; i++) {
|
||||
if ((strcmp(name, scsi_dinit[i].name) == 0) &&
|
||||
unit == scsi_dinit[i].unit)
|
||||
{
|
||||
*bus = scsi_dinit[i].cunit;
|
||||
*target = scsi_dinit[i].target;
|
||||
*lun = scsi_dinit[i].lun;
|
||||
}
|
||||
}
|
||||
}
|
||||
#endif /* NSCTARG > 0 */
|
||||
|
||||
void scsi_configure_start(void)
|
||||
{
|
||||
scsi_init();
|
||||
}
|
||||
|
||||
void scsi_configure_finish(void)
|
||||
{
|
||||
|
||||
static errval scsi_attach_sctarg __P((void));
|
||||
#if NSCTARG > 0
|
||||
scsi_attach_sctarg();
|
||||
#endif
|
||||
|
||||
}
|
||||
|
||||
/*
|
||||
* The routine called by the adapter boards to get all their
|
||||
* devices configured in.
|
||||
|
@ -560,8 +597,6 @@ scsi_attachdevs(sc_link_proto)
|
|||
int scsibus;
|
||||
struct scsibus_data *scbus;
|
||||
|
||||
scsi_init();
|
||||
|
||||
if ( (scsibus = scsi_bus_conf(sc_link_proto)) == -1) {
|
||||
return;
|
||||
}
|
||||
|
@ -621,7 +656,6 @@ static int
|
|||
scsi_alloc_unit(struct scsi_link *sc_link)
|
||||
{
|
||||
u_int32 unit;
|
||||
struct scsi_link **strealloc;
|
||||
struct scsi_data *sd;
|
||||
struct scsi_device *dsw;
|
||||
|
||||
|
@ -667,6 +701,120 @@ scsi_free_unit(struct scsi_link *sc_link)
|
|||
extend_release(sc_link->device->links, sc_link->dev_unit);
|
||||
}
|
||||
|
||||
#if NSCTARG > 0
|
||||
|
||||
/* XXX: It is a bug that the sc_link has this information
|
||||
* about the adapter in it. The sc_link should refer to
|
||||
* a structure that is host adpater specific. That will also
|
||||
* pull all knowledge of an sc_link out of the adapter drivers.
|
||||
*/
|
||||
|
||||
errval
|
||||
scsi_set_bus(int bus, struct scsi_link *sc_link)
|
||||
{
|
||||
struct scsi_link *ad_link;
|
||||
struct scsibus_data *scsibus_data;
|
||||
|
||||
if (bus < 0 || bus > scbusses->nelem) {
|
||||
return ENXIO;
|
||||
}
|
||||
|
||||
scsibus_data = (struct scsibus_data *)extend_get(scbusses, bus);
|
||||
|
||||
if(!scsibus_data) {
|
||||
return ENXIO;
|
||||
}
|
||||
|
||||
ad_link = scsibus_data->adapter_link;
|
||||
|
||||
sc_link->adapter_unit = ad_link->adapter_unit;
|
||||
sc_link->adapter_targ = ad_link->adapter_targ;
|
||||
sc_link->adapter = ad_link->adapter;
|
||||
sc_link->device = ad_link->device;
|
||||
sc_link->flags = ad_link->flags;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Allocate and attach as many SCSI target devices as configured.
|
||||
* There are two ways that you can configure the target device:
|
||||
* 1. In the configuration file. That is handled here.
|
||||
* 2. Via the minor number. That takes precedence over the config file.
|
||||
*/
|
||||
static errval
|
||||
scsi_attach_sctarg()
|
||||
{
|
||||
struct scsi_link *sc_link = NULL;
|
||||
int dev_unit;
|
||||
struct scsi_device *sctarg = scsi_device_lookup(T_TARGET);
|
||||
|
||||
if (sctarg == 0) {
|
||||
return ENXIO;
|
||||
}
|
||||
|
||||
for (dev_unit = 0; dev_unit < NSCTARG; dev_unit++) {
|
||||
|
||||
int target, lun, bus;
|
||||
|
||||
/* If we don't have a link block allocate one.
|
||||
*/
|
||||
if (!sc_link) {
|
||||
sc_link = malloc(sizeof(*sc_link), M_TEMP, M_NOWAIT);
|
||||
}
|
||||
|
||||
scsi_sctarg_lookup(sctarg->name, dev_unit, &target, &lun, &bus);
|
||||
|
||||
if (IS_SPECIFIED(bus)) {
|
||||
struct scsibus_data *scsibus_data;
|
||||
|
||||
if (bus < 0 || bus > scbusses->nelem) {
|
||||
printf("%s%d: configured on illegal bus %d.\n",
|
||||
sctarg->name, dev_unit, bus);
|
||||
continue;
|
||||
}
|
||||
|
||||
scsibus_data = (struct scsibus_data *)extend_get(scbusses, bus);
|
||||
|
||||
if(!scsibus_data) {
|
||||
printf("%s%d: no bus %d.\n", sctarg->name, dev_unit, bus);
|
||||
continue;
|
||||
}
|
||||
|
||||
*sc_link = *scsibus_data->adapter_link; /* struct copy */
|
||||
sc_link->target = target;
|
||||
sc_link->lun = lun;
|
||||
}
|
||||
else {
|
||||
/* This will be configured in the open routine.
|
||||
*/
|
||||
sc_link->scsibus = SCCONF_UNSPEC;
|
||||
sc_link->target = SCCONF_UNSPEC;
|
||||
sc_link->lun = SCCONF_UNSPEC;
|
||||
}
|
||||
|
||||
sc_link->quirks = 0;
|
||||
sc_link->device = sctarg;
|
||||
sc_link->dev_unit = dev_unit;
|
||||
|
||||
if (scsi_alloc_unit(sc_link)) {
|
||||
|
||||
if (scsi_device_attach(sc_link) == 0) {
|
||||
sc_link = NULL; /* it's been used */
|
||||
}
|
||||
else
|
||||
scsi_free_unit(sc_link);
|
||||
}
|
||||
}
|
||||
|
||||
if (sc_link) {
|
||||
free(sc_link, M_TEMP);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
#endif /* NSCTARG > 0 */
|
||||
|
||||
/*
|
||||
* Probe the requested scsi bus. It must be already set up.
|
||||
* targ and lun optionally narrow the search if not -1
|
||||
|
@ -674,7 +822,7 @@ scsi_free_unit(struct scsi_link *sc_link)
|
|||
errval
|
||||
scsi_probe_bus(int bus, int targ, int lun)
|
||||
{
|
||||
struct scsibus_data *scsi ;
|
||||
struct scsibus_data *scsibus_data ;
|
||||
int maxtarg,mintarg,maxlun,minlun;
|
||||
struct scsi_link *sc_link_proto;
|
||||
u_int8 scsi_addr ;
|
||||
|
@ -686,9 +834,9 @@ scsi_probe_bus(int bus, int targ, int lun)
|
|||
if ((bus < 0 ) || ( bus >= scbusses->nelem)) {
|
||||
return ENXIO;
|
||||
}
|
||||
scsi = (struct scsibus_data *)extend_get(scbusses, bus);
|
||||
if(!scsi) return ENXIO;
|
||||
sc_link_proto = scsi->adapter_link;
|
||||
scsibus_data = (struct scsibus_data *)extend_get(scbusses, bus);
|
||||
if(!scsibus_data) return ENXIO;
|
||||
sc_link_proto = scsibus_data->adapter_link;
|
||||
scsi_addr = sc_link_proto->adapter_targ;
|
||||
if(targ == -1){
|
||||
maxtarg = 7;
|
||||
|
@ -716,7 +864,7 @@ scsi_probe_bus(int bus, int targ, int lun)
|
|||
* The spot appears to already have something
|
||||
* linked in, skip past it. Must be doing a 'reprobe'
|
||||
*/
|
||||
if(scsi->sc_link[targ][lun])
|
||||
if(scsibus_data->sc_link[targ][lun])
|
||||
{/* don't do this one, but check other luns */
|
||||
maybe_more = 1;
|
||||
continue;
|
||||
|
@ -751,7 +899,7 @@ scsi_probe_bus(int bus, int targ, int lun)
|
|||
if (scsi_alloc_unit(sc_link)) {
|
||||
|
||||
if (scsi_device_attach(sc_link) == 0) {
|
||||
scsi->sc_link[targ][lun] = sc_link;
|
||||
scsibus_data->sc_link[targ][lun] = sc_link;
|
||||
sc_link = NULL; /* it's been used */
|
||||
}
|
||||
else
|
||||
|
@ -779,16 +927,33 @@ scsi_link_get(bus, targ, lun)
|
|||
int targ;
|
||||
int lun;
|
||||
{
|
||||
struct scsibus_data *scsi =
|
||||
struct scsibus_data *scsibus_data =
|
||||
(struct scsibus_data *)extend_get(scbusses, bus);
|
||||
return (scsi) ? scsi->sc_link[targ][lun] : 0;
|
||||
return (scsibus_data) ? scsibus_data->sc_link[targ][lun] : 0;
|
||||
}
|
||||
|
||||
static void rm_spaces(char *text, int n)
|
||||
/* make_readable: Make the inquiry data readable. Anything less than a ' '
|
||||
* is made a '?' and trailing spaces are removed.
|
||||
*/
|
||||
static void
|
||||
make_readable(to, from, n)
|
||||
char *to;
|
||||
char *from;
|
||||
size_t n;
|
||||
{
|
||||
while (n && text[n - 1] == ' ')
|
||||
n--;
|
||||
text[n] = 0; /* Zap */
|
||||
int i;
|
||||
|
||||
for (i = 0; from[i] && i < n - 1; i++) {
|
||||
if (from[i] < ' ')
|
||||
to[i]='?';
|
||||
else
|
||||
to[i] = from[i];
|
||||
}
|
||||
|
||||
while (i && to[i - 1] == ' ')
|
||||
i--;
|
||||
|
||||
to[i] = 0;
|
||||
}
|
||||
|
||||
/*
|
||||
|
@ -802,20 +967,17 @@ scsi_probedev(sc_link, maybe_more, type_p)
|
|||
struct scsi_link *sc_link;
|
||||
int *type_p;
|
||||
{
|
||||
u_int8 unit = sc_link->adapter_unit;
|
||||
u_int8 target = sc_link->target;
|
||||
u_int8 lu = sc_link->lun;
|
||||
struct scsi_adapter *scsi_adapter = sc_link->adapter;
|
||||
struct scsidevs *bestmatch = (struct scsidevs *) 0;
|
||||
char *dtype = (char *) 0, *desc;
|
||||
char *qtype;
|
||||
struct scsi_inquiry_data *inqbuf;
|
||||
u_int32 len, qualifier, type;
|
||||
boolean remov;
|
||||
char manu[32];
|
||||
char model[32];
|
||||
char version[32];
|
||||
int z;
|
||||
char manu[8 + 1];
|
||||
char model[16 + 1];
|
||||
char version[4 + 1];
|
||||
|
||||
inqbuf = &sc_link->inqbuf;
|
||||
|
||||
|
@ -864,7 +1026,7 @@ scsi_probedev(sc_link, maybe_more, type_p)
|
|||
|
||||
/*
|
||||
* Any device qualifier that has the top bit set (qualifier&4 != 0)
|
||||
* is vendor specific and won't match in this switch.
|
||||
* is vendor specific and will match in the default of this switch.
|
||||
*/
|
||||
|
||||
switch ((int)qualifier) {
|
||||
|
@ -873,28 +1035,25 @@ scsi_probedev(sc_link, maybe_more, type_p)
|
|||
break;
|
||||
|
||||
case SID_QUAL_LU_OFFLINE:
|
||||
qtype = ", Unit not Connected!";
|
||||
qtype = "Supported device currently not connected";
|
||||
break;
|
||||
|
||||
case SID_QUAL_RSVD:
|
||||
qtype = ", Reserved Peripheral Qualifier!";
|
||||
case SID_QUAL_RSVD: /* Peripheral qualifier reserved in SCSI-2 spec */
|
||||
*maybe_more = 1;
|
||||
return (struct scsidevs *) 0;
|
||||
break;
|
||||
|
||||
case SID_QUAL_BAD_LU:
|
||||
case SID_QUAL_BAD_LU: /* Target can not support a device on this unit */
|
||||
/*
|
||||
* Check for a non-existent unit. If the device is returning
|
||||
* this much, then we must set the flag that has
|
||||
* the searchers keep looking on other luns.
|
||||
*/
|
||||
qtype = ", The Target can't support this Unit!";
|
||||
*maybe_more = 1;
|
||||
return (struct scsidevs *) 0;
|
||||
|
||||
default:
|
||||
dtype = "vendor specific";
|
||||
qtype = "";
|
||||
dtype = "";
|
||||
qtype = "Vendor specific peripheral qualifier";
|
||||
*maybe_more = 1;
|
||||
break;
|
||||
}
|
||||
|
@ -918,40 +1077,34 @@ scsi_probedev(sc_link, maybe_more, type_p)
|
|||
len = sizeof(struct scsi_inquiry_data) - 1;
|
||||
desc = inqbuf->vendor;
|
||||
desc[len - (desc - (char *) inqbuf)] = 0;
|
||||
strncpy(manu, inqbuf->vendor, 8);
|
||||
strncpy(model, inqbuf->product, 16);
|
||||
strncpy(version, inqbuf->revision, 4);
|
||||
for(z = 0; z < 4; z++) {
|
||||
if (version[z]<' ') version[z]='?';
|
||||
}
|
||||
} else
|
||||
make_readable(manu, inqbuf->vendor, sizeof(manu));
|
||||
make_readable(model, inqbuf->product, sizeof(model));
|
||||
make_readable(version, inqbuf->revision, sizeof(version));
|
||||
} else {
|
||||
/*
|
||||
* If not advanced enough, use default values
|
||||
*/
|
||||
{
|
||||
desc = "early protocol device";
|
||||
strncpy(manu, "unknown", 8);
|
||||
strncpy(model, "unknown", 16);
|
||||
strncpy(version, "????", 4);
|
||||
make_readable(manu, "unknown", sizeof(manu));
|
||||
make_readable(model, "unknown", sizeof(model));
|
||||
make_readable(version, "????", sizeof(version));
|
||||
}
|
||||
|
||||
rm_spaces(manu, 8);
|
||||
rm_spaces(model, 16);
|
||||
rm_spaces(version, 4);
|
||||
sc_print_start(sc_link);
|
||||
|
||||
sc_print_addr(sc_link);
|
||||
|
||||
printf("\"%s %s %s\" is a ", manu, model, version );
|
||||
printf("\"%s %s %s\" ", manu, model, version );
|
||||
printf("type %ld %sSCSI %d"
|
||||
,type
|
||||
,remov ? "removable " : "fixed "
|
||||
,inqbuf->version & SID_ANSII
|
||||
);
|
||||
if (qtype[0]) {
|
||||
printf(" qualifier %ld(%s)" ,qualifier ,qtype);
|
||||
sc_print_addr(sc_link);
|
||||
printf(" qualifier %ld: %s" ,qualifier ,qtype);
|
||||
}
|
||||
|
||||
printf("\n");
|
||||
sc_print_finish();
|
||||
|
||||
/*
|
||||
* Try make as good a match as possible with
|
||||
|
|
|
@ -14,7 +14,7 @@
|
|||
*
|
||||
* Ported to run under 386BSD by Julian Elischer (julian@tfs.com) Sept 1992
|
||||
*
|
||||
* $Id: scsiconf.h,v 1.20 1995/03/16 18:15:50 bde Exp $
|
||||
* $Id: scsiconf.h,v 1.21 1995/03/21 11:21:05 dufault Exp $
|
||||
*/
|
||||
#ifndef SCSI_SCSICONF_H
|
||||
#define SCSI_SCSICONF_H 1
|
||||
|
@ -31,9 +31,11 @@ typedef unsigned char u_int8;
|
|||
#include <scsi/scsi_all.h>
|
||||
#include <scsi/scsi_driver.h>
|
||||
|
||||
#include <machine/cpu.h> /* XXX For bootverbose (funny place) */
|
||||
|
||||
/* Minor number fields:
|
||||
*
|
||||
* OLD STYLE SCSI devices:
|
||||
* NON-FIXED SCSI devices:
|
||||
*
|
||||
* ???? ???? ???? ???N MMMMMMMM mmmmmmmm
|
||||
*
|
||||
|
@ -42,7 +44,7 @@ typedef unsigned char u_int8;
|
|||
* M: Major device number.
|
||||
* m: old style minor device number.
|
||||
*
|
||||
* NEW (FIXED) SCSI devices:
|
||||
* FIXED SCSI devices:
|
||||
*
|
||||
* ???? SBBB LLLI IIIN MMMMMMMM mmmmmmmm
|
||||
*
|
||||
|
@ -62,7 +64,7 @@ typedef unsigned char u_int8;
|
|||
#define SCSI_BUS(DEV) (((DEV) & 0x07000000) >> 24)
|
||||
#define SCSI_LUN(DEV) (((DEV) & 0x00E00000) >> 21)
|
||||
#define SCSI_ID(DEV) (((DEV) & 0x001E0000) >> 17)
|
||||
#define SCSI_NEW(DEV) (((DEV) & 0x00010000) >> 16)
|
||||
#define SCSI_FIXED(DEV) (((DEV) & 0x00010000) >> 16)
|
||||
|
||||
|
||||
#define SCSI_MKDEV(B, L, I) ( \
|
||||
|
@ -243,8 +245,8 @@ void NAME##strategy(struct buf *bp) \
|
|||
/* A unit, type, etc can be SCCONF_ANY to indicate it is a '?'
|
||||
* in the config.
|
||||
*/
|
||||
#define SCCONF_UNSPEC -1
|
||||
#define SCCONF_ANY -2
|
||||
#define SCCONF_UNSPEC 255
|
||||
#define SCCONF_ANY 254
|
||||
|
||||
struct isa_driver;
|
||||
struct scsi_ctlr_config
|
||||
|
@ -340,20 +342,25 @@ struct scsi_link
|
|||
/* 44+*/struct scsi_inquiry_data inqbuf; /* Inquiry data */
|
||||
};
|
||||
|
||||
/* XXX dufault@hda.com: SDEV_BOUNCE is set down in the adapter drivers
|
||||
/* XXX-HA: dufault@hda.com: SDEV_BOUNCE is set down in the adapter drivers
|
||||
* in an sc_link structure to indicate that this host adapter requires
|
||||
* ISA DMA bounce buffers. I think the link structure should
|
||||
* be associated only with the type drive and not the adapter driver,
|
||||
* and the bounce flag should be in something associated with the
|
||||
* adapter driver.
|
||||
* XXX-HA And I added the "supports residuals properly" flag that ALSO goes
|
||||
* in an adapter structure. I figure I'll fix both at once.
|
||||
*/
|
||||
|
||||
#define SDEV_MEDIA_LOADED 0x0001 /* device figures are still valid */
|
||||
#define SDEV_WAITING 0x0002 /* a process is waiting for this */
|
||||
#define SDEV_OPEN 0x0004 /* at least 1 open session */
|
||||
#define SDEV_BOUNCE 0x0008 /* unit requires DMA bounce buffer */
|
||||
#define SDEV_BOUNCE 0x0008 /* XXX-HA: unit needs DMA bounce buffer */
|
||||
#define SDEV_DBX 0x00F0 /* debugging flags (scsi_debug.h) */
|
||||
#define SDEV_ONCE_ONLY 0x0100 /* unit can only be opened once */
|
||||
#define SDEV_BOOTVERBOSE 0x0200 /* be noisy during boot */
|
||||
#define SDEV_RESIDS_WORK 0x0400 /* XXX-HA: Residuals work */
|
||||
#define SDEV_TARGET_OPS 0x0800 /* XXX-HA: Supports target ops */
|
||||
|
||||
/*
|
||||
* One of these is allocated and filled in for each scsi bus.
|
||||
|
@ -406,7 +413,9 @@ struct scsi_xfer
|
|||
#define SCSI_NOMASK 0x02 /* dont allow interrupts.. booting */
|
||||
#define SCSI_NOSTART 0x04 /* left over from ancient history */
|
||||
#define SCSI_USER 0x08 /* Is a user cmd, call scsi_user_done */
|
||||
#define SCSI_ITSDONE 0x10 /* the transfer is as done as it gets */
|
||||
#define ITSDONE 0x10 /* the transfer is as done as it gets */
|
||||
#define SCSI_INUSE 0x20 /* The scsi_xfer block is in use */
|
||||
#define INUSE 0x20 /* The scsi_xfer block is in use */
|
||||
#define SCSI_SILENT 0x40 /* Don't report errors to console */
|
||||
#define SCSI_ERR_OK 0x80 /* An error on this operation is OK. */
|
||||
|
@ -416,6 +425,8 @@ struct scsi_xfer
|
|||
#define SCSI_DATA_OUT 0x800 /* expect data to flow OUT of memory */
|
||||
#define SCSI_TARGET 0x1000 /* This defines a TARGET mode op. */
|
||||
#define SCSI_ESCAPE 0x2000 /* Escape operation */
|
||||
#define SCSI_EOF 0x4000 /* The operation should return EOF */
|
||||
#define SCSI_RESID_VALID 0x8000 /* The resid field contains valid data */
|
||||
|
||||
/*
|
||||
* Escape op codes. This provides an extensible setup for operations
|
||||
|
@ -447,10 +458,13 @@ void free_xs(struct scsi_xfer *xs, struct scsi_link *sc_link,u_int32 flags);
|
|||
u_int32 scsi_read_capacity __P(( struct scsi_link *sc_link,
|
||||
u_int32 *blk_size, u_int32 flags));
|
||||
errval scsi_test_unit_ready( struct scsi_link *sc_link, u_int32 flags);
|
||||
errval scsi_reset_target __P((struct scsi_link *));
|
||||
errval scsi_target_mode __P((struct scsi_link *, int));
|
||||
errval scsi_change_def( struct scsi_link *sc_link, u_int32 flags);
|
||||
errval scsi_inquire( struct scsi_link *sc_link,
|
||||
struct scsi_inquiry_data *inqbuf, u_int32 flags);
|
||||
errval scsi_prevent( struct scsi_link *sc_link, u_int32 type,u_int32 flags);
|
||||
errval scsi_probe_bus __P((int, int, int));
|
||||
errval scsi_probe_busses __P(( int, int, int));
|
||||
errval scsi_start_unit( struct scsi_link *sc_link, u_int32 flags);
|
||||
errval scsi_stop_unit(struct scsi_link *sc_link, u_int32 eject, u_int32 flags);
|
||||
|
@ -469,9 +483,11 @@ dev_t scsi_dev_lookup __P((int (*opener)(dev_t dev, int flags, int fmt,
|
|||
struct proc *p)));
|
||||
|
||||
int scsi_opened_ok __P((dev_t dev, int flag, int type, struct scsi_link *sc_link));
|
||||
errval scsi_set_bus __P((int, struct scsi_link *));
|
||||
|
||||
char *scsi_sense_desc __P((int, int));
|
||||
void scsi_sense_print __P((struct scsi_xfer *));
|
||||
int scsi_sense_qualifiers __P((struct scsi_xfer *, int *, int *));
|
||||
void show_scsi_xs __P((struct scsi_xfer *));
|
||||
void show_scsi_cmd __P((struct scsi_xfer *));
|
||||
void show_mem __P((unsigned char * , u_int32));
|
||||
|
@ -494,7 +510,10 @@ void scsi_device_register __P((struct scsi_device *sd));
|
|||
|
||||
extern struct kern_devconf kdc_scbus0; /* XXX should go away */
|
||||
|
||||
#endif
|
||||
void scsi_configure_start __P((void));
|
||||
void scsi_configure_finish __P((void));
|
||||
|
||||
#endif /* KERNEL */
|
||||
|
||||
#define SCSI_EXTERNALLEN (sizeof(struct scsi_link))
|
||||
|
||||
|
|
271
sys/scsi/sctarg.c
Normal file
271
sys/scsi/sctarg.c
Normal file
|
@ -0,0 +1,271 @@
|
|||
/*
|
||||
* sctarg: Processor Type driver.
|
||||
*
|
||||
* Copyright (C) 1995, HD Associates, Inc.
|
||||
* PO Box 276
|
||||
* Pepperell, MA 01463
|
||||
* 508 433 5266
|
||||
* dufault@hda.com
|
||||
*
|
||||
* This code is contributed to the University of California at Berkeley:
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
* modification, are permitted provided that the following conditions
|
||||
* are met:
|
||||
* 1. Redistributions of source code must retain the above copyright
|
||||
* notice, this list of conditions and the following disclaimer.
|
||||
* 2. Redistributions in binary form must reproduce the above copyright
|
||||
* notice, this list of conditions and the following disclaimer in the
|
||||
* documentation and/or other materials provided with the distribution.
|
||||
* 3. All advertising materials mentioning features or use of this software
|
||||
* must display the following acknowledgement:
|
||||
* This product includes software developed by the University of
|
||||
* California, Berkeley and its contributors.
|
||||
* 4. Neither the name of the University nor the names of its contributors
|
||||
* may be used to endorse or promote products derived from this software
|
||||
* without specific prior written permission.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
|
||||
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||||
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
|
||||
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
||||
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
||||
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
||||
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
||||
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||||
* SUCH DAMAGE.
|
||||
*
|
||||
* $Id: sctarg.c,v 1.1 1995/03/04 20:50:46 dufault Exp $
|
||||
*/
|
||||
|
||||
/*
|
||||
* XXX dufault@hda.com: We need the "kern devconf" stuff, but I'm not
|
||||
* going to add it until it is done in a simple way that provides
|
||||
* base behavior in scsi_driver.c
|
||||
*/
|
||||
|
||||
#include <sys/param.h>
|
||||
#include <sys/systm.h>
|
||||
#include <sys/buf.h>
|
||||
#include <sys/proc.h>
|
||||
#include <scsi/scsi_all.h>
|
||||
#include <scsi/scsiconf.h>
|
||||
|
||||
#define OPEN 0x01
|
||||
|
||||
struct scsi_data {
|
||||
struct buf *buf_queue; /* the queue of pending IO operations */
|
||||
int flags; /* Already open */
|
||||
};
|
||||
|
||||
errval sctarg_open(dev_t dev, int flags, int fmt, struct proc *p,
|
||||
struct scsi_link *sc_link);
|
||||
void sctargstart(u_int32 unit);
|
||||
errval sctarg_close(dev_t dev, int flag, int fmt, struct proc *p,
|
||||
struct scsi_link *sc_link);
|
||||
void sctarg_strategy(struct buf *bp, struct scsi_link *sc_link);
|
||||
|
||||
SCSI_DEVICE_ENTRIES(sctarg)
|
||||
|
||||
struct scsi_device sctarg_switch =
|
||||
{
|
||||
NULL,
|
||||
sctargstart, /* we have a queue, and this is how we service it */
|
||||
NULL,
|
||||
NULL,
|
||||
"sctarg",
|
||||
0,
|
||||
{0, 0},
|
||||
SDEV_ONCE_ONLY,
|
||||
0,
|
||||
"Processor Target",
|
||||
sctargopen,
|
||||
sizeof(struct scsi_data),
|
||||
T_TARGET,
|
||||
0,
|
||||
0,
|
||||
sctarg_open,
|
||||
0,
|
||||
0,
|
||||
sctarg_strategy,
|
||||
};
|
||||
|
||||
errval sctarg_open(dev_t dev, int flags, int fmt, struct proc *p,
|
||||
struct scsi_link *sc_link)
|
||||
{
|
||||
int ret = 0;
|
||||
|
||||
/* Does this host adapter support target mode operation?
|
||||
*/
|
||||
if ((sc_link->flags & SDEV_TARGET_OPS) == 0)
|
||||
return ENODEV; /* Operation not supported */
|
||||
|
||||
if (SCSI_FIXED(dev)) {
|
||||
sc_link->scsibus = SCSI_BUS(dev);
|
||||
scsi_set_bus(sc_link->scsibus, sc_link);
|
||||
|
||||
sc_link->target = SCSI_ID(dev);
|
||||
sc_link->lun = SCSI_LUN(dev);
|
||||
}
|
||||
|
||||
if (sc_link->scsibus == SCCONF_UNSPEC ||
|
||||
sc_link->target == SCCONF_UNSPEC ||
|
||||
sc_link->lun == SCCONF_UNSPEC)
|
||||
return ENXIO;
|
||||
|
||||
/* XXX: You can have more than one target device on a single
|
||||
* host adapter. We need a reference count.
|
||||
*/
|
||||
if ((sc_link->sd->flags & OPEN) == 0) /* Enable target mode */
|
||||
{
|
||||
ret = scsi_target_mode(sc_link, 1);
|
||||
sc_link->sd->flags |= OPEN;
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
errval sctarg_close(dev_t dev, int flags, int fmt, struct proc *p,
|
||||
struct scsi_link *sc_link)
|
||||
{
|
||||
int ret = 0;
|
||||
|
||||
/* XXX: You can have more than one target device on a single
|
||||
* host adapter. We need a reference count.
|
||||
*/
|
||||
ret = scsi_target_mode(sc_link, 0);
|
||||
|
||||
sc_link->sd->flags &= ~OPEN;
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
/*
|
||||
* sctargstart looks to see if there is a buf waiting for the device
|
||||
* and that the device is not already busy. If both are true,
|
||||
* It dequeues the buf and creates a scsi command to perform the
|
||||
* transfer required. The transfer request will call scsi_done
|
||||
* on completion, which will in turn call this routine again
|
||||
* so that the next queued transfer is performed.
|
||||
* The bufs are queued by the strategy routine (sctargstrategy)
|
||||
*
|
||||
* This routine is also called after other non-queued requests
|
||||
* have been made of the scsi driver, to ensure that the queue
|
||||
* continues to be drained.
|
||||
* sctargstart() is called at splbio
|
||||
*/
|
||||
void
|
||||
sctargstart(unit)
|
||||
u_int32 unit;
|
||||
{
|
||||
struct scsi_link *sc_link = SCSI_LINK(&sctarg_switch, unit);
|
||||
struct scsi_data *sctarg = sc_link->sd;
|
||||
register struct buf *bp = 0;
|
||||
struct
|
||||
{
|
||||
#define PROCESSOR_SEND 0x0A
|
||||
#define PROCESSOR_RECEIVE 0x08
|
||||
u_char op_code;
|
||||
u_char byte2;
|
||||
u_char len[3];
|
||||
u_char control;
|
||||
} cmd;
|
||||
|
||||
u_int32 flags;
|
||||
|
||||
SC_DEBUG(sc_link, SDEV_DB2, ("sctargstart "));
|
||||
/*
|
||||
* See if there is a buf to do and we are not already
|
||||
* doing one
|
||||
*/
|
||||
while (sc_link->opennings != 0) {
|
||||
|
||||
/* if a special awaits, let it proceed first */
|
||||
if (sc_link->flags & SDEV_WAITING) {
|
||||
sc_link->flags &= ~SDEV_WAITING;
|
||||
wakeup((caddr_t)sc_link);
|
||||
return;
|
||||
}
|
||||
if ((bp = sctarg->buf_queue) == NULL) {
|
||||
return; /* no work to bother with */
|
||||
}
|
||||
sctarg->buf_queue = bp->b_actf;
|
||||
|
||||
/*
|
||||
* Fill out the scsi command
|
||||
*/
|
||||
bzero(&cmd, sizeof(cmd));
|
||||
flags = SCSI_TARGET;
|
||||
if ((bp->b_flags & B_READ) == B_WRITE) {
|
||||
cmd.op_code = PROCESSOR_SEND;
|
||||
flags |= SCSI_DATA_OUT;
|
||||
} else {
|
||||
cmd.op_code = PROCESSOR_RECEIVE;
|
||||
flags |= SCSI_DATA_IN;
|
||||
}
|
||||
|
||||
scsi_uto3b(bp->b_bcount, cmd.len);
|
||||
/*
|
||||
* go ask the adapter to do all this for us
|
||||
*/
|
||||
if (scsi_scsi_cmd(sc_link,
|
||||
(struct scsi_generic *) &cmd,
|
||||
sizeof(cmd),
|
||||
(u_char *) bp->b_un.b_addr,
|
||||
bp->b_bcount,
|
||||
0,
|
||||
100000,
|
||||
bp,
|
||||
flags | SCSI_NOSLEEP) == SUCCESSFULLY_QUEUED) {
|
||||
} else {
|
||||
printf("sctarg%ld: oops not queued\n", unit);
|
||||
bp->b_flags |= B_ERROR;
|
||||
bp->b_error = EIO;
|
||||
biodone(bp);
|
||||
}
|
||||
} /* go back and see if we can cram more work in.. */
|
||||
}
|
||||
|
||||
void
|
||||
sctarg_strategy(struct buf *bp, struct scsi_link *sc_link)
|
||||
{
|
||||
struct buf **dp;
|
||||
unsigned char unit;
|
||||
u_int32 opri;
|
||||
struct scsi_data *sctarg;
|
||||
|
||||
unit = STUNIT((bp->b_dev));
|
||||
sctarg = sc_link->sd;
|
||||
|
||||
opri = splbio();
|
||||
|
||||
/*
|
||||
* Use a bounce buffer if necessary
|
||||
*/
|
||||
#ifdef BOUNCE_BUFFERS
|
||||
if (sc_link->flags & SDEV_BOUNCE)
|
||||
vm_bounce_alloc(bp);
|
||||
#endif
|
||||
|
||||
/*
|
||||
* Place it at the end of the queue of activities for this device.
|
||||
*/
|
||||
dp = &(sctarg->buf_queue);
|
||||
while (*dp) {
|
||||
dp = &((*dp)->b_actf);
|
||||
}
|
||||
*dp = bp;
|
||||
bp->b_actf = NULL;
|
||||
|
||||
/*
|
||||
* Tell the device to get going on the transfer if it's
|
||||
* not doing anything, otherwise just wait for completion
|
||||
* (All a bit silly if we're only allowing 1 open but..)
|
||||
*/
|
||||
sctargstart(unit);
|
||||
|
||||
splx(opri);
|
||||
return;
|
||||
}
|
|
@ -50,10 +50,6 @@ u_int32 sdstrats, sdqueues;
|
|||
#define SD_RETRIES 4
|
||||
#define MAXTRANSFER 8 /* 1 page at a time */
|
||||
|
||||
#define SDUNITSHIFT 3
|
||||
#define SDUNIT(DEV) SH3_UNIT(DEV)
|
||||
#define SDSETUNIT(DEV, U) SH3SETUNIT((DEV), (U))
|
||||
|
||||
#define PARTITION(dev) dkpart(dev)
|
||||
#define SDUNIT(dev) dkunit(dev)
|
||||
|
||||
|
@ -187,19 +183,20 @@ sdattach(struct scsi_link *sc_link)
|
|||
* (could happen on removable media - MOD)
|
||||
* -- this avoids the division below from falling over
|
||||
*/
|
||||
if(dp->secsiz == 0) dp->secsiz = 512;
|
||||
printf("%luMB (%lu S), %u C %u H %u S/T %u B/S",
|
||||
if(dp->secsiz == 0) dp->secsiz = SECSIZE;
|
||||
printf("%ldMB (%ld %d byte sectors)",
|
||||
dp->disksize / ((1024L * 1024L) / dp->secsiz),
|
||||
dp->disksize,
|
||||
dp->cyls,
|
||||
dp->heads,
|
||||
dp->sectors,
|
||||
dp->secsiz);
|
||||
|
||||
#ifndef SCSI_REPORT_GEOMETRY
|
||||
if ( (sc_link->flags & SDEV_BOOTVERBOSE) )
|
||||
#endif
|
||||
{
|
||||
printf("\n");
|
||||
sc_print_addr(sc_link);
|
||||
printf("with %d cyls, %d heads, and an average %d sectors/track",
|
||||
dp->cyls, dp->heads, dp->sectors);
|
||||
}
|
||||
|
||||
sd->flags |= SDINIT;
|
||||
|
@ -237,7 +234,7 @@ sd_open(dev, mode, fmt, p, sc_link)
|
|||
}
|
||||
|
||||
SC_DEBUG(sc_link, SDEV_DB1,
|
||||
("sd_open: dev=0x%x (unit %d, partition %d)\n",
|
||||
("sd_open: dev=0x%lx (unit %ld, partition %d)\n",
|
||||
dev, unit, PARTITION(dev)));
|
||||
|
||||
/*
|
||||
|
@ -318,7 +315,7 @@ sd_open(dev, mode, fmt, p, sc_link)
|
|||
goto bad;
|
||||
SC_DEBUG(sc_link, SDEV_DB3, ("Slice tables initialized "));
|
||||
|
||||
SC_DEBUG(sc_link, SDEV_DB3, ("open %d %d\n", sdstrats, sdqueues));
|
||||
SC_DEBUG(sc_link, SDEV_DB3, ("open %ld %ld\n", sdstrats, sdqueues));
|
||||
|
||||
return 0;
|
||||
|
||||
|
@ -376,12 +373,7 @@ sd_strategy(struct buf *bp, struct scsi_link *sc_link)
|
|||
bp->b_error = EIO;
|
||||
goto bad;
|
||||
}
|
||||
/*
|
||||
* If it's a null transfer, return immediatly
|
||||
*/
|
||||
if (bp->b_bcount == 0) {
|
||||
goto done;
|
||||
}
|
||||
|
||||
/*
|
||||
* Odd number of bytes
|
||||
*/
|
||||
|
@ -503,7 +495,7 @@ sdstart(u_int32 unit)
|
|||
* With this thing..
|
||||
*/
|
||||
blkno = bp->b_pblkno;
|
||||
if (bp->b_bcount & 511)
|
||||
if (bp->b_bcount & (SECSIZE - 1))
|
||||
{
|
||||
goto bad;
|
||||
}
|
||||
|
@ -733,7 +725,7 @@ sd_get_parms(unit, flags)
|
|||
} else {
|
||||
|
||||
SC_DEBUG(sc_link, SDEV_DB3,
|
||||
("%d cyls, %d heads, %d precomp, %d red_write, %d land_zone\n",
|
||||
("%ld cyls, %d heads, %d precomp, %d red_write, %d land_zone\n",
|
||||
scsi_3btou(&scsi_sense.pages.rigid_geometry.ncyl_2),
|
||||
scsi_sense.pages.rigid_geometry.nheads,
|
||||
b2tol(scsi_sense.pages.rigid_geometry.st_cyl_wp),
|
||||
|
@ -822,6 +814,11 @@ int sd_sense_handler(struct scsi_xfer *xs)
|
|||
if ((sense->error_code & SSD_ERRCODE) == 0x71)
|
||||
return SCSIRET_CONTINUE;
|
||||
|
||||
if (((sense->error_code & SSD_ERRCODE) == 0x70) &&
|
||||
((sense->ext.extended.flags & SSD_KEY) == 0x05))
|
||||
/* No point in retrying Illegal Requests */
|
||||
return SCSIRET_CONTINUE;
|
||||
|
||||
inqbuf = &(xs->sc_link->inqbuf);
|
||||
|
||||
/* It is dangerous to retry on removable drives without
|
||||
|
@ -861,7 +858,6 @@ sddump(dev_t dev)
|
|||
static int sddoingadump = 0;
|
||||
struct scsi_xfer *xs = &sx;
|
||||
errval retval;
|
||||
int c;
|
||||
|
||||
addr = (char *) 0; /* starting address */
|
||||
|
||||
|
@ -938,11 +934,11 @@ sddump(dev_t dev)
|
|||
xs->timeout = 10000; /* 10000 millisecs for a disk ! */
|
||||
xs->cmd = (struct scsi_generic *) &cmd;
|
||||
xs->cmdlen = sizeof(cmd);
|
||||
xs->resid = blkcnt * 512;
|
||||
xs->resid = 0;
|
||||
xs->error = XS_NOERROR;
|
||||
xs->bp = 0;
|
||||
xs->data = (u_char *) CADDR1; /* XXX use pmap_enter() */
|
||||
xs->datalen = blkcnt * 512;
|
||||
xs->datalen = blkcnt * SECSIZE;
|
||||
|
||||
/*
|
||||
* Pass all this info to the scsi driver.
|
||||
|
@ -963,7 +959,7 @@ sddump(dev_t dev)
|
|||
/* update block count */
|
||||
num -= blkcnt;
|
||||
blknum += blkcnt;
|
||||
(int) addr += 512 * blkcnt;
|
||||
(int) addr += SECSIZE * blkcnt;
|
||||
|
||||
/* operator aborting dump? */
|
||||
if (cncheckc())
|
||||
|
|
241
sys/scsi/st.c
241
sys/scsi/st.c
|
@ -12,7 +12,7 @@
|
|||
* on the understanding that TFS is not responsible for the correct
|
||||
* functioning of this software in any circumstances.
|
||||
*
|
||||
* $Id: st.c,v 1.30 1995/03/15 14:22:11 dufault Exp $
|
||||
* $Id: st.c,v 1.31 1995/03/21 11:21:08 dufault Exp $
|
||||
*/
|
||||
|
||||
/*
|
||||
|
@ -343,10 +343,10 @@ stattach(struct scsi_link *sc_link)
|
|||
NULL, 0, 0)) {
|
||||
printf("drive offline");
|
||||
} else {
|
||||
printf("density code 0x%x, ", st->media_density);
|
||||
printf("density code 0x%lx, ", st->media_density);
|
||||
if (!scsi_test_unit_ready(sc_link, SCSI_NOSLEEP | SCSI_NOMASK | SCSI_SILENT)) {
|
||||
if (st->media_blksiz) {
|
||||
printf("%d-byte", st->media_blksiz);
|
||||
printf("%ld-byte", st->media_blksiz);
|
||||
} else {
|
||||
printf("variable");
|
||||
}
|
||||
|
@ -390,7 +390,7 @@ st_identify_drive(unit)
|
|||
*/
|
||||
if (scsi_inquire(sc_link, inqbuf,
|
||||
SCSI_NOSLEEP | SCSI_NOMASK | SCSI_SILENT) != 0) {
|
||||
printf("st%d: couldn't get device type, using default\n", unit);
|
||||
printf("st%ld: couldn't get device type, using default\n", unit);
|
||||
return;
|
||||
}
|
||||
if ((inqbuf->version & SID_ANSII) == 0) {
|
||||
|
@ -431,7 +431,7 @@ st_identify_drive(unit)
|
|||
strcmp("????????????????", finger->model) == 0)
|
||||
&& (strcmp(version, finger->version) == 0 ||
|
||||
strcmp("????", finger->version) == 0)) {
|
||||
printf("st%d: %s is a known rogue\n", unit, finger->name);
|
||||
printf("st%ld: %s is a known rogue\n", unit, finger->name);
|
||||
st->rogues = finger;
|
||||
st->drive_quirks = finger->quirks;
|
||||
st->quirks = finger->quirks; /*start value */
|
||||
|
@ -550,7 +550,7 @@ struct scsi_link *sc_link)
|
|||
* Check that the device is ready to use (media loaded?)
|
||||
* This time take notice of the return result
|
||||
*/
|
||||
if (errno = (scsi_test_unit_ready(sc_link, 0))) {
|
||||
if ( (errno = scsi_test_unit_ready(sc_link, 0)) ) {
|
||||
uprintf("st%d: not ready\n", unit);
|
||||
st_unmount(unit, NOEJECT);
|
||||
return (errno);
|
||||
|
@ -653,7 +653,7 @@ st_mount_tape(dev, flags)
|
|||
* If the media is new, then make sure we give it a chance to
|
||||
* to do a 'load' instruction. ( We assume it is new)
|
||||
*/
|
||||
if (errno = st_load(unit, LD_LOAD, 0)) {
|
||||
if ( (errno = st_load(unit, LD_LOAD, 0)) ) {
|
||||
return (errno);
|
||||
}
|
||||
/*
|
||||
|
@ -669,14 +669,14 @@ st_mount_tape(dev, flags)
|
|||
* asked to look at the media. This quirk does this.
|
||||
*/
|
||||
if (st->quirks & ST_Q_SNS_HLP) {
|
||||
if (errno = st_touch_tape(unit))
|
||||
if ( (errno = st_touch_tape(unit)) )
|
||||
return errno;
|
||||
}
|
||||
/*
|
||||
* Load the physical device parameters
|
||||
* loads: blkmin, blkmax
|
||||
*/
|
||||
if (errno = st_rd_blk_lim(unit, 0)) {
|
||||
if ( (errno = st_rd_blk_lim(unit, 0)) ) {
|
||||
return errno;
|
||||
}
|
||||
/*
|
||||
|
@ -685,7 +685,7 @@ st_mount_tape(dev, flags)
|
|||
* As we have a tape in, it should be reflected here.
|
||||
* If not you may need the "quirk" above.
|
||||
*/
|
||||
if (errno = st_mode_sense(unit, 0, NULL, 0, 0)) {
|
||||
if ( (errno = st_mode_sense(unit, 0, NULL, 0, 0)) ) {
|
||||
return errno;
|
||||
}
|
||||
/*
|
||||
|
@ -710,12 +710,12 @@ st_mount_tape(dev, flags)
|
|||
st->flags |= ST_FIXEDBLOCKS;
|
||||
}
|
||||
} else {
|
||||
if (errno = st_decide_mode(unit, FALSE)) {
|
||||
if ( (errno = st_decide_mode(unit, FALSE)) ) {
|
||||
return errno;
|
||||
}
|
||||
}
|
||||
if (errno = st_mode_select(unit, 0, NULL, 0)) {
|
||||
printf("st%d: Cannot set selected mode", unit);
|
||||
if ( (errno = st_mode_select(unit, 0, NULL, 0)) ) {
|
||||
printf("st%ld: Cannot set selected mode", unit);
|
||||
return errno;
|
||||
}
|
||||
scsi_prevent(sc_link, PR_PREVENT, 0); /* who cares if it fails? */
|
||||
|
@ -776,7 +776,7 @@ st_decide_mode(unit, first_read)
|
|||
*/
|
||||
switch ((int)(st->quirks & (ST_Q_FORCE_FIXED_MODE | ST_Q_FORCE_VAR_MODE))) {
|
||||
case (ST_Q_FORCE_FIXED_MODE | ST_Q_FORCE_VAR_MODE):
|
||||
printf("st%d: bad quirks\n", unit);
|
||||
printf("st%ld: bad quirks\n", unit);
|
||||
return (EINVAL);
|
||||
case ST_Q_FORCE_FIXED_MODE: /*specified fixed, but not what size */
|
||||
st->flags |= ST_FIXEDBLOCKS;
|
||||
|
@ -786,7 +786,7 @@ st_decide_mode(unit, first_read)
|
|||
st->blksiz = st->media_blksiz;
|
||||
else
|
||||
st->blksiz = DEF_FIXED_BSIZE;
|
||||
SC_DEBUG(sc_link, SDEV_DB3, ("Quirks force fixed mode(%d)\n",
|
||||
SC_DEBUG(sc_link, SDEV_DB3, ("Quirks force fixed mode(%ld)\n",
|
||||
st->blksiz));
|
||||
goto done;
|
||||
case ST_Q_FORCE_VAR_MODE:
|
||||
|
@ -803,7 +803,7 @@ st_decide_mode(unit, first_read)
|
|||
st->flags |= ST_FIXEDBLOCKS;
|
||||
st->blksiz = st->blkmin;
|
||||
SC_DEBUG(sc_link, SDEV_DB3,
|
||||
("blkmin == blkmax of %d\n", st->blkmin));
|
||||
("blkmin == blkmax of %ld\n", st->blkmin));
|
||||
goto done;
|
||||
}
|
||||
/*
|
||||
|
@ -851,7 +851,7 @@ st_decide_mode(unit, first_read)
|
|||
}
|
||||
st->blksiz = st->media_blksiz;
|
||||
SC_DEBUG(sc_link, SDEV_DB3,
|
||||
("Used media_blksiz of %d\n", st->media_blksiz));
|
||||
("Used media_blksiz of %ld\n", st->media_blksiz));
|
||||
goto done;
|
||||
}
|
||||
/*
|
||||
|
@ -897,7 +897,7 @@ void
|
|||
st_strategy(struct buf *bp, struct scsi_link *sc_link)
|
||||
{
|
||||
struct buf **dp;
|
||||
unsigned char unit;
|
||||
unsigned char unit; /* XXX Everywhere else unit is "u_int32". Please int? */
|
||||
u_int32 opri;
|
||||
struct scsi_data *st;
|
||||
|
||||
|
@ -915,7 +915,7 @@ st_strategy(struct buf *bp, struct scsi_link *sc_link)
|
|||
*/
|
||||
if (st->flags & ST_FIXEDBLOCKS) {
|
||||
if (bp->b_bcount % st->blksiz) {
|
||||
printf("st%d: bad request, must be multiple of %d\n",
|
||||
printf("st%d: bad request, must be multiple of %ld\n",
|
||||
unit, st->blksiz);
|
||||
bp->b_error = EIO;
|
||||
goto bad;
|
||||
|
@ -925,7 +925,7 @@ st_strategy(struct buf *bp, struct scsi_link *sc_link)
|
|||
* as are out-of-range requests on variable drives.
|
||||
*/
|
||||
else if (bp->b_bcount < st->blkmin || bp->b_bcount > st->blkmax) {
|
||||
printf("st%d: bad request, must be between %d and %d\n",
|
||||
printf("st%d: bad request, must be between %ld and %ld\n",
|
||||
unit, st->blkmin, st->blkmax);
|
||||
bp->b_error = EIO;
|
||||
goto bad;
|
||||
|
@ -1102,7 +1102,7 @@ ststart(unit)
|
|||
stqueues++;
|
||||
} else {
|
||||
badnews:
|
||||
printf("st%d: oops not queued\n", unit);
|
||||
printf("st%ld: oops not queued\n", unit);
|
||||
bp->b_flags |= B_ERROR;
|
||||
bp->b_error = EIO;
|
||||
biodone(bp);
|
||||
|
@ -1166,7 +1166,7 @@ struct proc *p, struct scsi_link *sc_link)
|
|||
case MTIOCTOP:
|
||||
{
|
||||
|
||||
SC_DEBUG(sc_link, SDEV_DB1, ("[ioctl: op=0x%x count=0x%x]\n",
|
||||
SC_DEBUG(sc_link, SDEV_DB1, ("[ioctl: op=0x%x count=0x%lx]\n",
|
||||
mt->mt_op, mt->mt_count));
|
||||
|
||||
/* compat: in U*x it is a short */
|
||||
|
@ -1259,7 +1259,7 @@ struct proc *p, struct scsi_link *sc_link)
|
|||
* Check that the mode being asked for is aggreeable to the
|
||||
* drive. If not, put it back the way it was.
|
||||
*/
|
||||
if (errcode = st_mode_select(unit, 0, NULL, 0)) { /* put it back as it was */
|
||||
if ( (errcode = st_mode_select(unit, 0, NULL, 0)) ) { /* put back as it was */
|
||||
printf("st%d: Cannot set selected mode", unit);
|
||||
st->density = hold_density;
|
||||
st->blksiz = hold_blksiz;
|
||||
|
@ -1364,7 +1364,7 @@ st_rd_blk_lim(unit, flags)
|
|||
/*
|
||||
* do the command, update the global values
|
||||
*/
|
||||
if (errno = scsi_scsi_cmd(sc_link,
|
||||
if ( (errno = scsi_scsi_cmd(sc_link,
|
||||
(struct scsi_generic *) &scsi_cmd,
|
||||
sizeof(scsi_cmd),
|
||||
(u_char *) & scsi_blkl,
|
||||
|
@ -1372,14 +1372,14 @@ st_rd_blk_lim(unit, flags)
|
|||
ST_RETRIES,
|
||||
5000,
|
||||
NULL,
|
||||
flags | SCSI_DATA_IN)) {
|
||||
flags | SCSI_DATA_IN)) ) {
|
||||
return errno;
|
||||
}
|
||||
st->blkmin = b2tol(scsi_blkl.min_length);
|
||||
st->blkmax = scsi_3btou(&scsi_blkl.max_length_2);
|
||||
|
||||
SC_DEBUG(sc_link, SDEV_DB3,
|
||||
("(%d <= blksiz <= %d)\n", st->blkmin, st->blkmax));
|
||||
("(%ld <= blksiz <= %ld)\n", st->blkmin, st->blkmax));
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
@ -1446,7 +1446,7 @@ st_mode_sense(unit, flags, page, pagelen, pagecode)
|
|||
* or if we need it as a template for the mode select
|
||||
* store it away.
|
||||
*/
|
||||
if (errno = scsi_scsi_cmd(sc_link,
|
||||
if ( (errno = scsi_scsi_cmd(sc_link,
|
||||
(struct scsi_generic *) &scsi_cmd,
|
||||
sizeof(scsi_cmd),
|
||||
(u_char *) &dat,
|
||||
|
@ -1454,7 +1454,7 @@ st_mode_sense(unit, flags, page, pagelen, pagecode)
|
|||
ST_RETRIES,
|
||||
5000,
|
||||
NULL,
|
||||
flags | SCSI_DATA_IN)) {
|
||||
flags | SCSI_DATA_IN)) ) {
|
||||
return errno;
|
||||
}
|
||||
st->numblks = scsi_3btou(dat.blk_desc.nblocks);
|
||||
|
@ -1464,7 +1464,7 @@ st_mode_sense(unit, flags, page, pagelen, pagecode)
|
|||
st->flags |= ST_READONLY;
|
||||
}
|
||||
SC_DEBUG(sc_link, SDEV_DB3,
|
||||
("density code 0x%x, %d-byte blocks, write-%s, ",
|
||||
("density code 0x%lx, %ld-byte blocks, write-%s, ",
|
||||
st->media_density, st->media_blksiz,
|
||||
st->flags & ST_READONLY ? "protected" : "enabled"));
|
||||
SC_DEBUG(sc_link, SDEV_DB3, ("%sbuffered\n",
|
||||
|
@ -1559,14 +1559,14 @@ u_int32 unit,mode;
|
|||
bzero(&page, sizeof(page));
|
||||
pagesize = sizeof(page.pages.configuration) + PAGE_HEADERLEN;
|
||||
|
||||
if ( retval = st_mode_sense(unit, 0,
|
||||
&page, pagesize, ST_PAGE_CONFIGURATION))
|
||||
if ( (retval = st_mode_sense(unit, 0,
|
||||
&page, pagesize, ST_PAGE_CONFIGURATION)) )
|
||||
{
|
||||
printf("sense returned an error of %d\n",retval);
|
||||
return retval;
|
||||
}
|
||||
if ( noisy_st)
|
||||
printf("drive reports value of %d, setting %d\n",
|
||||
printf("drive reports value of %d, setting %ld\n",
|
||||
page.pages.configuration.data_compress_alg,mode);
|
||||
|
||||
page.pg_code &= ST_P_CODE;
|
||||
|
@ -1581,10 +1581,10 @@ u_int32 unit,mode;
|
|||
page.pages.configuration.data_compress_alg = 1;
|
||||
break;
|
||||
default:
|
||||
printf("st%d: bad value for compression mode\n",unit);
|
||||
printf("st%ld: bad value for compression mode\n",unit);
|
||||
return EINVAL;
|
||||
}
|
||||
if ( retval = st_mode_select(unit, 0, &page, pagesize))
|
||||
if ( (retval = st_mode_select(unit, 0, &page, pagesize)) )
|
||||
{
|
||||
printf("select returned an error of %d\n",retval);
|
||||
return retval;
|
||||
|
@ -1859,12 +1859,6 @@ st_erase(unit, immed, flags)
|
|||
flags));
|
||||
}
|
||||
|
||||
#ifdef NETBSD
|
||||
#define SIGNAL_SHORT_READ
|
||||
#else
|
||||
#define SIGNAL_SHORT_READ bp->b_flags |= B_ERROR;
|
||||
#endif
|
||||
|
||||
/*
|
||||
* Look at the returned sense and act on the error and detirmine
|
||||
* The unix error number to pass back... (0 = report no error)
|
||||
|
@ -1877,7 +1871,6 @@ st_interpret_sense(xs)
|
|||
struct scsi_link *sc_link = xs->sc_link;
|
||||
struct scsi_sense_data *sense = &(xs->sense);
|
||||
boolean silent = xs->flags & SCSI_SILENT;
|
||||
struct buf *bp = xs->bp;
|
||||
u_int32 unit = sc_link->dev_unit;
|
||||
struct scsi_data *st = SCSI_DATA(&st_switch, unit);
|
||||
u_int32 key;
|
||||
|
@ -1889,97 +1882,92 @@ st_interpret_sense(xs)
|
|||
if (sense->error_code & SSD_ERRCODE_VALID) {
|
||||
info = ntohl(*((int32 *) sense->ext.extended.info));
|
||||
} else {
|
||||
info = xs->datalen; /* bad choice if fixed blocks */
|
||||
if (st->flags & ST_FIXEDBLOCKS) {
|
||||
info = xs->datalen / st->blksiz;
|
||||
} else {
|
||||
info = xs->datalen;
|
||||
}
|
||||
}
|
||||
if ((sense->error_code & SSD_ERRCODE) != 0x70) {
|
||||
return SCSIRET_CONTINUE; /* let the generic code handle it */
|
||||
return SCSIRET_CONTINUE;/* let the generic code handle it */
|
||||
}
|
||||
if (st->flags & ST_FIXEDBLOCKS) {
|
||||
xs->resid = info * st->blksiz;
|
||||
if (sense->ext.extended.flags & SSD_EOM) {
|
||||
st->flags |= ST_EIO_PENDING;
|
||||
if (bp) {
|
||||
bp->b_resid = xs->resid;
|
||||
SIGNAL_SHORT_READ
|
||||
if(sense->ext.extended.flags & (SSD_EOM|SSD_FILEMARK|SSD_ILI)) {
|
||||
if (st->flags & ST_FIXEDBLOCKS) {
|
||||
xs->resid = info * st->blksiz;
|
||||
xs->flags |= SCSI_RESID_VALID;
|
||||
if (sense->ext.extended.flags & SSD_EOM) {
|
||||
st->flags |= ST_EIO_PENDING;
|
||||
}
|
||||
}
|
||||
if (sense->ext.extended.flags & SSD_FILEMARK) {
|
||||
st->flags |= ST_AT_FILEMARK;
|
||||
if (bp) {
|
||||
bp->b_resid = xs->resid;
|
||||
SIGNAL_SHORT_READ
|
||||
if (sense->ext.extended.flags & SSD_FILEMARK) {
|
||||
st->flags |= ST_AT_FILEMARK;
|
||||
}
|
||||
}
|
||||
if (sense->ext.extended.flags & SSD_ILI) {
|
||||
st->flags |= ST_EIO_PENDING;
|
||||
if (bp) {
|
||||
bp->b_resid = xs->resid;
|
||||
SIGNAL_SHORT_READ
|
||||
}
|
||||
if (sense->error_code & SSD_ERRCODE_VALID &&
|
||||
!silent)
|
||||
printf("st%d: block wrong size"
|
||||
", %d blocks residual\n", unit
|
||||
,info);
|
||||
|
||||
/*
|
||||
* This quirk code helps the drive read
|
||||
* the first tape block, regardless of
|
||||
* format. That is required for these
|
||||
* drives to return proper MODE SENSE
|
||||
* information.
|
||||
*/
|
||||
if ((st->quirks & ST_Q_SNS_HLP) &&
|
||||
!(sc_link->flags & SDEV_MEDIA_LOADED)) {
|
||||
st->blksiz -= 512;
|
||||
}
|
||||
}
|
||||
/*
|
||||
* If no data was tranfered, do it immediatly
|
||||
*/
|
||||
if (xs->resid >= xs->datalen) {
|
||||
if (st->flags & ST_EIO_PENDING) {
|
||||
return EIO;
|
||||
}
|
||||
if (st->flags & ST_AT_FILEMARK) {
|
||||
if (bp) {
|
||||
bp->b_resid = xs->resid;
|
||||
SIGNAL_SHORT_READ
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
} else { /* must be variable mode */
|
||||
xs->resid = xs->datalen; /* to be sure */
|
||||
if (sense->ext.extended.flags & SSD_EOM) {
|
||||
return (EIO);
|
||||
}
|
||||
if (sense->ext.extended.flags & SSD_FILEMARK) {
|
||||
if (bp)
|
||||
bp->b_resid = bp->b_bcount;
|
||||
return 0;
|
||||
}
|
||||
if (sense->ext.extended.flags & SSD_ILI) {
|
||||
if (info < 0) {
|
||||
if (sense->ext.extended.flags & SSD_ILI) {
|
||||
st->flags |= ST_EIO_PENDING;
|
||||
if (sense->error_code & SSD_ERRCODE_VALID &&
|
||||
!silent)
|
||||
printf("st%ld: block wrong size"
|
||||
", %ld blocks residual\n", unit
|
||||
,info);
|
||||
/*XXX*/ /* is this how it works ? */
|
||||
/* check def of ILI for fixed blk tapes */
|
||||
|
||||
/*
|
||||
* the record was bigger than the read
|
||||
*/
|
||||
if (!silent)
|
||||
printf("st%d: %d-byte record "
|
||||
"too big\n", unit,
|
||||
xs->datalen - info);
|
||||
* This quirk code helps the drive read
|
||||
* the first tape block, regardless of
|
||||
* format. That is required for these
|
||||
* drives to return proper MODE SENSE
|
||||
* information.
|
||||
*/
|
||||
if ((st->quirks & ST_Q_SNS_HLP) &&
|
||||
!(sc_link->flags & SDEV_MEDIA_LOADED)) {
|
||||
st->blksiz -= 512;
|
||||
}
|
||||
}
|
||||
/*
|
||||
* If no data was tranfered, do it immediatly
|
||||
*/
|
||||
if (xs->resid >= xs->datalen) {
|
||||
xs->flags &= ~SCSI_RESID_VALID;
|
||||
if (st->flags & ST_AT_FILEMARK) {
|
||||
xs->flags |= SCSI_EOF;
|
||||
st->flags &= ~ST_AT_FILEMARK;
|
||||
return 0;
|
||||
}
|
||||
if (st->flags & ST_EIO_PENDING) {
|
||||
st->flags &= ~ST_EIO_PENDING;
|
||||
return EIO;
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
} else { /* must be variable mode */
|
||||
xs->resid = xs->datalen; /* to be sure */
|
||||
if (sense->ext.extended.flags & SSD_EOM) {
|
||||
return (EIO);
|
||||
}
|
||||
xs->resid = info;
|
||||
if (bp) {
|
||||
bp->b_resid = info;
|
||||
SIGNAL_SHORT_READ
|
||||
if (sense->ext.extended.flags & SSD_FILEMARK) {
|
||||
xs->flags |= SCSI_EOF;
|
||||
}
|
||||
if (sense->ext.extended.flags & SSD_ILI) {
|
||||
if (info < 0) {
|
||||
/*
|
||||
* the record was bigger than the read
|
||||
*/
|
||||
if (!silent)
|
||||
printf("st%ld: %ld-byte record "
|
||||
"too big\n", unit,
|
||||
xs->datalen - info);
|
||||
return (EIO);
|
||||
}
|
||||
xs->resid = info;
|
||||
xs->flags |= SCSI_RESID_VALID;
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
key = sense->ext.extended.flags & SSD_KEY;
|
||||
|
||||
if (key == 0x8) {
|
||||
xs->flags |= SCSI_EOF; /* some drives need this */
|
||||
/*
|
||||
* This quirk code helps the drive read the
|
||||
* first tape block, regardless of format. That
|
||||
|
@ -1987,19 +1975,16 @@ st_interpret_sense(xs)
|
|||
* MODE SENSE information.
|
||||
*/
|
||||
if ((st->quirks & ST_Q_SNS_HLP) &&
|
||||
!(sc_link->flags & SDEV_MEDIA_LOADED)) { /* still starting */
|
||||
!(sc_link->flags & SDEV_MEDIA_LOADED)) {
|
||||
/* still starting */
|
||||
st->blksiz -= 512;
|
||||
} else if (!(st->flags & (ST_2FM_AT_EOD | ST_BLANK_READ))) {
|
||||
st->flags |= ST_BLANK_READ;
|
||||
xs->resid = xs->datalen;
|
||||
if (bp) {
|
||||
bp->b_resid = xs->resid;
|
||||
/*return an EOF */
|
||||
}
|
||||
xs->flags |= SCSI_EOF;
|
||||
return (ESUCCESS);
|
||||
}
|
||||
}
|
||||
return SCSIRET_CONTINUE; /* let the default/generic handler handle it */
|
||||
return SCSIRET_CONTINUE; /* Use the the generic handler */
|
||||
}
|
||||
|
||||
/*
|
||||
|
@ -2031,7 +2016,7 @@ st_touch_tape(unit)
|
|||
if (!buf)
|
||||
return (ENOMEM);
|
||||
|
||||
if (errno = st_mode_sense(unit, 0, NULL, 0, 0)) {
|
||||
if (( errno = st_mode_sense(unit, 0, NULL, 0, 0)) ) {
|
||||
goto bad;
|
||||
}
|
||||
st->blksiz = 1024;
|
||||
|
@ -2045,11 +2030,11 @@ st_touch_tape(unit)
|
|||
default:
|
||||
readsiz = 1;
|
||||
st->flags &= ~ST_FIXEDBLOCKS;
|
||||
} if (errno = st_mode_select(unit, 0, NULL, 0)) {
|
||||
} if ( (errno = st_mode_select(unit, 0, NULL, 0)) ) {
|
||||
goto bad;
|
||||
}
|
||||
st_read(unit, buf, readsiz, SCSI_SILENT);
|
||||
if (errno = st_rewind(unit, FALSE, 0)) {
|
||||
if ( (errno = st_rewind(unit, FALSE, 0)) ) {
|
||||
bad: free(buf, M_TEMP);
|
||||
return (errno);
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue