summaryrefslogtreecommitdiff
path: root/sys/dev
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev')
-rw-r--r--sys/dev/ic/ncr53c9x.c1213
-rw-r--r--sys/dev/ic/ncr53c9xreg.h26
-rw-r--r--sys/dev/ic/ncr53c9xvar.h127
-rw-r--r--sys/dev/pci/pcscp.c4
-rw-r--r--sys/dev/sbus/esp_sbus.c7
5 files changed, 1025 insertions, 352 deletions
diff --git a/sys/dev/ic/ncr53c9x.c b/sys/dev/ic/ncr53c9x.c
index 6f1d3056902..aefb82b5c5c 100644
--- a/sys/dev/ic/ncr53c9x.c
+++ b/sys/dev/ic/ncr53c9x.c
@@ -1,5 +1,5 @@
-/* $OpenBSD: ncr53c9x.c,v 1.9 2000/07/21 11:20:35 art Exp $ */
-/* $NetBSD: ncr53c9x.c,v 1.26 1998/05/26 23:17:34 thorpej Exp $ */
+/* $OpenBSD: ncr53c9x.c,v 1.10 2001/09/27 04:01:42 jason Exp $ */
+/* $NetBSD: ncr53c9x.c,v 1.56 2000/11/30 14:41:46 thorpej Exp $ */
/*
* Copyright (c) 1996 Charles M. Hannum. All rights reserved.
@@ -82,6 +82,7 @@
#include <sys/proc.h>
#include <sys/user.h>
#include <sys/queue.h>
+#include <sys/pool.h>
#include <scsi/scsi_all.h>
#include <scsi/scsiconf.h>
@@ -93,13 +94,15 @@
#include <dev/ic/ncr53c9xvar.h>
int ncr53c9x_debug = 0; /*NCR_SHOWPHASE|NCR_SHOWMISC|NCR_SHOWTRAC|NCR_SHOWCMDS;*/
+#ifdef DEBUG
+int ncr53c9x_notag = 0;
+#endif
/*static*/ void ncr53c9x_readregs __P((struct ncr53c9x_softc *));
/*static*/ void ncr53c9x_select __P((struct ncr53c9x_softc *,
struct ncr53c9x_ecb *));
-/*static*/ int ncr53c9x_reselect __P((struct ncr53c9x_softc *, int));
+/*static*/ int ncr53c9x_reselect __P((struct ncr53c9x_softc *, int, int, int));
/*static*/ void ncr53c9x_scsi_reset __P((struct ncr53c9x_softc *));
-/*static*/ void ncr53c9x_init __P((struct ncr53c9x_softc *, int));
/*static*/ int ncr53c9x_poll __P((struct ncr53c9x_softc *,
struct scsi_xfer *, int));
/*static*/ void ncr53c9x_sched __P((struct ncr53c9x_softc *));
@@ -108,6 +111,7 @@ int ncr53c9x_debug = 0; /*NCR_SHOWPHASE|NCR_SHOWMISC|NCR_SHOWTRAC|NCR_SHOWCMDS;*
/*static*/ void ncr53c9x_msgin __P((struct ncr53c9x_softc *));
/*static*/ void ncr53c9x_msgout __P((struct ncr53c9x_softc *));
/*static*/ void ncr53c9x_timeout __P((void *arg));
+/*static*/ void ncr53c9x_watch __P((void *arg));
/*static*/ void ncr53c9x_abort __P((struct ncr53c9x_softc *,
struct ncr53c9x_ecb *));
/*static*/ void ncr53c9x_dequeue __P((struct ncr53c9x_softc *,
@@ -122,6 +126,28 @@ struct ncr53c9x_ecb *ncr53c9x_get_ecb __P((struct ncr53c9x_softc *, int));
static inline int ncr53c9x_stp2cpb __P((struct ncr53c9x_softc *, int));
static inline void ncr53c9x_setsync __P((struct ncr53c9x_softc *,
struct ncr53c9x_tinfo *));
+static struct ncr53c9x_linfo *ncr53c9x_lunsearch __P((struct ncr53c9x_tinfo *,
+ int64_t lun));
+
+static void ncr53c9x_wrfifo(struct ncr53c9x_softc *, u_char *, int);
+static int ncr53c9x_rdfifo(struct ncr53c9x_softc *, int);
+#define NCR_RDFIFO_START 0
+#define NCR_RDFIFO_CONTINUE 1
+
+#define NCR_SET_COUNT(sc, size) do { \
+ NCR_WRITE_REG((sc), NCR_TCL, (size)); \
+ NCR_WRITE_REG((sc), NCR_TCM, (size) >> 8); \
+ if ((sc->sc_cfg2 & NCRCFG2_FE) || \
+ (sc->sc_rev == NCR_VARIANT_FAS366)) { \
+ NCR_WRITE_REG((sc), NCR_TCH, (size) >> 16); \
+ } \
+ if (sc->sc_rev == NCR_VARIANT_FAS366) { \
+ NCR_WRITE_REG(sc, NCR_RCH, 0); \
+ } \
+} while (0)
+
+static int ecb_pool_initialized = 0;
+static struct pool ecb_pool;
struct cfdriver esp_cd = {
NULL, "esp", DV_DULL
@@ -141,9 +167,25 @@ const char *ncr53c9x_variant_names[] = {
"FAS408",
"FAS216",
"AM53C974",
+ "FAS366/HME",
};
/*
+ * Search linked list for LUN info by LUN id.
+ */
+static struct ncr53c9x_linfo *
+ncr53c9x_lunsearch(ti, lun)
+ struct ncr53c9x_tinfo *ti;
+ int64_t lun;
+{
+ struct ncr53c9x_linfo *li;
+ LIST_FOREACH(li, &ti->luns, link)
+ if (li->lun == lun)
+ return (li);
+ return (NULL);
+}
+
+/*
* Attach this instance, and then all the sub-devices
*/
void
@@ -152,7 +194,7 @@ ncr53c9x_attach(sc, adapter, dev)
struct scsi_adapter *adapter;
struct scsi_device *dev;
{
-
+ timeout_set(&sc->sc_watchdog, ncr53c9x_watch, sc);
/*
* Allocate SCSI message buffers.
* Front-ends can override allocation to avoid alignment
@@ -222,11 +264,12 @@ ncr53c9x_attach(sc, adapter, dev)
* Now try to attach all the sub-devices
*/
config_found(&sc->sc_dev, &sc->sc_link, scsiprint);
+ timeout_add(&sc->sc_watchdog, 60*hz);
}
/*
- * This is the generic esp reset function. It does not reset the SCSI bus,
- * only this controllers, but kills any on-going commands, and also stops
+ * This is the generic ncr53c9x reset function. It does not reset the SCSI bus,
+ * only this controller, but kills any on-going commands, and also stops
* and resets the DMA.
*
* After reset, registers are loaded with the defaults from the attach
@@ -259,6 +302,7 @@ ncr53c9x_reset(sc)
sc->sc_features |= NCR_F_HASCFG3;
NCR_WRITE_REG(sc, NCR_CFG3, sc->sc_cfg3);
case NCR_VARIANT_ESP100A:
+ sc->sc_features |= NCR_F_SELATN3;
NCR_WRITE_REG(sc, NCR_CFG2, sc->sc_cfg2);
case NCR_VARIANT_ESP100:
NCR_WRITE_REG(sc, NCR_CFG1, sc->sc_cfg1);
@@ -266,6 +310,19 @@ ncr53c9x_reset(sc)
NCR_WRITE_REG(sc, NCR_SYNCOFF, 0);
NCR_WRITE_REG(sc, NCR_TIMEOUT, sc->sc_timeout);
break;
+ case NCR_VARIANT_FAS366:
+ sc->sc_features |=
+ NCR_F_SELATN3 | NCR_F_HASCFG3 | NCR_F_FASTSCSI;
+ sc->sc_cfg3 = NCRFASCFG3_FASTCLK | NCRFASCFG3_OBAUTO;
+ sc->sc_cfg3_fscsi = NCRFASCFG3_FASTSCSI;
+ NCR_WRITE_REG(sc, NCR_CFG3, sc->sc_cfg3);
+ sc->sc_cfg2 = 0; /* NCRCFG2_HMEFE | NCRCFG2_HME32 */
+ NCR_WRITE_REG(sc, NCR_CFG2, sc->sc_cfg2);
+ NCR_WRITE_REG(sc, NCR_CFG1, sc->sc_cfg1);
+ NCR_WRITE_REG(sc, NCR_CCF, sc->sc_ccf);
+ NCR_WRITE_REG(sc, NCR_SYNCOFF, 0);
+ NCR_WRITE_REG(sc, NCR_TIMEOUT, sc->sc_timeout);
+ break;
default:
printf("%s: unknown revision code, assuming ESP100\n",
sc->sc_dev.dv_xname);
@@ -277,6 +334,15 @@ ncr53c9x_reset(sc)
if (sc->sc_rev == NCR_VARIANT_AM53C974)
NCR_WRITE_REG(sc, NCR_AMDCFG4, sc->sc_cfg4);
+
+#if 0
+ printf("%s: ncr53c9x_reset: revision %d\n",
+ sc->sc_dev.dv_xname, sc->sc_rev);
+ printf("%s: ncr53c9x_reset: cfg1 0x%x, cfg2 0x%x, cfg3 0x%x, ccf 0x%x, timeout 0x%x\n",
+ sc->sc_dev.dv_xname,
+ sc->sc_cfg1, sc->sc_cfg2, sc->sc_cfg3,
+ sc->sc_ccf, sc->sc_timeout);
+#endif
}
/*
@@ -294,7 +360,7 @@ ncr53c9x_scsi_reset(sc)
}
/*
- * Initialize esp state machine
+ * Initialize ncr53c9x state machine
*/
void
ncr53c9x_init(sc, doreset)
@@ -302,34 +368,56 @@ ncr53c9x_init(sc, doreset)
int doreset;
{
struct ncr53c9x_ecb *ecb;
- int r;
+ struct ncr53c9x_linfo *li;
+ int r, i;
- NCR_TRACE(("[NCR_INIT(%d)] ", doreset));
+ NCR_TRACE(("[NCR_INIT(%d) %d] ", doreset, sc->sc_state));
+
+ if (!ecb_pool_initialized) {
+ /* All instances share this pool */
+ pool_init(&ecb_pool, sizeof(struct ncr53c9x_ecb), 0, 0, 0,
+ "ncr53c9x_ecb", 0, NULL, NULL, 0);
+ ecb_pool_initialized = 1;
+ }
if (sc->sc_state == 0) {
/* First time through; initialize. */
+
TAILQ_INIT(&sc->ready_list);
- TAILQ_INIT(&sc->nexus_list);
- TAILQ_INIT(&sc->free_list);
sc->sc_nexus = NULL;
- ecb = sc->sc_ecb;
- bzero(ecb, sizeof(sc->sc_ecb));
- for (r = 0; r < sizeof(sc->sc_ecb) / sizeof(*ecb); r++) {
- timeout_set(&ecb->to, ncr53c9x_timeout, ecb);
- TAILQ_INSERT_TAIL(&sc->free_list, ecb, chain);
- ecb++;
- }
bzero(sc->sc_tinfo, sizeof(sc->sc_tinfo));
+ for (r = 0; r < NCR_NTARG; r++) {
+ LIST_INIT(&sc->sc_tinfo[r].luns);
+ }
} else {
/* Cancel any active commands. */
sc->sc_state = NCR_CLEANING;
+ sc->sc_msgify = 0;
if ((ecb = sc->sc_nexus) != NULL) {
ecb->xs->error = XS_TIMEOUT;
ncr53c9x_done(sc, ecb);
}
- while ((ecb = sc->nexus_list.tqh_first) != NULL) {
- ecb->xs->error = XS_TIMEOUT;
- ncr53c9x_done(sc, ecb);
+ for (r = 0; r < 8; r++) {
+ LIST_FOREACH(li, &sc->sc_tinfo[r].luns, link) {
+ if ((ecb = li->untagged)) {
+ li->untagged = NULL;
+ /*
+ * XXXXXXX
+ * Should we terminate a command
+ * that never reached the disk?
+ */
+ li->busy = 0;
+ ecb->xs->error = XS_TIMEOUT;
+ ncr53c9x_done(sc, ecb);
+ }
+ for (i = 0; i<256; i++)
+ if ((ecb = li->queued[i])) {
+ li->queued[i] = NULL;
+ ecb->xs->error = XS_TIMEOUT;
+ ncr53c9x_done(sc, ecb);
+ }
+ li->used = 0;
+ }
}
}
@@ -343,12 +431,17 @@ ncr53c9x_init(sc, doreset)
struct ncr53c9x_tinfo *ti = &sc->sc_tinfo[r];
/* XXX - config flags per target: low bits: no reselect; high bits: no synch */
- ti->flags = ((sc->sc_minsync && !(sc->sc_cfflags & (1<<(r+8))))
- ? T_NEGOTIATE : 0) |
- ((sc->sc_cfflags & (1<<r)) ? T_RSELECTOFF : 0) |
- T_NEED_TO_RESET;
+ ti->flags = ((sc->sc_minsync && !(sc->sc_cfflags & (1<<(r+8))))
+ ? 0 : T_SYNCHOFF) |
+ ((sc->sc_cfflags & (1<<r)) ? T_RSELECTOFF : 0) |
+ T_NEED_TO_RESET;
+#ifdef DEBUG
+ if (ncr53c9x_notag)
+ ti->flags &= ~T_TAG;
+#endif
ti->period = sc->sc_minsync;
ti->offset = 0;
+ ti->cfg3 = 0;
}
if (doreset) {
@@ -377,6 +470,10 @@ ncr53c9x_readregs(sc)
sc->sc_espstat = NCR_READ_REG(sc, NCR_STAT);
/* Only the stepo bits are of interest */
sc->sc_espstep = NCR_READ_REG(sc, NCR_STEP) & NCRSTEP_MASK;
+
+ if (sc->sc_rev == NCR_VARIANT_FAS366)
+ sc->sc_espstat2 = NCR_READ_REG(sc, NCR_STAT2);
+
sc->sc_espintr = NCR_READ_REG(sc, NCR_INTR);
if (sc->sc_glue->gl_clear_latched_intr != NULL)
@@ -391,8 +488,8 @@ ncr53c9x_readregs(sc)
? /* Disconnected */ BUSFREE_PHASE
: sc->sc_espstat & NCRSTAT_PHASE;
- NCR_MISC(("regs[intr=%02x,stat=%02x,step=%02x] ",
- sc->sc_espintr, sc->sc_espstat, sc->sc_espstep));
+ NCR_MISC(("regs[intr=%02x,stat=%02x,step=%02x,stat2=%02x] ",
+ sc->sc_espintr, sc->sc_espstat, sc->sc_espstep, sc->sc_espstat2));
}
/*
@@ -416,7 +513,8 @@ ncr53c9x_setsync(sc, ti)
struct ncr53c9x_softc *sc;
struct ncr53c9x_tinfo *ti;
{
- u_char syncoff, synctp, cfg3 = sc->sc_cfg3;
+ u_char syncoff, synctp;
+ u_char cfg3 = sc->sc_cfg3 | ti->cfg3;
if (ti->flags & T_SYNCMODE) {
syncoff = ti->offset;
@@ -434,7 +532,8 @@ ncr53c9x_setsync(sc, ti)
* chip into Fast SCSI mode so that it doesn't
* have to be figured out here each time.
*/
- cfg3 |= sc->sc_cfg3_fscsi;
+ cfg3 |= (sc->sc_rev == NCR_VARIANT_AM53C974) ?
+ NCRAMDCFG3_FSCSI : NCRCFG3_FSCSI;
}
/*
@@ -456,7 +555,6 @@ ncr53c9x_setsync(sc, ti)
NCR_WRITE_REG(sc, NCR_SYNCTP, synctp);
}
-int ncr53c9x_dmaselect = 0;
/*
* Send a command to a target, set the driver state to NCR_SELECTING
* and let the caller take care of the rest.
@@ -476,89 +574,123 @@ ncr53c9x_select(sc, ecb)
int tiflags = ti->flags;
u_char *cmd;
int clen;
+ int selatn3, selatns;
size_t dmasize;
- NCR_TRACE(("[ncr53c9x_select(t%d,l%d,cmd:%x)] ",
- target, lun, ecb->cmd.cmd.opcode));
+ NCR_TRACE(("[ncr53c9x_select(t%d,l%d,cmd:%x,tag%x,%x)] ",
+ target, lun, ecb->cmd.cmd.opcode, ecb->tag[0], ecb->tag[1]));
sc->sc_state = NCR_SELECTING;
-
/*
* Schedule the timeout now, the first time we will go away
* expecting to come back due to an interrupt, because it is
* always possible that the interrupt may never happen.
*/
- if ((ecb->xs->flags & SCSI_POLL) == 0)
- timeout_add(&ecb->to, (ecb->timeout * hz) / 1000);
+ if ((ecb->xs->flags & SCSI_POLL) == 0) {
+ int timeout = ecb->timeout;
+
+ if (hz > 100 && timeout > 1000)
+ timeout = (timeout / 1000) * hz;
+ else
+ timeout = (timeout * hz) / 1000;
+ timeout_add(&ecb->to, timeout);
+ }
/*
* The docs say the target register is never reset, and I
* can't think of a better place to set it
*/
- NCR_WRITE_REG(sc, NCR_SELID, target);
+ if (sc->sc_rev == NCR_VARIANT_FAS366) {
+ NCRCMD(sc, NCRCMD_FLUSH);
+ NCR_WRITE_REG(sc, NCR_SELID, target | NCR_BUSID_HME);
+ } else {
+ NCR_WRITE_REG(sc, NCR_SELID, target);
+ }
ncr53c9x_setsync(sc, ti);
- if (ecb->flags & ECB_SENSE) {
+ if ((ecb->flags & ECB_SENSE) != 0) {
/*
* For REQUEST SENSE, we should not send an IDENTIFY or
* otherwise mangle the target. There should be no MESSAGE IN
* phase.
*/
- if (ncr53c9x_dmaselect) {
+ if (sc->sc_features & NCR_F_DMASELECT) {
/* setup DMA transfer for command */
dmasize = clen = ecb->clen;
sc->sc_cmdlen = clen;
- sc->sc_cmdp = (caddr_t)&ecb->cmd + 1;
- NCRDMA_SETUP(sc, &sc->sc_cmdp, &sc->sc_cmdlen, 0,
- &dmasize);
+ sc->sc_cmdp = (caddr_t)&ecb->cmd.cmd;
/* Program the SCSI counter */
- NCR_WRITE_REG(sc, NCR_TCL, dmasize);
- NCR_WRITE_REG(sc, NCR_TCM, dmasize >> 8);
- if (sc->sc_cfg2 & NCRCFG2_FE) {
- NCR_WRITE_REG(sc, NCR_TCH, dmasize >> 16);
- }
+ NCR_SET_COUNT(sc, dmasize);
- /* load the count in */
- NCRCMD(sc, NCRCMD_NOP|NCRCMD_DMA);
+ if (sc->sc_rev != NCR_VARIANT_FAS366)
+ NCRCMD(sc, NCRCMD_NOP|NCRCMD_DMA);
/* And get the targets attention */
NCRCMD(sc, NCRCMD_SELNATN | NCRCMD_DMA);
+ NCRDMA_SETUP(sc, &sc->sc_cmdp, &sc->sc_cmdlen, 0, &dmasize);
NCRDMA_GO(sc);
} else {
- /* Now the command into the FIFO */
- cmd = (u_char *)&ecb->cmd.cmd;
- clen = ecb->clen;
- while (clen--)
- NCR_WRITE_REG(sc, NCR_FIFO, *cmd++);
-
+ ncr53c9x_wrfifo(sc, (u_char *)&ecb->cmd.cmd, ecb->clen);
NCRCMD(sc, NCRCMD_SELNATN);
}
return;
}
- if (ncr53c9x_dmaselect && (tiflags & T_NEGOTIATE) == 0) {
- ecb->cmd.id =
- MSG_IDENTIFY(lun, (tiflags & T_RSELECTOFF)?0:1);
+ selatn3 = selatns = 0;
+ if (ecb->tag[0] != 0) {
+ if (sc->sc_features & NCR_F_SELATN3)
+ /* use SELATN3 to send tag messages */
+ selatn3 = 1;
+ else
+ /* We don't have SELATN3; use SELATNS to send tags */
+ selatns = 1;
+ }
+
+ if (ti->flags & T_NEGOTIATE) {
+ /* We have to use SELATNS to send sync/wide messages */
+ selatn3 = 0;
+ selatns = 1;
+ }
+
+ cmd = (u_char *)&ecb->cmd.cmd;
+
+ if (selatn3) {
+ /* We'll use tags with SELATN3 */
+ clen = ecb->clen + 3;
+ cmd -= 3;
+ cmd[0] = MSG_IDENTIFY(lun, 1); /* msg[0] */
+ cmd[1] = ecb->tag[0]; /* msg[1] */
+ cmd[2] = ecb->tag[1]; /* msg[2] */
+ } else {
+ /* We don't have tags, or will send messages with SELATNS */
+ clen = ecb->clen + 1;
+ cmd -= 1;
+ cmd[0] = MSG_IDENTIFY(lun, (tiflags & T_RSELECTOFF) == 0);
+ }
+
+ if ((sc->sc_features & NCR_F_DMASELECT) && !selatns) {
/* setup DMA transfer for command */
- dmasize = clen = ecb->clen + 1;
+ dmasize = clen;
sc->sc_cmdlen = clen;
- sc->sc_cmdp = (caddr_t)&ecb->cmd;
- NCRDMA_SETUP(sc, &sc->sc_cmdp, &sc->sc_cmdlen, 0, &dmasize);
+ sc->sc_cmdp = cmd;
/* Program the SCSI counter */
- NCR_WRITE_REG(sc, NCR_TCL, dmasize);
- NCR_WRITE_REG(sc, NCR_TCM, dmasize >> 8);
- if (sc->sc_cfg2 & NCRCFG2_FE) {
- NCR_WRITE_REG(sc, NCR_TCH, dmasize >> 16);
- }
+ NCR_SET_COUNT(sc, dmasize);
/* load the count in */
- NCRCMD(sc, NCRCMD_NOP|NCRCMD_DMA);
+ /* if (sc->sc_rev != NCR_VARIANT_FAS366) */
+ NCRCMD(sc, NCRCMD_NOP|NCRCMD_DMA);
/* And get the targets attention */
- NCRCMD(sc, NCRCMD_SELATN | NCRCMD_DMA);
+ if (selatn3) {
+ sc->sc_msgout = SEND_TAG;
+ sc->sc_flags |= NCR_ATN;
+ NCRCMD(sc, NCRCMD_SELATN3 | NCRCMD_DMA);
+ } else
+ NCRCMD(sc, NCRCMD_SELATN | NCRCMD_DMA);
+ NCRDMA_SETUP(sc, &sc->sc_cmdp, &sc->sc_cmdlen, 0, &dmasize);
NCRDMA_GO(sc);
return;
}
@@ -567,23 +699,21 @@ ncr53c9x_select(sc, ecb)
* Who am I. This is where we tell the target that we are
* happy for it to disconnect etc.
*/
- NCR_WRITE_REG(sc, NCR_FIFO,
- MSG_IDENTIFY(lun, (tiflags & T_RSELECTOFF)?0:1));
- if (ti->flags & T_NEGOTIATE) {
- /* Arbitrate, select and stop after IDENTIFY message */
- NCRCMD(sc, NCRCMD_SELATNS);
- return;
- }
-
- /* Now the command into the FIFO */
- cmd = (u_char *)&ecb->cmd.cmd;
- clen = ecb->clen;
- while (clen--)
- NCR_WRITE_REG(sc, NCR_FIFO, *cmd++);
+ /* Now get the command into the FIFO */
+ ncr53c9x_wrfifo(sc, cmd, clen);
/* And get the targets attention */
- NCRCMD(sc, NCRCMD_SELATN);
+ if (selatns) {
+ NCR_MISC(("SELATNS \n"));
+ /* Arbitrate, select and stop after IDENTIFY message */
+ NCRCMD(sc, NCRCMD_SELATNS);
+ } else if (selatn3) {
+ sc->sc_msgout = SEND_TAG;
+ sc->sc_flags |= NCR_ATN;
+ NCRCMD(sc, NCRCMD_SELATN3);
+ } else
+ NCRCMD(sc, NCRCMD_SELATN);
}
void
@@ -595,17 +725,8 @@ ncr53c9x_free_ecb(sc, ecb, flags)
int s;
s = splbio();
-
ecb->flags = 0;
- TAILQ_INSERT_HEAD(&sc->free_list, ecb, chain);
-
- /*
- * If there were none, wake anybody waiting for one to come free,
- * starting with queued entries.
- */
- if (ecb->chain.tqe_next == 0)
- wakeup(&sc->free_list);
-
+ pool_put(&ecb_pool, (void *)ecb);
splx(s);
}
@@ -615,19 +736,19 @@ ncr53c9x_get_ecb(sc, flags)
int flags;
{
struct ncr53c9x_ecb *ecb;
- int s;
+ int s, wait = 0;
- s = splbio();
-
- while ((ecb = sc->free_list.tqh_first) == NULL &&
- (flags & SCSI_NOSLEEP) == 0)
- tsleep(&sc->free_list, PRIBIO, "especb", 0);
- if (ecb) {
- TAILQ_REMOVE(&sc->free_list, ecb, chain);
- ecb->flags |= ECB_ALLOC;
- }
+ if ((curproc != NULL) && ((flags & SCSI_NOSLEEP) == 0))
+ wait = PR_WAITOK;
+ s = splbio();
+ ecb = (struct ncr53c9x_ecb *)pool_get(&ecb_pool, wait);
splx(s);
+ if (ecb == NULL)
+ return (NULL);
+ bzero(ecb, sizeof(*ecb));
+ timeout_set(&ecb->to, ncr53c9x_timeout, ecb);
+ ecb->flags |= ECB_ALLOC;
return (ecb);
}
@@ -647,6 +768,9 @@ ncr53c9x_scsi_cmd(xs)
struct scsi_link *sc_link = xs->sc_link;
struct ncr53c9x_softc *sc = sc_link->adapter_softc;
struct ncr53c9x_ecb *ecb;
+ struct ncr53c9x_tinfo *ti;
+ struct ncr53c9x_linfo *li;
+ int64_t lun = sc_link->lun;
int s, flags;
NCR_TRACE(("[ncr53c9x_scsi_cmd] "));
@@ -654,6 +778,27 @@ ncr53c9x_scsi_cmd(xs)
sc_link->target));
flags = xs->flags;
+ ti = &sc->sc_tinfo[sc_link->target];
+ li = TINFO_LUN(ti, lun);
+ if (li == NULL) {
+ int wait = M_NOWAIT;
+
+ /* Initialize LUN info and add to list. */
+ if ((curproc != NULL) && ((flags & SCSI_NOSLEEP) == 0))
+ wait = M_WAITOK;
+ if ((li = malloc(sizeof(*li), M_DEVBUF, M_NOWAIT)) == NULL) {
+ return (TRY_AGAIN_LATER);
+ }
+ bzero(li, sizeof(*li));
+ li->last_used = time.tv_sec;
+ li->lun = lun;
+ s = splbio();
+ LIST_INSERT_HEAD(&ti->luns, li, link);
+ if (lun < NCR_NLUN)
+ ti->lun[lun] = li;
+ splx(s);
+ }
+
if ((ecb = ncr53c9x_get_ecb(sc, flags)) == NULL)
return (TRY_AGAIN_LATER);
@@ -676,6 +821,7 @@ ncr53c9x_scsi_cmd(xs)
s = splbio();
TAILQ_INSERT_TAIL(&sc->ready_list, ecb, chain);
+ ecb->flags |= ECB_READY;
if (sc->sc_state == NCR_IDLE)
ncr53c9x_sched(sc);
@@ -742,6 +888,9 @@ ncr53c9x_sched(sc)
struct ncr53c9x_ecb *ecb;
struct scsi_link *sc_link;
struct ncr53c9x_tinfo *ti;
+ int lun;
+ struct ncr53c9x_linfo *li;
+ int s, tag;
NCR_TRACE(("[ncr53c9x_sched] "));
if (sc->sc_state != NCR_IDLE)
@@ -754,8 +903,109 @@ ncr53c9x_sched(sc)
for (ecb = sc->ready_list.tqh_first; ecb; ecb = ecb->chain.tqe_next) {
sc_link = ecb->xs->sc_link;
ti = &sc->sc_tinfo[sc_link->target];
- if ((ti->lubusy & (1 << sc_link->lun)) == 0) {
+ lun = sc_link->lun;
+
+ /* Select type of tag for this command */
+ if ((ti->flags & (T_RSELECTOFF)) != 0)
+ tag = 0;
+ else if ((ti->flags & T_TAG) == 0)
+ tag = 0;
+ else if ((ecb->flags & ECB_SENSE) != 0)
+ tag = 0;
+ else if (ecb->xs->flags & SCSI_URGENT)
+ tag = MSG_HEAD_OF_Q_TAG;
+ else
+ tag = MSG_SIMPLE_Q_TAG;
+#if 0
+ /* XXXX Use tags for polled commands? */
+ if (ecb->xs->flags & SCSI_POLL)
+ tag = 0;
+#endif
+ s = splbio();
+ li = TINFO_LUN(ti, lun);
+ if (!li) {
+ int wait = M_NOWAIT;
+ int flags = ecb->flags;
+
+ /* Initialize LUN info and add to list. */
+ if ((curproc != NULL) && ((flags & SCSI_NOSLEEP) == 0))
+ wait = M_WAITOK;
+ if ((li = malloc(sizeof(*li), M_DEVBUF, M_NOWAIT)) == NULL) {
+ splx(s);
+ continue;
+ }
+ bzero(li, sizeof(*li));
+ li->lun = lun;
+
+ LIST_INSERT_HEAD(&ti->luns, li, link);
+ if (lun < NCR_NLUN)
+ ti->lun[lun] = li;
+ }
+ li->last_used = time.tv_sec;
+ if (!tag) {
+ /* Try to issue this as an un-tagged command */
+ if (!li->untagged)
+ li->untagged = ecb;
+ }
+ if (li->untagged) {
+ tag = 0;
+ if ((li->busy != 1) && !li->used) {
+ /* We need to issue this untagged command now */
+ ecb = li->untagged;
+ sc_link = ecb->xs->sc_link;
+ }
+ else {
+ /* Not ready yet */
+ splx(s);
+ continue;
+ }
+ }
+ ecb->tag[0] = tag;
+ if (tag) {
+ int i;
+
+ /* Allocate a tag */
+ if (li->used == 255) {
+ /* no free tags */
+ splx(s);
+ continue;
+ }
+ /* Start from the last used location */
+ for (i=li->avail; i<256; i++) {
+ if (li->queued[i] == NULL)
+ break;
+ }
+ /* Couldn't find one, start again from the beginning */
+ if (i == 256) {
+ for (i = 0; i<256; i++) {
+ if (li->queued[i] == NULL)
+ break;
+ }
+ }
+#ifdef DIAGNOSTIC
+ if (i == 256)
+ panic("ncr53c9x_sched: tag alloc failure\n");
+#endif
+
+ /* Save where to start next time. */
+ li->avail = i+1;
+ li->used++;
+
+ li->queued[i] = ecb;
+ ecb->tag[1] = i;
+ }
+ splx(s);
+ if (li->untagged && (li->busy != 1)) {
+ li->busy = 1;
TAILQ_REMOVE(&sc->ready_list, ecb, chain);
+ ecb->flags &= ~ECB_READY;
+ sc->sc_nexus = ecb;
+ ncr53c9x_select(sc, ecb);
+ break;
+ }
+ if (!li->untagged && tag) {
+ TAILQ_REMOVE(&sc->ready_list, ecb, chain);
+ ecb->flags &= ~ECB_READY;
sc->sc_nexus = ecb;
ncr53c9x_select(sc, ecb);
break;
@@ -775,6 +1025,8 @@ ncr53c9x_sense(sc, ecb)
struct scsi_link *sc_link = xs->sc_link;
struct ncr53c9x_tinfo *ti = &sc->sc_tinfo[sc_link->target];
struct scsi_sense *ss = (void *)&ecb->cmd.cmd;
+ struct ncr53c9x_linfo *li;
+ int lun = sc_link->lun;
NCR_MISC(("requesting sense "));
/* Next, setup a request sense command block */
@@ -788,14 +1040,16 @@ ncr53c9x_sense(sc, ecb)
ecb->flags |= ECB_SENSE;
ecb->timeout = NCR_SENSE_TIMEOUT;
ti->senses++;
- if (ecb->flags & ECB_NEXUS)
- ti->lubusy &= ~(1 << sc_link->lun);
+ li = TINFO_LUN(ti, lun);
+ if (li->busy) li->busy = 0;
+ ncr53c9x_dequeue(sc, ecb);
+ li->untagged = ecb;
+ li->busy = 2;
if (ecb == sc->sc_nexus) {
- ecb->flags &= ~ECB_NEXUS;
ncr53c9x_select(sc, ecb);
} else {
- ncr53c9x_dequeue(sc, ecb);
TAILQ_INSERT_HEAD(&sc->ready_list, ecb, chain);
+ ecb->flags |= ECB_READY;
if (sc->sc_state == NCR_IDLE)
ncr53c9x_sched(sc);
}
@@ -812,11 +1066,23 @@ ncr53c9x_done(sc, ecb)
struct scsi_xfer *xs = ecb->xs;
struct scsi_link *sc_link = xs->sc_link;
struct ncr53c9x_tinfo *ti = &sc->sc_tinfo[sc_link->target];
+ int lun = sc_link->lun;
+ struct ncr53c9x_linfo *li = TINFO_LUN(ti, lun);
NCR_TRACE(("[ncr53c9x_done(error:%x)] ", xs->error));
timeout_del(&ecb->to);
+ if (ecb->stat == SCSI_QUEUE_FULL) {
+ /*
+ * Set current throttle -- we should reset
+ * this periodically
+ */
+ sc_link->openings = li->used - 1;
+ printf("\n%s: QFULL -- throttling to %d commands\n",
+ sc->sc_dev.dv_xname, sc_link->openings);
+ }
+
/*
* Now, if we've come here with no error code, i.e. we've kept the
* initial XS_NOERROR, and the status code signals that we should
@@ -857,17 +1123,25 @@ ncr53c9x_done(sc, ecb)
/*
* Remove the ECB from whatever queue it's on.
*/
- if (ecb->flags & ECB_NEXUS)
- ti->lubusy &= ~(1 << sc_link->lun);
+ ncr53c9x_dequeue(sc, ecb);
if (ecb == sc->sc_nexus) {
sc->sc_nexus = NULL;
if (sc->sc_state != NCR_CLEANING) {
sc->sc_state = NCR_IDLE;
ncr53c9x_sched(sc);
}
- } else
- ncr53c9x_dequeue(sc, ecb);
-
+ }
+
+ if (xs->error == XS_SELTIMEOUT) {
+ /* Selection timeout -- discard this LUN if empty */
+ if (!li->untagged && !li->used) {
+ if (lun < NCR_NLUN)
+ ti->lun[lun] = NULL;
+ LIST_REMOVE(li, link);
+ free(li, M_DEVBUF);
+ }
+ }
+
ncr53c9x_free_ecb(sc, ecb, xs->flags);
ti->cmds++;
scsi_done(xs);
@@ -878,14 +1152,38 @@ ncr53c9x_dequeue(sc, ecb)
struct ncr53c9x_softc *sc;
struct ncr53c9x_ecb *ecb;
{
+ struct ncr53c9x_tinfo *ti =
+ &sc->sc_tinfo[ecb->xs->sc_link->target];
+ struct ncr53c9x_linfo *li;
+ int64_t lun = ecb->xs->sc_link->lun;
+
+ li = TINFO_LUN(ti, lun);
+#ifdef DIAGNOSTIC
+ if ((!li) || (li->lun != lun))
+ panic("ncr53c9x_dequeue: lun %qx for ecb %p does not exist\n",
+ (long long)lun, ecb);
+#endif
+ if (li->untagged == ecb) {
+ li->busy = 0;
+ li->untagged = NULL;
+ }
+ if (ecb->tag[0] && li->queued[ecb->tag[1]]) {
+#ifdef DIAGNOSTIC
+ if (li->queued[ecb->tag[1]] && (li->queued[ecb->tag[1]] != ecb))
+ panic("ncr53c9x_dequeue: slot %d for lun %qx has %p "
+ "instead of ecb %p\n", ecb->tag[1],
+ (long long)lun,
+ li->queued[ecb->tag[1]], ecb);
+#endif
+ li->queued[ecb->tag[1]] = NULL;
+ li->used --;
- if (ecb->flags & ECB_NEXUS) {
- TAILQ_REMOVE(&sc->nexus_list, ecb, chain);
- ecb->flags &= ~ECB_NEXUS;
- } else {
+ }
+ if (ecb->flags & ECB_READY) {
+ ecb->flags &= ~ECB_READY;
TAILQ_REMOVE(&sc->ready_list, ecb, chain);
}
-}
+ }
/*
* INTERRUPT/PROTOCOL ENGINE
@@ -898,33 +1196,138 @@ ncr53c9x_dequeue(sc, ecb)
*/
#define ncr53c9x_sched_msgout(m) \
do { \
- NCR_MISC(("ncr53c9x_sched_msgout %d ", m)); \
+ NCR_MISC(("ncr53c9x_sched_msgout %x %d ", m, __LINE__)); \
NCRCMD(sc, NCRCMD_SETATN); \
sc->sc_flags |= NCR_ATN; \
sc->sc_msgpriq |= (m); \
} while (0)
+static void
+ncr53c9x_flushfifo(struct ncr53c9x_softc *sc)
+{
+ NCR_MISC(("[flushfifo] "));
+
+ NCRCMD(sc, NCRCMD_FLUSH);
+
+ if (sc->sc_phase == COMMAND_PHASE ||
+ sc->sc_phase == MESSAGE_OUT_PHASE)
+ DELAY(2);
+}
+
+static int
+ncr53c9x_rdfifo(struct ncr53c9x_softc *sc, int how)
+{
+ int i, n;
+ u_char *buf;
+
+ switch(how) {
+ case NCR_RDFIFO_START:
+ buf = sc->sc_imess;
+ sc->sc_imlen = 0;
+ break;
+ case NCR_RDFIFO_CONTINUE:
+ buf = sc->sc_imess + sc->sc_imlen;
+ break;
+ default:
+ panic("ncr53c9x_rdfifo: bad flag\n");
+ break;
+ }
+
+ /*
+ * XXX buffer (sc_imess) size for message
+ */
+
+ n = NCR_READ_REG(sc, NCR_FFLAG) & NCRFIFO_FF;
+
+ if (sc->sc_rev == NCR_VARIANT_FAS366) {
+ n *= 2;
+
+ for (i = 0; i < n; i++)
+ buf[i] = NCR_READ_REG(sc, NCR_FIFO);
+
+ if (sc->sc_espstat2 & FAS_STAT2_ISHUTTLE) {
+
+ NCR_WRITE_REG(sc, NCR_FIFO, 0);
+ buf[i++] = NCR_READ_REG(sc, NCR_FIFO);
+
+ NCR_READ_REG(sc, NCR_FIFO);
+
+ ncr53c9x_flushfifo(sc);
+ }
+ } else {
+ for (i = 0; i < n; i++)
+ buf[i] = NCR_READ_REG(sc, NCR_FIFO);
+ }
+
+ sc->sc_imlen += i;
+
+#ifdef NCR53C9X_DEBUG
+ {
+ int j;
+
+ NCR_TRACE(("\n[rdfifo %s (%d):",
+ (how == NCR_RDFIFO_START) ? "start" : "cont",
+ (int)sc->sc_imlen));
+ if (ncr53c9x_debug & NCR_SHOWTRAC) {
+ for (j = 0; j < sc->sc_imlen; j++)
+ printf(" %02x", sc->sc_imess[j]);
+ printf("]\n");
+ }
+ }
+#endif
+ return sc->sc_imlen;
+}
+
+static void
+ncr53c9x_wrfifo(struct ncr53c9x_softc *sc, u_char *p, int len)
+{
+ int i;
+
+#ifdef NCR53C9X_DEBUG
+ NCR_MISC(("[wrfifo(%d):", len));
+ if (ncr53c9x_debug & NCR_SHOWTRAC) {
+ for (i = 0; i < len; i++)
+ printf(" %02x", p[i]);
+ printf("]\n");
+ }
+#endif
+
+ for (i = 0; i < len; i++) {
+ NCR_WRITE_REG(sc, NCR_FIFO, p[i]);
+
+ if (sc->sc_rev == NCR_VARIANT_FAS366)
+ NCR_WRITE_REG(sc, NCR_FIFO, 0);
+ }
+}
+
int
-ncr53c9x_reselect(sc, message)
+ncr53c9x_reselect(sc, message, tagtype, tagid)
struct ncr53c9x_softc *sc;
int message;
{
u_char selid, target, lun;
- struct ncr53c9x_ecb *ecb;
- struct scsi_link *sc_link;
+ struct ncr53c9x_ecb *ecb = NULL;
struct ncr53c9x_tinfo *ti;
+ struct ncr53c9x_linfo *li;
- /*
- * The SCSI chip made a snapshot of the data bus while the reselection
- * was being negotiated. This enables us to determine which target did
- * the reselect.
- */
- selid = sc->sc_selid & ~(1 << sc->sc_id);
- if (selid & (selid - 1)) {
- printf("%s: reselect with invalid selid %02x;"
- " sending DEVICE RESET\n", sc->sc_dev.dv_xname, selid);
- goto reset;
+ if (sc->sc_rev == NCR_VARIANT_FAS366) {
+ target = sc->sc_selid;
+ } else {
+ /*
+ * The SCSI chip made a snapshot of the data bus while the reselection
+ * was being negotiated. This enables us to determine which target did
+ * the reselect.
+ */
+ selid = sc->sc_selid & ~(1 << sc->sc_id);
+ if (selid & (selid - 1)) {
+ printf("%s: reselect with invalid selid %02x;"
+ " sending DEVICE RESET\n", sc->sc_dev.dv_xname, selid);
+ goto reset;
+
+ }
+ target = ffs(selid) - 1;
}
+ lun = message & 0x07;
/*
* Search wait queue for disconnected cmd
@@ -932,33 +1335,33 @@ ncr53c9x_reselect(sc, message)
* any more sophisticated structures than a simple
* singly linked list.
*/
- target = ffs(selid) - 1;
- lun = message & 0x07;
- for (ecb = sc->nexus_list.tqh_first; ecb != NULL;
- ecb = ecb->chain.tqe_next) {
- sc_link = ecb->xs->sc_link;
- if (sc_link->target == target &&
- sc_link->lun == lun)
- break;
+ ti = &sc->sc_tinfo[target];
+ li = TINFO_LUN(ti, lun);
+
+ /*
+ * We can get as far as the LUN with the IDENTIFY
+ * message. Check to see if we're running an
+ * un-tagged command. Otherwise ack the IDENTIFY
+ * and wait for a tag message.
+ */
+ if (li != NULL) {
+ if (li->untagged != NULL && li->busy)
+ ecb = li->untagged;
+ else if (tagtype != MSG_SIMPLE_Q_TAG) {
+ /* Wait for tag to come by */
+ sc->sc_state = NCR_IDENTIFIED;
+ return (0);
+ } else if (tagtype) ecb = li->queued[tagid];
}
if (ecb == NULL) {
- printf("%s: reselect from target %d lun %d with no nexus;"
- " sending ABORT\n", sc->sc_dev.dv_xname, target, lun);
+ printf("%s: reselect from target %d lun %d tag %x:%x with no nexus;"
+ " sending ABORT\n", sc->sc_dev.dv_xname, target, lun, tagtype, tagid);
goto abort;
}
/* Make this nexus active again. */
- TAILQ_REMOVE(&sc->nexus_list, ecb, chain);
sc->sc_state = NCR_CONNECTED;
sc->sc_nexus = ecb;
- ti = &sc->sc_tinfo[target];
-#ifdef NCR53C9X_DEBUG
- if ((ti->lubusy & (1 << lun)) == 0) {
- printf("%s: reselect: target %d, lun %d: should be busy\n",
- sc->sc_dev.dv_xname, target, lun);
- ti->lubusy |= (1 << lun);
- }
-#endif
ncr53c9x_setsync(sc, ti);
if (ecb->flags & ECB_RESET)
@@ -985,6 +1388,21 @@ abort:
#define IS2BYTEMSG(m) (((m) & 0xf0) == 0x20)
#define ISEXTMSG(m) ((m) == 1)
+static inline int
+__verify_msg_format(u_char *p, int len)
+{
+
+ if (len == 1 && IS1BYTEMSG(p[0]))
+ return 1;
+ if (len == 2 && IS2BYTEMSG(p[0]))
+ return 1;
+ if (len >= 3 && ISEXTMSG(p[0]) &&
+ len == p[1] + 2)
+ return 1;
+
+ return 0;
+}
+
/*
* Get an incoming message as initiator.
*
@@ -993,15 +1411,13 @@ abort:
*/
void
ncr53c9x_msgin(sc)
- register struct ncr53c9x_softc *sc;
+ struct ncr53c9x_softc *sc;
{
- register int v;
NCR_TRACE(("[ncr53c9x_msgin(curmsglen:%ld)] ", (long)sc->sc_imlen));
- if ((NCR_READ_REG(sc, NCR_FFLAG) & NCRFIFO_FF) == 0) {
- printf("%s: msgin: no msg byte available\n",
- sc->sc_dev.dv_xname);
+ if (sc->sc_imlen == 0) {
+ printf("%s: msgin: no msg byte available\n", sc->sc_dev.dv_xname);
return;
}
@@ -1011,32 +1427,19 @@ ncr53c9x_msgin(sc)
* MESSAGE_IN_PHASE. If we have been in some other phase,
* then this is a new message.
*/
- if (sc->sc_prevphase != MESSAGE_IN_PHASE) {
+ if (sc->sc_prevphase != MESSAGE_IN_PHASE && sc->sc_state != NCR_RESELECTED) {
+ printf("%s: phase change, dropping message, prev %d, state %d\n",
+ sc->sc_dev.dv_xname, sc->sc_prevphase, sc->sc_state);
sc->sc_flags &= ~NCR_DROP_MSGI;
sc->sc_imlen = 0;
}
- v = NCR_READ_REG(sc, NCR_FIFO);
- NCR_MISC(("<msgbyte:0x%02x>", v));
-
-#if 0
- if (sc->sc_state == NCR_RESELECTED && sc->sc_imlen == 0) {
- /*
- * Which target is reselecting us? (The ID bit really)
- */
- sc->sc_selid = v;
- NCR_MISC(("selid=0x%2x ", sc->sc_selid));
- return;
- }
-#endif
-
- sc->sc_imess[sc->sc_imlen] = v;
+ NCR_TRACE(("<msgbyte:0x%02x>", sc->sc_imess[0]));
/*
* If we're going to reject the message, don't bother storing
* the incoming bytes. But still, we need to ACK them.
*/
-
if ((sc->sc_flags & NCR_DROP_MSGI)) {
NCRCMD(sc, NCRCMD_MSGOK);
printf("<dropping msg byte %x>",
@@ -1048,27 +1451,37 @@ ncr53c9x_msgin(sc)
ncr53c9x_sched_msgout(SEND_REJECT);
sc->sc_flags |= NCR_DROP_MSGI;
} else {
- sc->sc_imlen++;
- /*
- * This testing is suboptimal, but most
- * messages will be of the one byte variety, so
- * it should not effect performance
- * significantly.
- */
- if (sc->sc_imlen == 1 && IS1BYTEMSG(sc->sc_imess[0]))
- goto gotit;
- if (sc->sc_imlen == 2 && IS2BYTEMSG(sc->sc_imess[0]))
- goto gotit;
- if (sc->sc_imlen >= 3 && ISEXTMSG(sc->sc_imess[0]) &&
- sc->sc_imlen == sc->sc_imess[1] + 2)
+ u_char *pb;
+ int plen;
+
+ switch (sc->sc_state) {
+ /*
+ * if received message is the first of reselection
+ * then first byte is selid, and then message
+ */
+ case NCR_RESELECTED:
+ pb = sc->sc_imess + 1;
+ plen = sc->sc_imlen - 1;
+ break;
+ default:
+ pb = sc->sc_imess;
+ plen = sc->sc_imlen;
+ break;
+ }
+
+ if (__verify_msg_format(pb, plen))
goto gotit;
}
+
/* Ack what we have so far */
NCRCMD(sc, NCRCMD_MSGOK);
return;
gotit:
- NCR_MSGS(("gotmsg(%x)", sc->sc_imess[0]));
+ NCR_MSGS(("gotmsg(%x) state %d", sc->sc_imess[0], sc->sc_state));
+ /* we got complete message, flush the imess, XXX nobody uses imlen below */
+ sc->sc_imlen = 0;
+
/*
* Now we should have a complete message (1 byte, 2 byte
* and moderately long extended messages). We only handle
@@ -1078,6 +1491,8 @@ gotit:
switch (sc->sc_state) {
struct ncr53c9x_ecb *ecb;
struct ncr53c9x_tinfo *ti;
+ struct ncr53c9x_linfo *li;
+ int lun;
case NCR_CONNECTED:
ecb = sc->sc_nexus;
@@ -1087,10 +1502,9 @@ gotit:
case MSG_CMDCOMPLETE:
NCR_MSGS(("cmdcomplete "));
if (sc->sc_dleft < 0) {
- struct scsi_link *sc_link = ecb->xs->sc_link;
- printf("%s: %ld extra bytes from %d:%d\n",
- sc->sc_dev.dv_xname, -(long)sc->sc_dleft,
- sc_link->target, sc_link->lun);
+ sc_print_addr(ecb->xs->sc_link);
+ printf("got %ld extra bytes\n",
+ -(long)sc->sc_dleft);
sc->sc_dleft = 0;
}
ecb->dleft = (ecb->flags & ECB_TENTATIVE_DONE)
@@ -1104,11 +1518,46 @@ gotit:
case MSG_MESSAGE_REJECT:
NCR_MSGS(("msg reject (msgout=%x) ", sc->sc_msgout));
switch (sc->sc_msgout) {
+ case SEND_TAG:
+ /* Target does not like tagged queuing.
+ * - Flush the command queue
+ * - Disable tagged queuing for the target
+ * - Dequeue ecb from the queued array.
+ */
+ printf("%s: tagged queuing rejected: target %d\n",
+ sc->sc_dev.dv_xname, ecb->xs->sc_link->target);
+
+ NCR_MSGS(("(rejected sent tag)"));
+ NCRCMD(sc, NCRCMD_FLUSH);
+ DELAY(1);
+ ti->flags &= ~T_TAG;
+ lun = ecb->xs->sc_link->lun;
+ li = TINFO_LUN(ti, lun);
+ if (ecb->tag[0] &&
+ li->queued[ecb->tag[1]] != NULL) {
+ li->queued[ecb->tag[1]] = NULL;
+ li->used--;
+ }
+ ecb->tag[0] = ecb->tag[1] = 0;
+ li->untagged = ecb;
+ li->busy = 1;
+ break;
+
case SEND_SDTR:
+ printf("%s: sync transfer rejected: target %d\n",
+ sc->sc_dev.dv_xname, ecb->xs->sc_link->target);
sc->sc_flags &= ~NCR_SYNCHNEGO;
ti->flags &= ~(T_NEGOTIATE | T_SYNCMODE);
ncr53c9x_setsync(sc, ti);
break;
+
+ case SEND_WDTR:
+ printf("%s: wide transfer rejected: target %d\n",
+ sc->sc_dev.dv_xname, ecb->xs->sc_link->target);
+ ti->flags &= ~T_WIDE;
+ ti->width = 0;
+ break;
+
case SEND_INIT_DET_ERR:
goto abort;
}
@@ -1118,6 +1567,12 @@ gotit:
NCR_MSGS(("noop "));
break;
+ case MSG_HEAD_OF_Q_TAG:
+ case MSG_SIMPLE_Q_TAG:
+ case MSG_ORDERED_Q_TAG:
+ NCR_MSGS(("TAG %x:%x", sc->sc_imess[0], sc->sc_imess[1]));
+ break;
+
case MSG_DISCONNECT:
NCR_MSGS(("disconnect "));
ti->dconns++;
@@ -1160,8 +1615,10 @@ gotit:
if (sc->sc_minsync == 0 ||
ti->offset == 0 ||
ti->period > 124) {
- printf("%s:%d: async\n", "esp",
- ecb->xs->sc_link->target);
+#ifdef NCR53C9X_DEBUG
+ sc_print_addr(ecb->xs->sc_link);
+ printf("async mode\n");
+#endif
if ((sc->sc_flags&NCR_SYNCHNEGO)
== 0) {
/*
@@ -1184,7 +1641,7 @@ gotit:
ti->period = ncr53c9x_cpb2stp(sc, p);
#ifdef NCR53C9X_DEBUG
sc_print_addr(ecb->xs->sc_link);
- printf("max sync rate %d.%02dMb/s\n",
+ printf("max sync rate %d.%02dMB/s\n",
r, s);
#endif
if ((sc->sc_flags&NCR_SYNCHNEGO) == 0) {
@@ -1209,34 +1666,67 @@ gotit:
ncr53c9x_setsync(sc, ti);
break;
+ case MSG_EXT_WDTR:
+ printf("%s: wide mode %d\n",
+ sc->sc_dev.dv_xname, sc->sc_imess[3]);
+ if (sc->sc_imess[3] == 1) {
+ ti->cfg3 |= NCRFASCFG3_EWIDE;
+ ncr53c9x_setsync(sc, ti);
+ } else
+ ti->width = 0;
+ ti->flags &= ~T_WIDE;
+ break;
default:
- printf("%s: unrecognized MESSAGE EXTENDED;"
- " sending REJECT\n", sc->sc_dev.dv_xname);
+ sc_print_addr(ecb->xs->sc_link);
+ printf("unrecognized MESSAGE EXTENDED;"
+ " sending REJECT\n");
goto reject;
}
break;
default:
NCR_MSGS(("ident "));
- printf("%s: unrecognized MESSAGE; sending REJECT\n",
- sc->sc_dev.dv_xname);
+ sc_print_addr(ecb->xs->sc_link);
+ printf("unrecognized MESSAGE; sending REJECT\n");
reject:
ncr53c9x_sched_msgout(SEND_REJECT);
break;
}
break;
+ case NCR_IDENTIFIED:
+ /*
+ * IDENTIFY message was recived and queue tag is expected now
+ */
+ if ((sc->sc_imess[0] != MSG_SIMPLE_Q_TAG) ||
+ (sc->sc_msgify == 0)) {
+ printf("%s: TAG reselect without IDENTIFY;"
+ " MSG %x;"
+ " sending DEVICE RESET\n",
+ sc->sc_dev.dv_xname,
+ sc->sc_imess[0]);
+ goto reset;
+ }
+ (void) ncr53c9x_reselect(sc, sc->sc_msgify,
+ sc->sc_imess[0], sc->sc_imess[1]);
+ break;
+
case NCR_RESELECTED:
- if (!MSG_ISIDENTIFY(sc->sc_imess[0])) {
+ if (MSG_ISIDENTIFY(sc->sc_imess[1])) {
+ sc->sc_msgify = sc->sc_imess[1];
+ } else {
printf("%s: reselect without IDENTIFY;"
- " sending DEVICE RESET\n", sc->sc_dev.dv_xname);
+ " MSG %x;"
+ " sending DEVICE RESET\n",
+ sc->sc_dev.dv_xname,
+ sc->sc_imess[1]);
goto reset;
}
-
- (void) ncr53c9x_reselect(sc, sc->sc_imess[0]);
+ (void) ncr53c9x_reselect(sc, sc->sc_msgify, 0, 0);
break;
default:
+ sc_print_addr(ecb->xs->sc_link);
printf("%s: unexpected MESSAGE IN; sending DEVICE RESET\n",
sc->sc_dev.dv_xname);
reset:
@@ -1248,6 +1738,10 @@ gotit:
break;
}
+ /* if we have more messages to send set ATN */
+ if (sc->sc_msgpriq)
+ NCRCMD(sc, NCRCMD_SETATN);
+
/* Ack last message byte */
NCRCMD(sc, NCRCMD_MSGOK);
@@ -1262,7 +1756,7 @@ gotit:
*/
void
ncr53c9x_msgout(sc)
- register struct ncr53c9x_softc *sc;
+ struct ncr53c9x_softc *sc;
{
struct ncr53c9x_tinfo *ti;
struct ncr53c9x_ecb *ecb;
@@ -1282,7 +1776,7 @@ ncr53c9x_msgout(sc)
if (sc->sc_prevphase != MESSAGE_OUT_PHASE) {
new:
NCRCMD(sc, NCRCMD_FLUSH);
- DELAY(1);
+/* DELAY(1); */
sc->sc_msgoutq = 0;
sc->sc_omlen = 0;
}
@@ -1317,6 +1811,15 @@ ncr53c9x_msgout(sc)
ncr53c9x_setsync(sc, ti);
}
break;
+ case SEND_WDTR:
+ ecb = sc->sc_nexus;
+ ti = &sc->sc_tinfo[ecb->xs->sc_link->target];
+ sc->sc_omess[0] = MSG_EXTENDED;
+ sc->sc_omess[1] = 2;
+ sc->sc_omess[2] = MSG_EXT_WDTR;
+ sc->sc_omess[3] = ti->width;
+ sc->sc_omlen = 4;
+ break;
case SEND_IDENTIFY:
if (sc->sc_state != NCR_CONNECTED) {
printf("%s at line %d: no nexus\n",
@@ -1326,13 +1829,25 @@ ncr53c9x_msgout(sc)
sc->sc_omess[0] =
MSG_IDENTIFY(ecb->xs->sc_link->lun, 0);
break;
+ case SEND_TAG:
+ if (sc->sc_state != NCR_CONNECTED) {
+ printf("%s at line %d: no nexus\n",
+ sc->sc_dev.dv_xname, __LINE__);
+ }
+ ecb = sc->sc_nexus;
+ sc->sc_omess[0] = ecb->tag[0];
+ sc->sc_omess[1] = ecb->tag[1];
+ sc->sc_omlen = 2;
+ break;
case SEND_DEV_RESET:
sc->sc_flags |= NCR_ABORTING;
sc->sc_omess[0] = MSG_BUS_DEV_RESET;
ecb = sc->sc_nexus;
ti = &sc->sc_tinfo[ecb->xs->sc_link->target];
ti->flags &= ~T_SYNCMODE;
- ti->flags |= T_NEGOTIATE;
+ if ((ti->flags & T_SYNCHOFF) == 0)
+ /* We can re-start sync negotiation */
+ ti->flags |= T_NEGOTIATE;
break;
case SEND_PARITY_ERROR:
sc->sc_omess[0] = MSG_PARITY_ERROR;
@@ -1368,19 +1883,33 @@ ncr53c9x_msgout(sc)
sc->sc_omp = sc->sc_omess;
}
- /* (re)send the message */
- size = min(sc->sc_omlen, sc->sc_maxxfer);
- NCRDMA_SETUP(sc, &sc->sc_omp, &sc->sc_omlen, 0, &size);
- /* Program the SCSI counter */
- NCR_WRITE_REG(sc, NCR_TCL, size);
- NCR_WRITE_REG(sc, NCR_TCM, size >> 8);
- if (sc->sc_cfg2 & NCRCFG2_FE) {
- NCR_WRITE_REG(sc, NCR_TCH, size >> 16);
+#ifdef DEBUG
+ {
+ int i;
+
+ for (i = 0; i<sc->sc_omlen; i++)
+ NCR_MISC(("<msgbyte:0x%02x>", sc->sc_omess[i]));
+ }
+#endif
+ if (sc->sc_rev == NCR_VARIANT_FAS366) {
+ /*
+ * XXX fifo size
+ */
+ ncr53c9x_flushfifo(sc);
+ ncr53c9x_wrfifo(sc, sc->sc_omp, sc->sc_omlen);
+ NCRCMD(sc, NCRCMD_TRANS);
+ } else {
+ /* (re)send the message */
+ size = min(sc->sc_omlen, sc->sc_maxxfer);
+ NCRDMA_SETUP(sc, &sc->sc_omp, &sc->sc_omlen, 0, &size);
+ /* Program the SCSI counter */
+ NCR_SET_COUNT(sc, size);
+
+ /* Load the count in and start the message-out transfer */
+ NCRCMD(sc, NCRCMD_NOP|NCRCMD_DMA);
+ NCRCMD(sc, NCRCMD_TRANS|NCRCMD_DMA);
+ NCRDMA_GO(sc);
}
- /* Load the count in and start the message-out transfer */
- NCRCMD(sc, NCRCMD_NOP|NCRCMD_DMA);
- NCRCMD(sc, NCRCMD_TRANS|NCRCMD_DMA);
- NCRDMA_GO(sc);
}
/*
@@ -1397,14 +1926,14 @@ int
ncr53c9x_intr(arg)
void *arg;
{
- register struct ncr53c9x_softc *sc = arg;
- register struct ncr53c9x_ecb *ecb;
- register struct scsi_link *sc_link;
+ struct ncr53c9x_softc *sc = arg;
+ struct ncr53c9x_ecb *ecb;
+ struct scsi_link *sc_link;
struct ncr53c9x_tinfo *ti;
size_t size;
int nfifo;
- NCR_TRACE(("[ncr53c9x_intr] "));
+ NCR_TRACE(("[ncr53c9x_intr: state %d] ", sc->sc_state));
if (!NCRDMA_ISINTR(sc))
return (0);
@@ -1547,13 +2076,13 @@ again:
if (sc->sc_phase != MESSAGE_IN_PHASE)
printf("%s: !TC on MSG OUT"
" [intr %x, stat %x, step %d]"
- " prevphase %x, resid %x\n",
+ " prevphase %x, resid %lx\n",
sc->sc_dev.dv_xname,
sc->sc_espintr,
sc->sc_espstat,
sc->sc_espstep,
sc->sc_prevphase,
- sc->sc_omlen);
+ (u_long)sc->sc_omlen);
} else if (sc->sc_dleft == 0) {
/*
* The DMA operation was started for
@@ -1586,11 +2115,12 @@ again:
}
if (sc->sc_espintr & NCRINTR_DIS) {
+ sc->sc_msgify = 0;
NCR_MISC(("<DISC [intr %x, stat %x, step %d]>",
sc->sc_espintr,sc->sc_espstat,sc->sc_espstep));
if (NCR_READ_REG(sc, NCR_FFLAG) & NCRFIFO_FF) {
NCRCMD(sc, NCRCMD_FLUSH);
- DELAY(1);
+/* DELAY(1); */
}
/*
* This command must (apparently) be issued within
@@ -1603,9 +2133,29 @@ again:
goto sched;
case NCR_SELECTING:
+ {
+ struct ncr53c9x_linfo *li;
+
ecb->xs->error = XS_SELTIMEOUT;
- goto finish;
+ /* Selection timeout -- discard all LUNs if empty */
+ sc_link = ecb->xs->sc_link;
+ ti = &sc->sc_tinfo[sc_link->target];
+ li = ti->luns.lh_first;
+ while (li) {
+ if (!li->untagged && !li->used) {
+ if (li->lun < NCR_NLUN)
+ ti->lun[li->lun] = NULL;
+ LIST_REMOVE(li, link);
+ free(li, M_DEVBUF);
+ /* Restart the search at the beginning */
+ li = ti->luns.lh_first;
+ continue;
+ }
+ li = li->link.le_next;
+ }
+ goto finish;
+ }
case NCR_CONNECTED:
if ((sc->sc_flags & NCR_SYNCHNEGO)) {
#ifdef NCR53C9X_DEBUG
@@ -1645,7 +2195,6 @@ again:
goto finish;
case NCR_DISCONNECT:
- TAILQ_INSERT_HEAD(&sc->nexus_list, ecb, chain);
sc->sc_nexus = NULL;
goto sched;
@@ -1684,17 +2233,32 @@ printf("<<RESELECT CONT'd>>");
#endif
break;
+ case NCR_IDENTIFIED:
+ ecb = sc->sc_nexus;
+ if (sc->sc_phase != MESSAGE_IN_PHASE) {
+ int i = (NCR_READ_REG(sc, NCR_FFLAG)
+ & NCRFIFO_FF);
+ /*
+ * Things are seriously fucked up.
+ * Pull the brakes, i.e. reset
+ */
+ printf("%s: target didn't send tag: %d bytes in fifo\n",
+ sc->sc_dev.dv_xname, i);
+ /* Drain and display fifo */
+ while (i-- > 0)
+ printf("[%d] ", NCR_READ_REG(sc, NCR_FIFO));
+ ncr53c9x_init(sc, 1);
+ return (1);
+ } else
+ goto msgin;
+
+ break;
case NCR_IDLE:
case NCR_SELECTING:
- sc->sc_msgpriq = sc->sc_msgout = sc->sc_msgoutq = 0;
- sc->sc_flags = 0;
ecb = sc->sc_nexus;
- if (ecb != NULL && (ecb->flags & ECB_NEXUS)) {
- sc_print_addr(ecb->xs->sc_link);
- printf("ECB_NEXUS while in state %x\n", sc->sc_state);
- }
-
if (sc->sc_espintr & NCRINTR_RESEL) {
+ sc->sc_msgpriq = sc->sc_msgout = sc->sc_msgoutq = 0;
+ sc->sc_flags = 0;
/*
* If we're trying to select a
* target ourselves, push our command
@@ -1703,9 +2267,9 @@ printf("<<RESELECT CONT'd>>");
if (sc->sc_state == NCR_SELECTING) {
NCR_MISC(("backoff selector "));
timeout_del(&ecb->to);
- sc_link = ecb->xs->sc_link;
- ti = &sc->sc_tinfo[sc_link->target];
+ ncr53c9x_dequeue(sc, ecb);
TAILQ_INSERT_HEAD(&sc->ready_list, ecb, chain);
+ ecb->flags |= ECB_READY;
ecb = sc->sc_nexus = NULL;
}
sc->sc_state = NCR_RESELECTED;
@@ -1729,7 +2293,8 @@ printf("<<RESELECT CONT'd>>");
* appear in the FIFO here, after the
* interrupt is taken.
*/
- nfifo = NCR_READ_REG(sc,NCR_FFLAG) & NCRFIFO_FF;
+ nfifo = ncr53c9x_rdfifo(sc, NCR_RDFIFO_START);
+
if (nfifo < 2 ||
(nfifo > 2 &&
sc->sc_rev != NCR_VARIANT_ESP100)) {
@@ -1745,8 +2310,8 @@ printf("<<RESELECT CONT'd>>");
ncr53c9x_init(sc, 1);
return (1);
}
- sc->sc_selid = NCR_READ_REG(sc, NCR_FIFO);
- NCR_MISC(("selid=0x%2x ", sc->sc_selid));
+ sc->sc_selid = sc->sc_imess[0];
+ NCR_MISC(("selid=%2x ", sc->sc_selid));
/* Handle identify message */
ncr53c9x_msgin(sc);
@@ -1757,15 +2322,17 @@ printf("<<RESELECT CONT'd>>");
*/
sc->sc_flags |= NCR_EXPECT_ILLCMD;
NCRCMD(sc, NCRCMD_FLUSH);
- } else if (ncr53c9x_dmaselect &&
+ } else if (sc->sc_features & NCR_F_DMASELECT &&
sc->sc_rev == NCR_VARIANT_ESP100) {
sc->sc_flags |= NCR_EXPECT_ILLCMD;
}
- if (sc->sc_state != NCR_CONNECTED) {
+ if (sc->sc_state != NCR_CONNECTED &&
+ sc->sc_state != NCR_IDENTIFIED) {
/* IDENTIFY fail?! */
- printf("%s: identify failed\n",
- sc->sc_dev.dv_xname);
+ printf("%s: identify failed, state %d, intr %02x\n",
+ sc->sc_dev.dv_xname, sc->sc_state,
+ sc->sc_espintr);
ncr53c9x_init(sc, 1);
return (1);
}
@@ -1780,7 +2347,7 @@ printf("<<RESELECT CONT'd>>");
*/
ecb = sc->sc_nexus;
if (!ecb)
- panic("esp: no nexus");
+ panic("ncr53c9x: no nexus");
sc_link = ecb->xs->sc_link;
ti = &sc->sc_tinfo[sc_link->target];
@@ -1799,7 +2366,8 @@ printf("<<RESELECT CONT'd>>");
NCRCMD(sc, NCRCMD_RSTATN);
break;
case 1:
- if ((ti->flags & T_NEGOTIATE) == 0) {
+ if ((ti->flags & T_NEGOTIATE) == 0 &&
+ ecb->tag[0] == 0) {
printf("%s: step 1 & !NEG\n",
sc->sc_dev.dv_xname);
goto reset;
@@ -1809,11 +2377,23 @@ printf("<<RESELECT CONT'd>>");
sc->sc_dev.dv_xname);
goto reset;
}
- /* Start negotiating */
- ti->period = sc->sc_minsync;
- ti->offset = 15;
- sc->sc_flags |= NCR_SYNCHNEGO;
- ncr53c9x_sched_msgout(SEND_SDTR);
+ if (ti->flags & T_WIDE) {
+ ncr53c9x_sched_msgout(SEND_WDTR);
+ }
+ if (ti->flags & T_NEGOTIATE) {
+ /* Start negotiating */
+ ti->period = sc->sc_minsync;
+ ti->offset = 15;
+ sc->sc_flags |= NCR_SYNCHNEGO;
+ if (ecb->tag[0])
+ ncr53c9x_sched_msgout(SEND_TAG|SEND_SDTR);
+ else
+ ncr53c9x_sched_msgout(SEND_SDTR);
+ } else {
+ /* Could not do ATN3 so send TAG */
+ ncr53c9x_sched_msgout(SEND_TAG);
+ }
+ sc->sc_prevphase = MESSAGE_OUT_PHASE; /* XXXX */
break;
case 3:
/*
@@ -1824,7 +2404,7 @@ printf("<<RESELECT CONT'd>>");
* Look at FIFO to see if command went out.
* (Timing problems?)
*/
- if (ncr53c9x_dmaselect) {
+ if (sc->sc_features & NCR_F_DMASELECT) {
if (sc->sc_cmdlen == 0)
/* Hope for the best.. */
break;
@@ -1851,15 +2431,15 @@ printf("<<RESELECT CONT'd>>");
NCRCMD(sc, NCRCMD_FLUSH);
break;
case 4:
- if (ncr53c9x_dmaselect &&
+ if (sc->sc_features & NCR_F_DMASELECT &&
sc->sc_cmdlen != 0)
printf("(%s:%d:%d): select; "
- "%d left in DMA buffer "
+ "%lu left in DMA buffer "
"[intr %x, stat %x, step %d]\n",
sc->sc_dev.dv_xname,
sc_link->target,
sc_link->lun,
- sc->sc_cmdlen,
+ (u_long)sc->sc_cmdlen,
sc->sc_espintr,
sc->sc_espstat,
sc->sc_espstep);
@@ -1867,9 +2447,6 @@ printf("<<RESELECT CONT'd>>");
break;
}
- ecb->flags |= ECB_NEXUS;
- ti->lubusy |= (1 << sc_link->lun);
-
sc->sc_prevphase = INVALID_PHASE; /* ?? */
/* Do an implicit RESTORE POINTERS. */
sc->sc_dp = ecb->daddr;
@@ -1909,16 +2486,12 @@ printf("<<RESELECT CONT'd>>");
sc->sc_espintr, sc->sc_espstat,
sc->sc_espstep);
}
- if ((NCR_READ_REG(sc, NCR_FFLAG)
- & NCRFIFO_FF) != 2) {
- /* Drop excess bytes from the queue */
- int i = (NCR_READ_REG(sc, NCR_FFLAG)
- & NCRFIFO_FF) - 2;
- while (i-- > 0)
- (void) NCR_READ_REG(sc, NCR_FIFO);
- }
- ecb->stat = NCR_READ_REG(sc, NCR_FIFO);
- msg = NCR_READ_REG(sc, NCR_FIFO);
+ ncr53c9x_rdfifo(sc, NCR_RDFIFO_START);
+ if (sc->sc_imlen < 2)
+ printf("%s: can't get status, only %d bytes\n",
+ sc->sc_dev.dv_xname, (int)sc->sc_imlen);
+ ecb->stat = sc->sc_imess[sc->sc_imlen - 2];
+ msg = sc->sc_imess[sc->sc_imlen - 1];
NCR_PHASE(("<stat:(%x,%x)>", ecb->stat, msg));
if (msg == MSG_CMDCOMPLETE) {
ecb->dleft = (ecb->flags & ECB_TENTATIVE_DONE)
@@ -1930,14 +2503,19 @@ printf("<<RESELECT CONT'd>>");
} else
printf("%s: STATUS_PHASE: msg %d\n",
sc->sc_dev.dv_xname, msg);
+ sc->sc_imlen = 0;
NCRCMD(sc, NCRCMD_MSGOK);
goto shortcut; /* ie. wait for disconnect */
}
break;
default:
- panic("%s: invalid state: %d",
+ /* Don't panic: reset. */
+ printf("%s: invalid state: %d",
sc->sc_dev.dv_xname,
sc->sc_state);
+ ncr53c9x_scsi_reset(sc);
+ goto out;
+ break;
}
/*
@@ -1945,7 +2523,7 @@ printf("<<RESELECT CONT'd>>");
* have a current command working the SCSI bus.
*/
if (sc->sc_state != NCR_CONNECTED || ecb == NULL) {
- panic("esp no nexus");
+ panic("ncr53c9x no nexus");
}
switch (sc->sc_phase) {
@@ -1955,10 +2533,13 @@ printf("<<RESELECT CONT'd>>");
sc->sc_prevphase = MESSAGE_OUT_PHASE;
break;
case MESSAGE_IN_PHASE:
+msgin:
NCR_PHASE(("MESSAGE_IN_PHASE "));
- sc->sc_prevphase = MESSAGE_IN_PHASE;
if (sc->sc_espintr & NCRINTR_BS) {
- NCRCMD(sc, NCRCMD_FLUSH);
+ if ((sc->sc_rev != NCR_VARIANT_FAS366) ||
+ !(sc->sc_espstat2 & FAS_STAT2_EMPTY)) {
+ NCRCMD(sc, NCRCMD_FLUSH);
+ }
sc->sc_flags |= NCR_WAITI;
NCRCMD(sc, NCRCMD_TRANS);
} else if (sc->sc_espintr & NCRINTR_FC) {
@@ -1970,6 +2551,9 @@ printf("<<RESELECT CONT'd>>");
sc->sc_espstep);
}
sc->sc_flags &= ~NCR_WAITI;
+ ncr53c9x_rdfifo(sc,
+ (sc->sc_prevphase == sc->sc_phase) ?
+ NCR_RDFIFO_CONTINUE : NCR_RDFIFO_START);
ncr53c9x_msgin(sc);
} else {
printf("%s: MSGIN: weird bits: "
@@ -1978,6 +2562,7 @@ printf("<<RESELECT CONT'd>>");
sc->sc_espintr, sc->sc_espstat,
sc->sc_espstep);
}
+ sc->sc_prevphase = MESSAGE_IN_PHASE;
goto shortcut; /* i.e. expect data to be ready */
break;
case COMMAND_PHASE:
@@ -1993,9 +2578,9 @@ printf("<<RESELECT CONT'd>>");
ecb->cmd.cmd.opcode, ecb->clen));
if (NCR_READ_REG(sc, NCR_FFLAG) & NCRFIFO_FF) {
NCRCMD(sc, NCRCMD_FLUSH);
- DELAY(1);
+/* DELAY(1); */
}
- if (ncr53c9x_dmaselect) {
+ if (sc->sc_features & NCR_F_DMASELECT) {
size_t size;
/* setup DMA transfer for command */
size = ecb->clen;
@@ -2004,11 +2589,7 @@ printf("<<RESELECT CONT'd>>");
NCRDMA_SETUP(sc, &sc->sc_cmdp, &sc->sc_cmdlen,
0, &size);
/* Program the SCSI counter */
- NCR_WRITE_REG(sc, NCR_TCL, size);
- NCR_WRITE_REG(sc, NCR_TCM, size >> 8);
- if (sc->sc_cfg2 & NCRCFG2_FE) {
- NCR_WRITE_REG(sc, NCR_TCH, size >> 16);
- }
+ NCR_SET_COUNT(sc, size);
/* load the count in */
NCRCMD(sc, NCRCMD_NOP|NCRCMD_DMA);
@@ -2017,11 +2598,7 @@ printf("<<RESELECT CONT'd>>");
NCRCMD(sc, NCRCMD_TRANS | NCRCMD_DMA);
NCRDMA_GO(sc);
} else {
- u_char *cmd = (u_char *)&ecb->cmd.cmd;
- int i;
- /* Now the command into the FIFO */
- for (i = 0; i < ecb->clen; i++)
- NCR_WRITE_REG(sc, NCR_FIFO, *cmd++);
+ ncr53c9x_wrfifo(sc, (u_char *)&ecb->cmd.cmd, ecb->clen);
NCRCMD(sc, NCRCMD_TRANS);
}
sc->sc_prevphase = COMMAND_PHASE;
@@ -2047,11 +2624,8 @@ printf("<<RESELECT CONT'd>>");
ecb->flags &= ~ECB_TENTATIVE_DONE;
/* Program the SCSI counter */
- NCR_WRITE_REG(sc, NCR_TCL, size);
- NCR_WRITE_REG(sc, NCR_TCM, size >> 8);
- if (sc->sc_cfg2 & NCRCFG2_FE) {
- NCR_WRITE_REG(sc, NCR_TCH, size >> 16);
- }
+ NCR_SET_COUNT(sc, size);
+
/* load the count in */
NCRCMD(sc, NCRCMD_NOP|NCRCMD_DMA);
@@ -2107,9 +2681,22 @@ shortcut:
* The delay is a heuristic. It is 2 when at 20Mhz, 2 at 25Mhz and 1
* at 40Mhz. This needs testing.
*/
- DELAY(50/sc->sc_freq);
- if (NCRDMA_ISINTR(sc))
- goto again;
+ {
+ struct timeval wait, cur;
+
+ microtime(&wait);
+ wait.tv_usec += 50/sc->sc_freq;
+ if (wait.tv_usec > 1000000) {
+ wait.tv_sec++;
+ wait.tv_usec -= 1000000;
+ }
+ do {
+ if (NCRDMA_ISINTR(sc))
+ goto again;
+ microtime(&cur);
+ } while (cur.tv_sec <= wait.tv_sec &&
+ cur.tv_usec <= wait.tv_usec);
+ }
goto out;
}
@@ -2124,6 +2711,8 @@ ncr53c9x_abort(sc, ecb)
ecb->flags |= ECB_ABORT;
if (ecb == sc->sc_nexus) {
+ int timeout;
+
/*
* If we're still selecting, the message will be scheduled
* after selection is complete.
@@ -2134,16 +2723,14 @@ ncr53c9x_abort(sc, ecb)
/*
* Reschedule timeout.
*/
- timeout_add(&ecb->to, (ecb->timeout * hz) / 1000);
+ if (hz > 100 && timeout > 1000)
+ timeout = (timeout / 1000) * hz;
+ else
+ timeout = (timeout * hz) / 1000;
+ timeout_add(&ecb->to, timeout);
} else {
- /* The command should be on the nexus list */
- if ((ecb->flags & ECB_NEXUS) == 0) {
- sc_print_addr(ecb->xs->sc_link);
- printf("ncr53c9x_abort: not NEXUS\n");
- ncr53c9x_init(sc, 1);
- }
/*
- * Just leave the command on the nexus list.
+ * Just leave the command where it is.
* XXX - what choice do we have but to reset the SCSI
* eventually?
*/
@@ -2203,3 +2790,35 @@ ncr53c9x_timeout(arg)
splx(s);
}
+
+void
+ncr53c9x_watch(arg)
+ void *arg;
+{
+ struct ncr53c9x_softc *sc = (struct ncr53c9x_softc *)arg;
+ struct ncr53c9x_tinfo *ti;
+ struct ncr53c9x_linfo *li;
+ int t, s;
+ /* Delete any structures that have not been used in 10min. */
+ time_t old = time.tv_sec - (10*60);
+
+ s = splbio();
+ for (t=0; t<NCR_NTARG; t++) {
+ ti = &sc->sc_tinfo[t];
+ li = ti->luns.lh_first;
+ while (li) {
+ if (li->last_used < old && !li->untagged && !li->used) {
+ if (li->lun < NCR_NLUN)
+ ti->lun[li->lun] = NULL;
+ LIST_REMOVE(li, link);
+ free(li, M_DEVBUF);
+ /* Restart the search at the beginning */
+ li = ti->luns.lh_first;
+ continue;
+ }
+ li = li->link.le_next;
+ }
+ }
+ splx(s);
+ timeout_add(&sc->sc_watchdog, 60*hz);
+}
diff --git a/sys/dev/ic/ncr53c9xreg.h b/sys/dev/ic/ncr53c9xreg.h
index cfebd6983c5..fe21b94a01f 100644
--- a/sys/dev/ic/ncr53c9xreg.h
+++ b/sys/dev/ic/ncr53c9xreg.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: ncr53c9xreg.h,v 1.5 2001/08/21 14:22:27 jason Exp $ */
+/* $OpenBSD: ncr53c9xreg.h,v 1.6 2001/09/27 04:01:42 jason Exp $ */
/* $NetBSD: ncr53c9xreg.h,v 1.4 1997/05/17 20:56:55 pk Exp $ */
/*
@@ -83,6 +83,8 @@
#define NCRSTAT_PHASE 0x07 /* Phase bits */
#define NCR_SELID 0x04 /* WO - Select/Reselect Bus ID */
+#define NCR_BUSID_HME 0x10 /* XXX HME reselect ID */
+#define NCR_BUSID_HME32 0x40 /* XXX HME to select more than 16 */
#define NCR_INTR 0x05 /* RO - Interrupt */
#define NCRINTR_SBR 0x80 /* SCSI Bus Reset */
@@ -139,6 +141,8 @@
#define NCRCFG2_BPA 0x04 /* Target Bad Parity Abort */
#define NCRCFG2_RPE 0x02 /* Register Parity Error */
#define NCRCFG2_DPE 0x01 /* DMA Parity Error */
+#define NCRCFG2_HMEFE 0x10 /* HME feature enable */
+#define NCRCFG2_HME32 0x80 /* HME 32 extended */
/* Config #3 only on 53C9X */
#define NCR_CFG3 0x0c /* RW - Configuration #3 */
@@ -178,6 +182,16 @@
#define NCRF9XCFG3_ADMA 0x02 /* Alternate DMA Mode */
#define NCRF9XCFG3_T8M 0x01 /* Threshold 8 Mode */
+/* Config #3 on FAS366 */
+#define NCRFASCFG3_OBAUTO 0x80 /* auto push odd-byte to dma */
+#define NCRFASCFG3_EWIDE 0x40 /* Enable Wide-SCSI */
+#define NCRFASCFG3_IDBIT3 0x20 /* Bit 3 of HME SCSI-ID */
+#define NCRFASCFG3_IDRESCHK 0x10 /* ID message checking */
+#define NCRFASCFG3_QUENB 0x08 /* 3-byte msg support */
+#define NCRFASCFG3_CDB10 0x04 /* group 2 scsi-2 support */
+#define NCRFASCFG3_FASTSCSI 0x02 /* 10 MB/S fast scsi mode */
+#define NCRFASCFG3_FASTCLK 0x01 /* fast clock mode */
+
/* Config #4 only on ESP406/FAS408 */
#define NCR_CFG4 0x0d /* RW - Configuration #4 */
#define NCRCFG4_CRS1 0x80 /* Select register set #1 */
@@ -260,3 +274,13 @@
#define NCR_RCH 0xf /* Recommand counter high */
#define NCR_UID NCR_RCL /* fas366 part-uniq id */
+/* status register #2 definitions (read only) */
+#define NCR_STAT2 NCR_CCF
+#define FAS_STAT2_SEQCNT 0x01 /* Sequence counter bit 7-3 enabled */
+#define FAS_STAT2_FLATCHED 0x02 /* FIFO flags register latched */
+#define FAS_STAT2_CLATCHED 0x04 /* Xfer cntr & recommand ctr latched */
+#define FAS_STAT2_CACTIVE 0x08 /* Command register is active */
+#define FAS_STAT2_SCSI16 0x10 /* SCSI interface is wide */
+#define FAS_STAT2_ISHUTTLE 0x20 /* FIFO Top register contains 1 byte */
+#define FAS_STAT2_OSHUTTLE 0x40 /* next byte from FIFO is MSB */
+#define FAS_STAT2_EMPTY 0x80 /* FIFO is empty */
diff --git a/sys/dev/ic/ncr53c9xvar.h b/sys/dev/ic/ncr53c9xvar.h
index 7bf24d63808..59d9e3556c7 100644
--- a/sys/dev/ic/ncr53c9xvar.h
+++ b/sys/dev/ic/ncr53c9xvar.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: ncr53c9xvar.h,v 1.8 2001/08/21 14:22:27 jason Exp $ */
+/* $OpenBSD: ncr53c9xvar.h,v 1.9 2001/09/27 04:01:42 jason Exp $ */
/* $NetBSD: ncr53c9xvar.h,v 1.13 1998/05/26 23:17:34 thorpej Exp $ */
/*-
@@ -72,6 +72,10 @@
/* Set this to 1 for normal debug, or 2 for per-target tracing. */
#define NCR53C9X_DEBUG 1
+/* Wide or differential can have 16 targets */
+#define NCR_NTARG 8
+#define NCR_NLUN 8
+
#define NCR_ABORT_TIMEOUT 2000 /* time to wait for abort */
#define NCR_SENSE_TIMEOUT 1000 /* time to wait for sense */
@@ -106,7 +110,7 @@ struct ncr53c9x_ecb {
struct scsi_xfer *xs; /* SCSI xfer ctrl block from above */
int flags;
#define ECB_ALLOC 0x01
-#define ECB_NEXUS 0x02
+#define ECB_READY 0x02
#define ECB_SENSE 0x04
#define ECB_ABORT 0x40
#define ECB_RESET 0x80
@@ -115,14 +119,15 @@ struct ncr53c9x_ecb {
struct timeout to;
struct {
- u_char id; /* Selection Id msg */
+ u_char msg[3]; /* Selection Id msg */
struct scsi_generic cmd; /* SCSI command block */
} cmd;
- int clen; /* Size of command in cmd.cmd */
char *daddr; /* Saved data pointer */
- int dleft; /* Residue */
- u_char stat; /* SCSI status byte */
- u_char pad[3];
+ int clen; /* Size of command in cmd.cmd */
+ int dleft; /* Residue */
+ u_char stat; /* SCSI status byte */
+ u_char tag[2]; /* TAG bytes */
+ u_char pad[1];
#if NCR53C9X_DEBUG > 1
char trace[1000];
@@ -140,17 +145,34 @@ struct ncr53c9x_ecb {
#endif
/*
- * Some info about each (possible) target on the SCSI bus. This should
- * probably have been a "per target+lunit" structure, but we'll leave it at
- * this for now. Is there a way to reliably hook it up to sc->fordriver??
+ * Some info about ech (possible) target and LUN on the SCSI bus.
+ *
+ * SCSI I and II devices can have up to 8 LUNs, each with up to 256
+ * outstanding tags. SCSI III devices have 64-bit LUN identifiers
+ * that can be sparsely allocated.
+ *
+ * Since SCSI II devices can have up to 8 LUNs, we use an array
+ * of 8 pointers to ncr53c9x_linfo structures for fast lookup.
+ * Longer LUNs need to traverse the linked list.
*/
+
+struct ncr53c9x_linfo {
+ int64_t lun;
+ LIST_ENTRY(ncr53c9x_linfo) link;
+ time_t last_used;
+ unsigned char used; /* # slots in use */
+ unsigned char avail; /* where to start scanning */
+ unsigned char busy;
+ struct ncr53c9x_ecb *untagged;
+ struct ncr53c9x_ecb *queued[256];
+};
+
struct ncr53c9x_tinfo {
- int cmds; /* #commands processed */
- int dconns; /* #disconnects */
- int touts; /* #timeouts */
- int perrs; /* #parity errors */
- int senses; /* #request sense commands sent */
- ushort lubusy; /* What local units/subr. are busy? */
+ int cmds; /* # of commands processed */
+ int dconns; /* # of disconnects */
+ int touts; /* # of timeouts */
+ int perrs; /* # of parity errors */
+ int senses; /* # of request sense commands sent */
u_char flags;
#define T_NEED_TO_RESET 0x01 /* Should send a BUS_DEV_RESET */
#define T_NEGOTIATE 0x02 /* (Re)Negotiate synchronous options */
@@ -158,10 +180,20 @@ struct ncr53c9x_tinfo {
#define T_SYNCMODE 0x08 /* sync mode has been negotiated */
#define T_SYNCHOFF 0x10 /* .. */
#define T_RSELECTOFF 0x20 /* .. */
+#define T_TAG 0x40 /* TAG QUEUEs are on */
+#define T_WIDE 0x80 /* Negotiate wide options */
u_char period; /* Period suggestion */
u_char offset; /* Offset suggestion */
- u_char pad[3];
-} tinfo_t;
+ u_char cfg3; /* per target config 3 */
+ u_char nextag; /* Next available tag */
+ u_char width; /* width suggestion */
+ LIST_HEAD(lun_list, ncr53c9x_linfo) luns;
+ struct ncr53c9x_linfo *lun[NCR_NLUN]; /* For speedy lookups */
+};
+
+/* Look up a lun in a tinfo */
+#define TINFO_LUN(t, l) ((((l) < NCR_NLUN) && (((t)->lun[(l)]) != NULL)) ? \
+ ((t)->lun[(l)]) : ncr53c9x_lunsearch((t), (int64_t)(l)))
/* Register a linenumber (for debugging) */
#define LOGLINE(p)
@@ -237,6 +269,7 @@ struct ncr53c9x_softc {
struct device sc_dev; /* us as a device */
struct evcnt sc_intrcnt; /* intr count */
+ struct timeout sc_watchdog; /* periodic timer */
struct scsi_link sc_link; /* scsi lint struct */
struct ncr53c9x_glue *sc_glue; /* glue to MD code */
@@ -257,16 +290,14 @@ struct ncr53c9x_softc {
u_char sc_espintr;
u_char sc_espstat;
u_char sc_espstep;
+ u_char sc_espstat2;
u_char sc_espfflags;
/* Lists of command blocks */
- TAILQ_HEAD(ecb_list, ncr53c9x_ecb) free_list,
- ready_list,
- nexus_list;
+ TAILQ_HEAD(ecb_list, ncr53c9x_ecb) ready_list;
struct ncr53c9x_ecb *sc_nexus; /* Current command */
- struct ncr53c9x_ecb sc_ecb[3*8]; /* Three per target */
- struct ncr53c9x_tinfo sc_tinfo[8];
+ struct ncr53c9x_tinfo sc_tinfo[NCR_NTARG];
/* Data about the current nexus (updated for every cmd switch) */
caddr_t sc_dp; /* Current data pointer */
@@ -281,9 +312,11 @@ struct ncr53c9x_softc {
u_char sc_lastcmd;
/* Message stuff */
- u_char sc_msgpriq; /* One or more messages to send (encoded) */
- u_char sc_msgout; /* What message is on its way out? */
- u_char sc_msgoutq; /* What messages have been sent so far? */
+ u_short sc_msgify; /* IDENTIFY message associated with this nexus */
+ u_short sc_msgout; /* What message is on its way out? */
+ u_short sc_msgpriq; /* One or more messages to send (encoded) */
+ u_short sc_msgoutq; /* What messages have been sent so far? */
+
u_char *sc_omess; /* MSGOUT buffer */
caddr_t sc_omp; /* Message pointer (for multibyte messages) */
size_t sc_omlen;
@@ -307,11 +340,12 @@ struct ncr53c9x_softc {
#define NCR_IDLE 1 /* waiting for something to do */
#define NCR_SELECTING 2 /* SCSI command is arbiting */
#define NCR_RESELECTED 3 /* Has been reselected */
-#define NCR_CONNECTED 4 /* Actively using the SCSI bus */
-#define NCR_DISCONNECT 5 /* MSG_DISCONNECT received */
-#define NCR_CMDCOMPLETE 6 /* MSG_CMDCOMPLETE received */
-#define NCR_CLEANING 7
-#define NCR_SBR 8 /* Expect a SCSI RST because we commanded it */
+#define NCR_IDENTIFIED 4 /* Has gotten IFY but not TAG */
+#define NCR_CONNECTED 5 /* Actively using the SCSI bus */
+#define NCR_DISCONNECT 6 /* MSG_DISCONNECT received */
+#define NCR_CMDCOMPLETE 7 /* MSG_CMDCOMPLETE received */
+#define NCR_CLEANING 8
+#define NCR_SBR 9 /* Expect a SCSI RST because we commanded it */
/* values for sc_flags */
#define NCR_DROP_MSGI 0x01 /* Discard all msgs (parity err detected) */
@@ -326,16 +360,19 @@ struct ncr53c9x_softc {
/* values for sc_features */
#define NCR_F_HASCFG3 0x01 /* chip has CFG3 register */
#define NCR_F_FASTSCSI 0x02 /* chip supports Fast mode */
+#define NCR_F_DMASELECT 0x04 /* can do dma select */
+#define NCR_F_SELATN3 0x08 /* can do selatn3 */
/* values for sc_msgout */
-#define SEND_DEV_RESET 0x01
-#define SEND_PARITY_ERROR 0x02
-#define SEND_INIT_DET_ERR 0x04
-#define SEND_REJECT 0x08
-#define SEND_IDENTIFY 0x10
-#define SEND_ABORT 0x20
-#define SEND_SDTR 0x40
-#define SEND_WDTR 0x80
+#define SEND_DEV_RESET 0x0001
+#define SEND_PARITY_ERROR 0x0002
+#define SEND_INIT_DET_ERR 0x0004
+#define SEND_REJECT 0x0008
+#define SEND_IDENTIFY 0x0010
+#define SEND_ABORT 0x0020
+#define SEND_WDTR 0x0040
+#define SEND_SDTR 0x0080
+#define SEND_TAG 0x0100
/* SCSI Status codes */
#define ST_MASK 0x3e /* bit 0,6,7 is reserved */
@@ -369,11 +406,11 @@ struct ncr53c9x_softc {
(*(sc)->sc_glue->gl_write_reg)((sc), (reg), (val))
#ifdef NCR53C9X_DEBUG
-#define NCRCMD(sc, cmd) do { \
- if (ncr53c9x_debug & NCR_SHOWCCMDS) \
- printf("<cmd:0x%x>", (unsigned)cmd); \
- sc->sc_lastcmd = cmd; \
- NCR_WRITE_REG(sc, NCR_CMD, cmd); \
+#define NCRCMD(sc, cmd) do { \
+ if (ncr53c9x_debug & NCR_SHOWCCMDS) \
+ printf("<cmd:0x%x %d>", (unsigned)cmd, __LINE__); \
+ sc->sc_lastcmd = cmd; \
+ NCR_WRITE_REG(sc, NCR_CMD, cmd); \
} while (0)
#else
#define NCRCMD(sc, cmd) NCR_WRITE_REG(sc, NCR_CMD, cmd)
@@ -403,5 +440,3 @@ int ncr53c9x_scsi_cmd __P((struct scsi_xfer *));
void ncr53c9x_reset __P((struct ncr53c9x_softc *));
int ncr53c9x_intr __P((void *));
void ncr53c9x_init __P((struct ncr53c9x_softc *, int));
-
-extern int ncr53c9x_dmaselect;
diff --git a/sys/dev/pci/pcscp.c b/sys/dev/pci/pcscp.c
index dd540935cd3..1f1763d09e4 100644
--- a/sys/dev/pci/pcscp.c
+++ b/sys/dev/pci/pcscp.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: pcscp.c,v 1.6 2001/08/26 03:32:22 jason Exp $ */
+/* $OpenBSD: pcscp.c,v 1.7 2001/09/27 04:01:42 jason Exp $ */
/* $NetBSD: pcscp.c,v 1.11 2000/11/14 18:42:58 thorpej Exp $ */
/*-
@@ -346,7 +346,7 @@ pcscp_attach(parent, self, aux)
ncr53c9x_attach(sc, &pcscp_adapter, NULL);
/* Turn on target selection using the `dma' method */
- ncr53c9x_dmaselect = 1;
+ sc->sc_features |= NCR_F_DMASELECT;
}
/*
diff --git a/sys/dev/sbus/esp_sbus.c b/sys/dev/sbus/esp_sbus.c
index 8d8f0b2a47a..d9550d80493 100644
--- a/sys/dev/sbus/esp_sbus.c
+++ b/sys/dev/sbus/esp_sbus.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: esp_sbus.c,v 1.4 2001/09/26 00:03:34 jason Exp $ */
+/* $OpenBSD: esp_sbus.c,v 1.5 2001/09/27 04:01:42 jason Exp $ */
/* $NetBSD: esp_sbus.c,v 1.14 2001/04/25 17:53:37 bouyer Exp $ */
/*-
@@ -538,13 +538,8 @@ espattach(esc, gluep)
evcnt_attach(&sc->sc_dev, "intr", &sc->sc_intrcnt);
/* Turn on target selection using the `dma' method */
-#ifdef notyet
if (sc->sc_rev != NCR_VARIANT_FAS366)
sc->sc_features |= NCR_F_DMASELECT;
-#else
- if (sc->sc_rev != NCR_VARIANT_FAS366)
- ncr53c9x_dmaselect = 1;
-#endif
/* Do the common parts of attachment. */
ncr53c9x_attach(sc, &esp_switch, &esp_dev);