#ifdef DDB
#define integrate
#else
#define integrate static __inline
#endif
#define SPC_USE_SYNCHRONOUS 0
#define SPC_SYNC_REQ_ACK_OFS 8
#define SPC_USE_WIDE 0
#define SPC_MAX_WIDTH 0
#define SPC_MSG_MAX_ATTEMPT 3
#define SPC_MSGIN_SPIN 1
#define SPC_MSGOUT_SPIN 1
#define SPC_ABORT_TIMEOUT 2000
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/kernel.h>
#include <sys/errno.h>
#include <sys/ioctl.h>
#include <sys/device.h>
#include <sys/buf.h>
#include <sys/proc.h>
#include <sys/queue.h>
#include <machine/intr.h>
#include <machine/bus.h>
#include <scsi/scsi_all.h>
#include <scsi/scsi_message.h>
#include <scsi/scsiconf.h>
#include <luna88k/dev/mb89352reg.h>
#include <luna88k/dev/mb89352var.h>
#ifndef DDB
#define db_enter() panic("should call debugger here (mb89352.c)")
#endif
#ifdef SPC_DEBUG
int spc_debug = 0x00;
#endif
void spc_done (struct spc_softc *, struct spc_acb *);
void spc_dequeue (struct spc_softc *, struct spc_acb *);
void spc_scsi_cmd (struct scsi_xfer *);
int spc_poll (struct spc_softc *, struct scsi_xfer *, int);
integrate void spc_sched_msgout(struct spc_softc *, u_char);
integrate void spc_setsync(struct spc_softc *, struct spc_tinfo *);
void spc_select (struct spc_softc *, struct spc_acb *);
void spc_timeout (void *);
void spc_scsi_reset (struct spc_softc *);
void spc_reset (struct spc_softc *);
void spc_acb_free (void *, void *);
void *spc_acb_alloc (void *);
int spc_reselect (struct spc_softc *, int);
void spc_sense (struct spc_softc *, struct spc_acb *);
void spc_msgin (struct spc_softc *);
void spc_abort (struct spc_softc *, struct spc_acb *);
void spc_msgout (struct spc_softc *);
int spc_dataout_pio (struct spc_softc *, u_char *, int);
int spc_datain_pio (struct spc_softc *, u_char *, int);
#ifdef SPC_DEBUG
void spc_print_acb (struct spc_acb *);
void spc_dump_driver (struct spc_softc *);
void spc_dump89352 (struct spc_softc *);
void spc_show_scsi_cmd(struct spc_acb *);
void spc_print_active_acb(void);
#endif
extern struct cfdriver spc_cd;
#define breathe() \
do { \
asm volatile ("or %r0, %r0, %r0"); \
asm volatile ("or %r0, %r0, %r0"); \
asm volatile ("or %r0, %r0, %r0"); \
asm volatile ("or %r0, %r0, %r0"); \
} while (0)
void
spc_attach(struct spc_softc *sc, const struct scsi_adapter *adapter)
{
struct scsibus_attach_args saa;
SPC_TRACE(("spc_attach "));
sc->sc_state = SPC_INIT;
sc->sc_freq = 20;
#if SPC_USE_SYNCHRONOUS
sc->sc_minsync = (2 * 250) / sc->sc_freq;
sc->sc_maxsync = (9 * 250) / sc->sc_freq;
#endif
spc_init(sc);
saa.saa_adapter_softc = sc;
saa.saa_adapter_target = sc->sc_initiator;
saa.saa_adapter = adapter;
saa.saa_luns = saa.saa_adapter_buswidth = 8;
saa.saa_openings = 2;
saa.saa_pool = &sc->sc_iopool;
saa.saa_flags = saa.saa_quirks = 0;
saa.saa_wwpn = saa.saa_wwnn = 0;
config_found(&sc->sc_dev, &saa, scsiprint);
}
void
spc_reset(struct spc_softc *sc)
{
bus_space_tag_t iot = sc->sc_iot;
bus_space_handle_t ioh = sc->sc_ioh;
SPC_TRACE(("spc_reset "));
bus_space_write_1(iot, ioh, SCTL, SCTL_DISABLE | SCTL_CTRLRST);
bus_space_write_1(iot, ioh, SCMD, 0);
bus_space_write_1(iot, ioh, TMOD, 0);
bus_space_write_1(iot, ioh, PCTL, 0);
bus_space_write_1(iot, ioh, TEMP, 0);
bus_space_write_1(iot, ioh, TCH, 0);
bus_space_write_1(iot, ioh, TCM, 0);
bus_space_write_1(iot, ioh, TCL, 0);
bus_space_write_1(iot, ioh, INTS, 0);
bus_space_write_1(iot, ioh, SCTL,
SCTL_DISABLE | SCTL_ABRT_ENAB | SCTL_PARITY_ENAB | SCTL_RESEL_ENAB);
bus_space_write_1(iot, ioh, BDID, sc->sc_initiator);
delay(400);
bus_space_write_1(iot, ioh, SCTL,
bus_space_read_1(iot, ioh, SCTL) & ~SCTL_DISABLE);
}
void
spc_scsi_reset(struct spc_softc *sc)
{
bus_space_tag_t iot = sc->sc_iot;
bus_space_handle_t ioh = sc->sc_ioh;
SPC_TRACE(("spc_scsi_reset "));
bus_space_write_1(iot, ioh, SCMD, bus_space_read_1(iot, ioh, SCMD) | SCMD_RST);
delay(500);
bus_space_write_1(iot, ioh, SCMD, bus_space_read_1(iot, ioh, SCMD) & ~SCMD_RST);
delay(50);
}
void
spc_init(struct spc_softc *sc)
{
struct spc_acb *acb;
int r;
SPC_TRACE(("spc_init "));
spc_reset(sc);
spc_scsi_reset(sc);
spc_reset(sc);
if (sc->sc_state == SPC_INIT) {
TAILQ_INIT(&sc->ready_list);
TAILQ_INIT(&sc->nexus_list);
TAILQ_INIT(&sc->free_list);
mtx_init(&sc->sc_acb_mtx, IPL_BIO);
scsi_iopool_init(&sc->sc_iopool, sc, spc_acb_alloc, spc_acb_free);
sc->sc_nexus = NULL;
acb = sc->sc_acb;
bzero(acb, sizeof(sc->sc_acb));
for (r = 0; r < sizeof(sc->sc_acb) / sizeof(*acb); r++) {
TAILQ_INSERT_TAIL(&sc->free_list, acb, chain);
acb++;
}
bzero(&sc->sc_tinfo, sizeof(sc->sc_tinfo));
} else {
sc->sc_state = SPC_CLEANING;
if ((acb = sc->sc_nexus) != NULL) {
acb->xs->error = XS_DRIVER_STUFFUP;
timeout_del(&acb->xs->stimeout);
spc_done(sc, acb);
}
while ((acb = TAILQ_FIRST(&sc->nexus_list)) != NULL) {
acb->xs->error = XS_DRIVER_STUFFUP;
timeout_del(&acb->xs->stimeout);
spc_done(sc, acb);
}
}
sc->sc_prevphase = PH_INVALID;
for (r = 0; r < 8; r++) {
struct spc_tinfo *ti = &sc->sc_tinfo[r];
ti->flags = 0;
#if SPC_USE_SYNCHRONOUS
ti->flags |= DO_SYNC;
ti->period = sc->sc_minsync;
ti->offset = SPC_SYNC_REQ_ACK_OFS;
#else
ti->period = ti->offset = 0;
#endif
#if SPC_USE_WIDE
ti->flags |= DO_WIDE;
ti->width = SPC_MAX_WIDTH;
#else
ti->width = 0;
#endif
}
sc->sc_state = SPC_IDLE;
bus_space_write_1(sc->sc_iot, sc->sc_ioh, SCTL,
bus_space_read_1(sc->sc_iot, sc->sc_ioh, SCTL) | SCTL_INTR_ENAB);
}
void
spc_acb_free(void *xsc, void *xacb)
{
struct spc_softc *sc = xsc;
struct spc_acb *acb = xacb;
SPC_TRACE(("spc_acb_free "));
acb->flags = 0;
mtx_enter(&sc->sc_acb_mtx);
TAILQ_INSERT_HEAD(&sc->free_list, acb, chain);
mtx_leave(&sc->sc_acb_mtx);
}
void *
spc_acb_alloc(void *xsc)
{
struct spc_softc *sc = xsc;
struct spc_acb *acb;
SPC_TRACE(("spc_acb_alloc "));
mtx_enter(&sc->sc_acb_mtx);
acb = TAILQ_FIRST(&sc->free_list);
if (acb)
TAILQ_REMOVE(&sc->free_list, acb, chain);
mtx_leave(&sc->sc_acb_mtx);
return acb;
}
void
spc_scsi_cmd(struct scsi_xfer *xs)
{
struct scsi_link *sc_link = xs->sc_link;
struct spc_softc *sc = sc_link->bus->sb_adapter_softc;
struct spc_acb *acb;
int s, flags;
SPC_TRACE(("spc_scsi_cmd "));
SPC_CMDS(("[0x%x, %d]->%d ", (int)xs->cmd.opcode, xs->cmdlen,
sc_link->target));
flags = xs->flags;
acb = xs->io;
acb->xs = xs;
timeout_set(&xs->stimeout, spc_timeout, acb);
if (xs->flags & SCSI_RESET) {
acb->flags |= ACB_RESET;
acb->scsi_cmd_length = 0;
acb->data_length = 0;
} else {
bcopy(&xs->cmd, &acb->scsi_cmd, xs->cmdlen);
acb->scsi_cmd_length = xs->cmdlen;
acb->data_addr = xs->data;
acb->data_length = xs->datalen;
}
acb->target_stat = 0;
s = splbio();
TAILQ_INSERT_TAIL(&sc->ready_list, acb, chain);
if (sc->sc_state == SPC_IDLE)
spc_sched(sc);
splx(s);
if ((flags & SCSI_POLL) == 0)
return;
s = splbio();
if (spc_poll(sc, xs, xs->timeout)) {
spc_timeout(acb);
if (spc_poll(sc, xs, xs->timeout))
spc_timeout(acb);
}
splx(s);
}
int
spc_poll(struct spc_softc *sc, struct scsi_xfer *xs, int count)
{
bus_space_tag_t iot = sc->sc_iot;
bus_space_handle_t ioh = sc->sc_ioh;
SPC_TRACE(("spc_poll "));
while (count) {
if (bus_space_read_1(iot, ioh, INTS) != 0)
spc_intr(sc);
if ((xs->flags & ITSDONE) != 0)
return 0;
delay(1000);
count--;
}
return 1;
}
integrate void
spc_sched_msgout(struct spc_softc *sc, u_char m)
{
bus_space_tag_t iot = sc->sc_iot;
bus_space_handle_t ioh = sc->sc_ioh;
SPC_TRACE(("spc_sched_msgout "));
if (sc->sc_msgpriq == 0)
bus_space_write_1(iot, ioh, SCMD, SCMD_SET_ATN);
sc->sc_msgpriq |= m;
}
integrate void
spc_setsync(struct spc_softc *sc, struct spc_tinfo *ti)
{
#if SPC_USE_SYNCHRONOUS
bus_space_tag_t iot = sc->sc_iot;
bus_space_handle_t ioh = sc->sc_ioh;
SPC_TRACE(("spc_setsync "));
if (ti->offset != 0)
bus_space_write_1(iot, ioh, TMOD,
((ti->period * sc->sc_freq) / 250 - 2) << 4 | ti->offset);
else
bus_space_write_1(iot, ioh, TMOD, 0);
#endif
}
void
spc_select(struct spc_softc *sc, struct spc_acb *acb)
{
struct scsi_link *sc_link = acb->xs->sc_link;
int target = sc_link->target;
struct spc_tinfo *ti = &sc->sc_tinfo[target];
bus_space_tag_t iot = sc->sc_iot;
bus_space_handle_t ioh = sc->sc_ioh;
SPC_TRACE(("spc_select "));
spc_setsync(sc, ti);
#if 0
bus_space_write_1(iot, ioh, SCMD, SCMD_SET_ATN);
#endif
bus_space_write_1(iot, ioh, PCTL, 0);
bus_space_write_1(iot, ioh, TEMP,
(1 << sc->sc_initiator) | (1 << target));
bus_space_write_1(iot, ioh, TCH, 2);
bus_space_write_1(iot, ioh, TCM, 113);
bus_space_write_1(iot, ioh, TCL, 3);
bus_space_write_1(iot, ioh, SCMD, SCMD_SELECT);
sc->sc_state = SPC_SELECTING;
}
int
spc_reselect(struct spc_softc *sc, int message)
{
u_char selid, target, lun;
struct spc_acb *acb;
struct scsi_link *sc_link;
struct spc_tinfo *ti;
SPC_TRACE(("spc_reselect "));
selid = sc->sc_selid & ~(1 << sc->sc_initiator);
if (selid & (selid - 1)) {
printf("%s: reselect with invalid selid %02x; "
"sending DEVICE RESET\n", sc->sc_dev.dv_xname, selid);
SPC_BREAK();
goto reset;
}
target = ffs(selid) - 1;
lun = message & 0x07;
TAILQ_FOREACH(acb, &sc->nexus_list, chain) {
sc_link = acb->xs->sc_link;
if (sc_link->target == target &&
sc_link->lun == lun)
break;
}
if (acb == NULL) {
printf("%s: reselect from target %d lun %d with no nexus; "
"sending ABORT\n", sc->sc_dev.dv_xname, target, lun);
SPC_BREAK();
goto abort;
}
TAILQ_REMOVE(&sc->nexus_list, acb, chain);
sc->sc_state = SPC_CONNECTED;
sc->sc_nexus = acb;
ti = &sc->sc_tinfo[target];
ti->lubusy |= (1 << lun);
spc_setsync(sc, ti);
if (acb->flags & ACB_RESET)
spc_sched_msgout(sc, SEND_DEV_RESET);
else if (acb->flags & ACB_ABORT)
spc_sched_msgout(sc, SEND_ABORT);
sc->sc_dp = acb->data_addr;
sc->sc_dleft = acb->data_length;
sc->sc_cp = (u_char *)&acb->scsi_cmd;
sc->sc_cleft = acb->scsi_cmd_length;
return (0);
reset:
spc_sched_msgout(sc, SEND_DEV_RESET);
return (1);
abort:
spc_sched_msgout(sc, SEND_ABORT);
return (1);
}
void
spc_sched(struct spc_softc *sc)
{
struct spc_acb *acb;
struct scsi_link *sc_link;
struct spc_tinfo *ti;
if (sc->sc_flags & SPC_INACTIVE)
return;
SPC_TRACE(("spc_sched "));
TAILQ_FOREACH(acb, &sc->ready_list, chain) {
sc_link = acb->xs->sc_link;
ti = &sc->sc_tinfo[sc_link->target];
if ((ti->lubusy & (1 << sc_link->lun)) == 0) {
SPC_MISC(("selecting %d:%d ",
sc_link->target, sc_link->lun));
TAILQ_REMOVE(&sc->ready_list, acb, chain);
sc->sc_nexus = acb;
spc_select(sc, acb);
return;
} else
SPC_MISC(("%d:%d busy\n",
sc_link->target, sc_link->lun));
}
SPC_MISC(("idle "));
}
void
spc_sense(struct spc_softc *sc, struct spc_acb *acb)
{
struct scsi_xfer *xs = acb->xs;
struct scsi_link *sc_link = xs->sc_link;
struct spc_tinfo *ti = &sc->sc_tinfo[sc_link->target];
struct scsi_sense *ss = (void *)&acb->scsi_cmd;
SPC_MISC(("requesting sense "));
bzero(ss, sizeof(*ss));
ss->opcode = REQUEST_SENSE;
ss->byte2 = sc_link->lun << 5;
ss->length = sizeof(struct scsi_sense_data);
acb->scsi_cmd_length = sizeof(*ss);
acb->data_addr = (char *)&xs->sense;
acb->data_length = sizeof(struct scsi_sense_data);
acb->flags |= ACB_SENSE;
ti->senses++;
if (acb->flags & ACB_NEXUS)
ti->lubusy &= ~(1 << sc_link->lun);
if (acb == sc->sc_nexus) {
spc_select(sc, acb);
} else {
spc_dequeue(sc, acb);
TAILQ_INSERT_HEAD(&sc->ready_list, acb, chain);
if (sc->sc_state == SPC_IDLE)
spc_sched(sc);
}
}
void
spc_done(struct spc_softc *sc, struct spc_acb *acb)
{
struct scsi_xfer *xs = acb->xs;
struct scsi_link *sc_link = xs->sc_link;
struct spc_tinfo *ti = &sc->sc_tinfo[sc_link->target];
SPC_TRACE(("spc_done "));
if (xs->error == XS_NOERROR) {
if (acb->flags & ACB_ABORT) {
xs->error = XS_DRIVER_STUFFUP;
} else if (acb->flags & ACB_SENSE) {
xs->error = XS_SENSE;
} else {
switch (acb->target_stat) {
case SCSI_CHECK:
xs->resid = acb->data_length;
xs->status = acb->target_stat;
spc_sense(sc, acb);
return;
case SCSI_BUSY:
xs->error = XS_BUSY;
break;
case SCSI_OK:
xs->resid = acb->data_length;
break;
default:
xs->error = XS_DRIVER_STUFFUP;
#ifdef SPC_DEBUG
printf("%s: spc_done: bad stat 0x%x\n",
sc->sc_dev.dv_xname, acb->target_stat);
#endif
break;
}
}
}
#ifdef SPC_DEBUG
if ((spc_debug & SPC_SHOWMISC) != 0) {
if (xs->resid != 0)
printf("resid=%d ", xs->resid);
if (xs->error == XS_SENSE)
printf("sense=0x%02x\n", xs->sense.error_code);
else
printf("error=%d\n", xs->error);
}
#endif
if (acb->flags & ACB_NEXUS)
ti->lubusy &= ~(1 << sc_link->lun);
if (acb == sc->sc_nexus) {
sc->sc_nexus = NULL;
sc->sc_state = SPC_IDLE;
spc_sched(sc);
} else
spc_dequeue(sc, acb);
ti->cmds++;
scsi_done(xs);
}
void
spc_dequeue(struct spc_softc *sc, struct spc_acb *acb)
{
SPC_TRACE(("spc_dequeue "));
if (acb->flags & ACB_NEXUS)
TAILQ_REMOVE(&sc->nexus_list, acb, chain);
else
TAILQ_REMOVE(&sc->ready_list, acb, chain);
}
void
spc_msgin(struct spc_softc *sc)
{
bus_space_tag_t iot = sc->sc_iot;
bus_space_handle_t ioh = sc->sc_ioh;
int n;
SPC_TRACE(("spc_msgin "));
if (sc->sc_prevphase == PH_MSGIN) {
n = sc->sc_imp - sc->sc_imess;
goto nextbyte;
}
sc->sc_flags &= ~SPC_DROP_MSGIN;
nextmsg:
n = 0;
sc->sc_imp = &sc->sc_imess[n];
nextbyte:
for (;;) {
u_int8_t intstat;
#if 0
for (;;) {
if ((bus_space_read_1(iot, ioh, PSNS) & PSNS_REQ) != 0)
break;
}
#endif
if (bus_space_read_1(iot, ioh, INTS) != 0) {
goto out;
}
if ((bus_space_read_1(iot, ioh, SERR) &
(SERR_SCSI_PAR|SERR_SPC_PAR)) != 0) {
sc->sc_flags |= SPC_DROP_MSGIN;
spc_sched_msgout(sc, SEND_PARITY_ERROR);
}
bus_space_write_1(iot, ioh, TCH, 0);
bus_space_write_1(iot, ioh, TCM, 0);
bus_space_write_1(iot, ioh, TCL, 1);
bus_space_write_1(iot, ioh, PCTL,
sc->sc_phase | PCTL_BFINT_ENAB);
#ifdef x68k
bus_space_write_1(iot, ioh, SCMD, SCMD_XFR);
#else
bus_space_write_1(iot, ioh, SCMD, SCMD_XFR | SCMD_PROG_XFR);
#endif
intstat = 0;
for (;;) {
if ((bus_space_read_1(iot, ioh, SSTS) & SSTS_DREG_EMPTY) == 0)
break;
if (intstat != 0)
goto out;
intstat = bus_space_read_1(iot, ioh, INTS);
}
if ((sc->sc_flags & SPC_DROP_MSGIN) == 0) {
if (n >= SPC_MAX_MSG_LEN) {
(void) bus_space_read_1(iot, ioh, DREG);
sc->sc_flags |= SPC_DROP_MSGIN;
spc_sched_msgout(sc, SEND_REJECT);
} else {
*sc->sc_imp++ = bus_space_read_1(iot, ioh, DREG);
n++;
if (n == 1 && IS1BYTEMSG(sc->sc_imess[0]))
break;
if (n == 2 && IS2BYTEMSG(sc->sc_imess[0]))
break;
if (n >= 3 && ISEXTMSG(sc->sc_imess[0]) &&
n == sc->sc_imess[1] + 2)
break;
}
} else
(void) bus_space_read_1(iot, ioh, DREG);
#if 0
while ((bus_space_read_1(iot, ioh, PSNS) & ACKI) != 0)
;
#endif
}
SPC_MISC(("n=%d imess=0x%02x ", n, sc->sc_imess[0]));
switch (sc->sc_state) {
struct spc_acb *acb;
struct scsi_link *sc_link;
struct spc_tinfo *ti;
case SPC_CONNECTED:
SPC_ASSERT(sc->sc_nexus != NULL);
acb = sc->sc_nexus;
ti = &sc->sc_tinfo[acb->xs->sc_link->target];
switch (sc->sc_imess[0]) {
case MSG_CMDCOMPLETE:
if (sc->sc_dleft < 0) {
sc_link = acb->xs->sc_link;
printf("%s: %ld extra bytes from %d:%d\n",
sc->sc_dev.dv_xname, -sc->sc_dleft,
sc_link->target, sc_link->lun);
acb->data_length = 0;
}
acb->xs->resid = acb->data_length = sc->sc_dleft;
sc->sc_state = SPC_CMDCOMPLETE;
break;
case MSG_PARITY_ERROR:
spc_sched_msgout(sc, sc->sc_lastmsg);
break;
case MSG_MESSAGE_REJECT:
SPC_MISC(("message rejected %02x ", sc->sc_lastmsg));
switch (sc->sc_lastmsg) {
#if SPC_USE_SYNCHRONOUS + SPC_USE_WIDE
case SEND_IDENTIFY:
ti->flags &= ~(DO_SYNC | DO_WIDE);
ti->period = ti->offset = 0;
spc_setsync(sc, ti);
ti->width = 0;
break;
#endif
#if SPC_USE_SYNCHRONOUS
case SEND_SDTR:
ti->flags &= ~DO_SYNC;
ti->period = ti->offset = 0;
spc_setsync(sc, ti);
break;
#endif
#if SPC_USE_WIDE
case SEND_WDTR:
ti->flags &= ~DO_WIDE;
ti->width = 0;
break;
#endif
case SEND_INIT_DET_ERR:
spc_sched_msgout(sc, SEND_ABORT);
break;
}
break;
case MSG_NOOP:
break;
case MSG_DISCONNECT:
ti->dconns++;
sc->sc_state = SPC_DISCONNECT;
break;
case MSG_SAVEDATAPOINTER:
acb->data_addr = sc->sc_dp;
acb->data_length = sc->sc_dleft;
break;
case MSG_RESTOREPOINTERS:
sc->sc_dp = acb->data_addr;
sc->sc_dleft = acb->data_length;
sc->sc_cp = (u_char *)&acb->scsi_cmd;
sc->sc_cleft = acb->scsi_cmd_length;
break;
case MSG_EXTENDED:
switch (sc->sc_imess[2]) {
#if SPC_USE_SYNCHRONOUS
case MSG_EXT_SDTR:
if (sc->sc_imess[1] != 3)
goto reject;
ti->period = sc->sc_imess[3];
ti->offset = sc->sc_imess[4];
ti->flags &= ~DO_SYNC;
if (ti->offset == 0) {
} else if (ti->period < sc->sc_minsync ||
ti->period > sc->sc_maxsync ||
ti->offset > 8) {
ti->period = ti->offset = 0;
spc_sched_msgout(sc, SEND_SDTR);
} else {
sc_print_addr(acb->xs->sc_link);
printf("sync, offset %d, "
"period %dnsec\n",
ti->offset, ti->period * 4);
}
spc_setsync(sc, ti);
break;
#endif
#if SPC_USE_WIDE
case MSG_EXT_WDTR:
if (sc->sc_imess[1] != 2)
goto reject;
ti->width = sc->sc_imess[3];
ti->flags &= ~DO_WIDE;
if (ti->width == 0) {
} else if (ti->width > SPC_MAX_WIDTH) {
ti->width = 0;
spc_sched_msgout(sc, SEND_WDTR);
} else {
sc_print_addr(acb->xs->sc_link);
printf("wide, width %d\n",
1 << (3 + ti->width));
}
break;
#endif
default:
printf("%s: unrecognized MESSAGE EXTENDED; "
"sending REJECT\n", sc->sc_dev.dv_xname);
SPC_BREAK();
goto reject;
}
break;
default:
printf("%s: unrecognized MESSAGE; sending REJECT\n",
sc->sc_dev.dv_xname);
SPC_BREAK();
reject:
spc_sched_msgout(sc, SEND_REJECT);
break;
}
break;
case SPC_RESELECTED:
if (!MSG_ISIDENTIFY(sc->sc_imess[0])) {
printf("%s: reselect without IDENTIFY; "
"sending DEVICE RESET\n", sc->sc_dev.dv_xname);
SPC_BREAK();
goto reset;
}
(void) spc_reselect(sc, sc->sc_imess[0]);
break;
default:
printf("%s: unexpected MESSAGE IN; sending DEVICE RESET\n",
sc->sc_dev.dv_xname);
SPC_BREAK();
reset:
spc_sched_msgout(sc, SEND_DEV_RESET);
break;
#ifdef notdef
abort:
spc_sched_msgout(sc, SEND_ABORT);
break;
#endif
}
#if 0
(void) bus_space_read_1(iot, ioh, DREG);
while ((bus_space_read_1(iot, ioh, PSNS) & ACKI) != 0)
;
#endif
goto nextmsg;
out:
bus_space_write_1(iot, ioh, SCMD, SCMD_RST_ACK);
SPC_MISC(("n=%d imess=0x%02x ", n, sc->sc_imess[0]));
}
void
spc_msgout(struct spc_softc *sc)
{
bus_space_tag_t iot = sc->sc_iot;
bus_space_handle_t ioh = sc->sc_ioh;
#if SPC_USE_SYNCHRONOUS
struct spc_tinfo *ti;
#endif
int n;
SPC_TRACE(("spc_msgout "));
if (sc->sc_prevphase == PH_MSGOUT) {
if (sc->sc_omp == sc->sc_omess) {
SPC_MISC(("retransmitting "));
sc->sc_msgpriq |= sc->sc_msgoutq;
bus_space_write_1(iot, ioh, SCMD,
SCMD_SET_ATN);
} else {
n = sc->sc_omp - sc->sc_omess;
goto nextbyte;
}
}
sc->sc_msgoutq = 0;
sc->sc_lastmsg = 0;
nextmsg:
sc->sc_currmsg = sc->sc_msgpriq & -sc->sc_msgpriq;
sc->sc_msgpriq &= ~sc->sc_currmsg;
sc->sc_msgoutq |= sc->sc_currmsg;
switch (sc->sc_currmsg) {
case SEND_IDENTIFY:
SPC_ASSERT(sc->sc_nexus != NULL);
sc->sc_omess[0] =
MSG_IDENTIFY(sc->sc_nexus->xs->sc_link->lun, 1);
n = 1;
break;
#if SPC_USE_SYNCHRONOUS
case SEND_SDTR:
SPC_ASSERT(sc->sc_nexus != NULL);
ti = &sc->sc_tinfo[sc->sc_nexus->xs->sc_link->target];
sc->sc_omess[4] = MSG_EXTENDED;
sc->sc_omess[3] = MSG_EXT_SDTR_LEN;
sc->sc_omess[2] = MSG_EXT_SDTR;
sc->sc_omess[1] = ti->period >> 2;
sc->sc_omess[0] = ti->offset;
n = 5;
break;
#endif
#if SPC_USE_WIDE
case SEND_WDTR:
SPC_ASSERT(sc->sc_nexus != NULL);
ti = &sc->sc_tinfo[sc->sc_nexus->xs->sc_link->target];
sc->sc_omess[3] = MSG_EXTENDED;
sc->sc_omess[2] = MSG_EXT_WDTR_LEN;
sc->sc_omess[1] = MSG_EXT_WDTR;
sc->sc_omess[0] = ti->width;
n = 4;
break;
#endif
case SEND_DEV_RESET:
sc->sc_flags |= SPC_ABORTING;
sc->sc_omess[0] = MSG_BUS_DEV_RESET;
n = 1;
break;
case SEND_REJECT:
sc->sc_omess[0] = MSG_MESSAGE_REJECT;
n = 1;
break;
case SEND_PARITY_ERROR:
sc->sc_omess[0] = MSG_PARITY_ERROR;
n = 1;
break;
case SEND_INIT_DET_ERR:
sc->sc_omess[0] = MSG_INITIATOR_DET_ERR;
n = 1;
break;
case SEND_ABORT:
sc->sc_flags |= SPC_ABORTING;
sc->sc_omess[0] = MSG_ABORT;
n = 1;
break;
default:
printf("%s: unexpected MESSAGE OUT; sending NOOP\n",
sc->sc_dev.dv_xname);
SPC_BREAK();
sc->sc_omess[0] = MSG_NOOP;
n = 1;
break;
}
sc->sc_omp = &sc->sc_omess[n];
nextbyte:
bus_space_write_1(iot, ioh, TCH, n >> 16);
bus_space_write_1(iot, ioh, TCM, n >> 8);
bus_space_write_1(iot, ioh, TCL, n);
bus_space_write_1(iot, ioh, PCTL, sc->sc_phase | PCTL_BFINT_ENAB);
#ifdef x68k
bus_space_write_1(iot, ioh, SCMD, SCMD_XFR);
#else
bus_space_write_1(iot, ioh, SCMD,
SCMD_XFR | SCMD_PROG_XFR | SCMD_ICPT_XFR);
#endif
for (;;) {
if ((bus_space_read_1(iot, ioh, SSTS) & SSTS_BUSY) != 0)
break;
if (bus_space_read_1(iot, ioh, INTS) != 0)
goto out;
}
for (;;) {
#if 0
for (;;) {
if ((bus_space_read_1(iot, ioh, PSNS) & PSNS_REQ) != 0)
break;
}
#endif
if (bus_space_read_1(iot, ioh, INTS) != 0) {
#if 0
if (sc->sc_msgpriq == 0)
bus_space_write_1(iot, ioh, SCMD, SCMD_RST_ATN);
#endif
goto out;
}
#if 0
if (n == 1 && sc->sc_msgpriq == 0)
bus_space_write_1(iot, ioh, SCMD, SCMD_RST_ATN);
#endif
while ((bus_space_read_1(iot, ioh, SSTS) & SSTS_DREG_FULL) != 0)
;
bus_space_write_1(iot, ioh, DREG, *--sc->sc_omp);
--n;
sc->sc_lastmsg = sc->sc_currmsg;
#if 0
while ((bus_space_read_1(iot, ioh, PSNS) & ACKI) != 0)
;
#endif
if (n == 0)
break;
}
if (sc->sc_msgpriq != 0) {
goto nextmsg;
}
out:
return;
}
int
spc_dataout_pio(struct spc_softc *sc, u_char *p, int n)
{
bus_space_tag_t iot = sc->sc_iot;
bus_space_handle_t ioh = sc->sc_ioh;
u_char intstat = 0;
int out = 0;
#define DOUTAMOUNT 8
SPC_TRACE(("spc_dataout_pio "));
bus_space_write_1(iot, ioh, TCH, n >> 16);
bus_space_write_1(iot, ioh, TCM, n >> 8);
bus_space_write_1(iot, ioh, TCL, n);
bus_space_write_1(iot, ioh, PCTL, sc->sc_phase | PCTL_BFINT_ENAB);
#ifdef x68k
bus_space_write_1(iot, ioh, SCMD, SCMD_XFR);
#else
bus_space_write_1(iot, ioh, SCMD,
SCMD_XFR | SCMD_PROG_XFR | SCMD_ICPT_XFR);
#endif
for (;;) {
if ((bus_space_read_1(iot, ioh, SSTS) & SSTS_BUSY) != 0)
break;
if (bus_space_read_1(iot, ioh, INTS) != 0)
break;
}
while (n > 0) {
int xfer;
for (;;) {
intstat = bus_space_read_1(iot, ioh, INTS);
if ((bus_space_read_1(iot, ioh, SSTS) &
SSTS_DREG_EMPTY) != 0)
break;
if (intstat != 0)
goto phasechange;
}
xfer = min(DOUTAMOUNT, n);
SPC_MISC(("%d> ", xfer));
n -= xfer;
out += xfer;
#if 0
bus_space_write_multi_1(iot, ioh, DREG, p, xfer);
p += xfer;
#else
switch (xfer) {
case 8:
bus_space_write_1(iot, ioh, DREG, *p++);
case 7:
bus_space_write_1(iot, ioh, DREG, *p++);
case 6:
bus_space_write_1(iot, ioh, DREG, *p++);
case 5:
bus_space_write_1(iot, ioh, DREG, *p++);
case 4:
bus_space_write_1(iot, ioh, DREG, *p++);
case 3:
bus_space_write_1(iot, ioh, DREG, *p++);
case 2:
bus_space_write_1(iot, ioh, DREG, *p++);
case 1:
bus_space_write_1(iot, ioh, DREG, *p++);
}
#endif
}
if (out == 0) {
for (;;) {
if (bus_space_read_1(iot, ioh, INTS) != 0)
break;
}
SPC_MISC(("extra data "));
} else {
for (;;) {
if ((bus_space_read_1(iot, ioh, SSTS) &
SSTS_DREG_EMPTY) != 0)
break;
intstat = bus_space_read_1(iot, ioh, INTS);
if (intstat != 0)
goto phasechange;
}
}
phasechange:
if (intstat != 0) {
int amount;
amount = ((bus_space_read_1(iot, ioh, TCH) << 16) |
(bus_space_read_1(iot, ioh, TCM) << 8) |
bus_space_read_1(iot, ioh, TCL));
if (amount > 0) {
out -= amount;
SPC_MISC(("+%d ", amount));
}
}
return out;
}
int
spc_datain_pio(struct spc_softc *sc, u_char *p, int n)
{
bus_space_tag_t iot = sc->sc_iot;
bus_space_handle_t ioh = sc->sc_ioh;
u_int8_t intstat, sstat;
int in = 0;
#define DINAMOUNT 8
SPC_TRACE(("spc_datain_pio "));
bus_space_write_1(iot, ioh, TCH, n >> 16);
bus_space_write_1(iot, ioh, TCM, n >> 8);
bus_space_write_1(iot, ioh, TCL, n);
bus_space_write_1(iot, ioh, PCTL, sc->sc_phase | PCTL_BFINT_ENAB);
#ifdef x68k
bus_space_write_1(iot, ioh, SCMD, SCMD_XFR);
#else
bus_space_write_1(iot, ioh, SCMD,
SCMD_XFR | SCMD_PROG_XFR);
#endif
for (;;) {
if ((bus_space_read_1(iot, ioh, SSTS) & SSTS_BUSY) != 0)
break;
if (bus_space_read_1(iot, ioh, INTS) != 0)
goto phasechange;
}
while (n > 0) {
int xfer;
for (;;) {
intstat = bus_space_read_1(iot, ioh, INTS);
sstat = bus_space_read_1(iot, ioh, SSTS);
if (intstat != 0 ||
(sstat & SSTS_DREG_FULL) != 0 ||
(sstat & SSTS_DREG_EMPTY) == 0)
break;
}
#if 1
if (intstat != 0)
goto phasechange;
#else
if (intstat != 0 &&
(sstat & SSTS_DREG_EMPTY) != 0)
goto phasechange;
#endif
if ((sstat & SSTS_DREG_FULL) != 0)
xfer = min(DINAMOUNT, n);
else
xfer = 1;
SPC_MISC((">%d ", xfer));
n -= xfer;
in += xfer;
#if 0
bus_space_read_multi_1(iot, ioh, DREG, p, xfer);
p += xfer;
#else
switch (xfer) {
case 8:
*p++ = bus_space_read_1(iot, ioh, DREG);
case 7:
*p++ = bus_space_read_1(iot, ioh, DREG);
case 6:
*p++ = bus_space_read_1(iot, ioh, DREG);
case 5:
*p++ = bus_space_read_1(iot, ioh, DREG);
case 4:
*p++ = bus_space_read_1(iot, ioh, DREG);
case 3:
*p++ = bus_space_read_1(iot, ioh, DREG);
case 2:
*p++ = bus_space_read_1(iot, ioh, DREG);
case 1:
*p++ = bus_space_read_1(iot, ioh, DREG);
}
#endif
if (intstat != 0)
goto phasechange;
}
if (in == 0) {
for (;;) {
if (bus_space_read_1(iot, ioh, INTS) != 0)
break;
}
SPC_MISC(("extra data "));
}
phasechange:
return in;
}
int
spc_intr(void *arg)
{
struct spc_softc *sc = arg;
bus_space_tag_t iot = sc->sc_iot;
bus_space_handle_t ioh = sc->sc_ioh;
u_char ints;
struct spc_acb *acb;
struct scsi_link *sc_link;
struct spc_tinfo *ti;
int n;
ints = bus_space_read_1(iot, ioh, INTS);
if (ints == 0)
return 0;
bus_space_write_1(iot, ioh, SCTL,
bus_space_read_1(iot, ioh, SCTL) & ~SCTL_INTR_ENAB);
SPC_TRACE(("spc_intr "));
loop:
#ifdef x68k
while ((ints = bus_space_read_1(iot, ioh, INTS)) == 0)
delay(1);
SPC_MISC(("ints = 0x%x ", ints));
#else
ints = bus_space_read_1(iot, ioh, INTS);
SPC_MISC(("ints = 0x%x ", ints));
#endif
if ((ints & INTS_RST) != 0) {
printf("%s: SCSI bus reset\n", sc->sc_dev.dv_xname);
goto reset;
}
if ((bus_space_read_1(iot, ioh, SERR) & (SERR_SCSI_PAR|SERR_SPC_PAR))
!= 0) {
printf("%s: SCSI bus parity error\n", sc->sc_dev.dv_xname);
if (sc->sc_prevphase == PH_MSGIN) {
sc->sc_flags |= SPC_DROP_MSGIN;
spc_sched_msgout(sc, SEND_PARITY_ERROR);
} else
spc_sched_msgout(sc, SEND_INIT_DET_ERR);
}
switch (sc->sc_state) {
case SPC_IDLE:
case SPC_SELECTING:
SPC_MISC(("ints:0x%02x ", ints));
if ((ints & INTS_SEL) != 0) {
printf("%s: target mode selected; going to BUS FREE\n",
sc->sc_dev.dv_xname);
goto sched;
} else if ((ints & INTS_RESEL) != 0) {
SPC_MISC(("reselected "));
if (sc->sc_state == SPC_SELECTING) {
SPC_MISC(("backoff selector "));
SPC_ASSERT(sc->sc_nexus != NULL);
acb = sc->sc_nexus;
sc->sc_nexus = NULL;
TAILQ_INSERT_HEAD(&sc->ready_list, acb, chain);
}
sc->sc_selid = bus_space_read_1(iot, ioh, TEMP);
sc->sc_state = SPC_RESELECTED;
} else if ((ints & INTS_CMD_DONE) != 0) {
SPC_MISC(("selected "));
if (sc->sc_state != SPC_SELECTING) {
printf("%s: selection out while idle; "
"resetting\n", sc->sc_dev.dv_xname);
SPC_BREAK();
goto reset;
}
SPC_ASSERT(sc->sc_nexus != NULL);
acb = sc->sc_nexus;
sc_link = acb->xs->sc_link;
ti = &sc->sc_tinfo[sc_link->target];
sc->sc_msgpriq = SEND_IDENTIFY;
if (acb->flags & ACB_RESET)
sc->sc_msgpriq |= SEND_DEV_RESET;
else if (acb->flags & ACB_ABORT)
sc->sc_msgpriq |= SEND_ABORT;
else {
#if SPC_USE_SYNCHRONOUS
if ((ti->flags & DO_SYNC) != 0)
sc->sc_msgpriq |= SEND_SDTR;
#endif
#if SPC_USE_WIDE
if ((ti->flags & DO_WIDE) != 0)
sc->sc_msgpriq |= SEND_WDTR;
#endif
}
acb->flags |= ACB_NEXUS;
ti->lubusy |= (1 << sc_link->lun);
sc->sc_dp = acb->data_addr;
sc->sc_dleft = acb->data_length;
sc->sc_cp = (u_char *)&acb->scsi_cmd;
sc->sc_cleft = acb->scsi_cmd_length;
if ((acb->xs->flags & SCSI_POLL) == 0)
timeout_add_msec(&acb->xs->stimeout,
acb->xs->timeout);
sc->sc_state = SPC_CONNECTED;
} else if ((ints & INTS_TIMEOUT) != 0) {
SPC_MISC(("selection timeout "));
if (sc->sc_state != SPC_SELECTING) {
printf("%s: selection timeout while idle; "
"resetting\n", sc->sc_dev.dv_xname);
SPC_BREAK();
goto reset;
}
SPC_ASSERT(sc->sc_nexus != NULL);
acb = sc->sc_nexus;
delay(250);
acb->xs->error = XS_SELTIMEOUT;
goto finish;
} else {
if (sc->sc_state != SPC_IDLE) {
printf("%s: BUS FREE while not idle; "
"state=%d\n",
sc->sc_dev.dv_xname, sc->sc_state);
SPC_BREAK();
goto out;
}
goto sched;
}
sc->sc_flags = 0;
sc->sc_prevphase = PH_INVALID;
goto dophase;
}
if ((ints & INTS_DISCON) != 0) {
bus_space_write_1(iot, ioh, PCTL,
bus_space_read_1(iot, ioh, PCTL) & ~PCTL_BFINT_ENAB);
bus_space_write_1(iot, ioh, INTS, ints);
switch (sc->sc_state) {
case SPC_RESELECTED:
goto sched;
case SPC_CONNECTED:
SPC_ASSERT(sc->sc_nexus != NULL);
acb = sc->sc_nexus;
#if SPC_USE_SYNCHRONOUS + SPC_USE_WIDE
if (sc->sc_prevphase == PH_MSGOUT) {
sc_link = acb->xs->sc_link;
ti = &sc->sc_tinfo[sc_link->target];
switch (sc->sc_lastmsg) {
#if SPC_USE_SYNCHRONOUS
case SEND_SDTR:
ti->flags &= ~DO_SYNC;
ti->period = ti->offset = 0;
break;
#endif
#if SPC_USE_WIDE
case SEND_WDTR:
ti->flags &= ~DO_WIDE;
ti->width = 0;
break;
#endif
}
}
#endif
if ((sc->sc_flags & SPC_ABORTING) == 0) {
printf("%s: unexpected disconnect; "
"sending REQUEST SENSE\n",
sc->sc_dev.dv_xname);
SPC_BREAK();
spc_sense(sc, acb);
goto out;
}
acb->xs->error = XS_DRIVER_STUFFUP;
goto finish;
case SPC_DISCONNECT:
SPC_ASSERT(sc->sc_nexus != NULL);
acb = sc->sc_nexus;
TAILQ_INSERT_HEAD(&sc->nexus_list, acb, chain);
sc->sc_nexus = NULL;
goto sched;
case SPC_CMDCOMPLETE:
SPC_ASSERT(sc->sc_nexus != NULL);
acb = sc->sc_nexus;
goto finish;
}
}
else if ((ints & INTS_CMD_DONE) != 0 &&
sc->sc_prevphase == PH_MSGIN && sc->sc_state != SPC_CONNECTED)
goto out;
dophase:
#if 0
if ((bus_space_read_1(iot, ioh, PSNS) & PSNS_REQ) == 0) {
goto out;
}
#else
bus_space_write_1(iot, ioh, INTS, ints);
ints = 0;
while ((bus_space_read_1(iot, ioh, PSNS) & PSNS_REQ) == 0)
breathe();
#endif
sc->sc_phase = bus_space_read_1(iot, ioh, PSNS) & PH_MASK;
#if 0
bus_space_write_1(iot, ioh, PCTL, sc->sc_phase);
#endif
SPC_MISC(("phase=%d\n", sc->sc_phase));
switch (sc->sc_phase) {
case PH_MSGOUT:
if (sc->sc_state != SPC_CONNECTED &&
sc->sc_state != SPC_RESELECTED)
break;
spc_msgout(sc);
sc->sc_prevphase = PH_MSGOUT;
goto loop;
case PH_MSGIN:
if (sc->sc_state != SPC_CONNECTED &&
sc->sc_state != SPC_RESELECTED)
break;
spc_msgin(sc);
sc->sc_prevphase = PH_MSGIN;
goto loop;
case PH_CMD:
if (sc->sc_state != SPC_CONNECTED)
break;
#ifdef SPC_DEBUG
if ((spc_debug & SPC_SHOWMISC) != 0) {
SPC_ASSERT(sc->sc_nexus != NULL);
acb = sc->sc_nexus;
printf("cmd=0x%02x+%d ",
acb->scsi_cmd.opcode, acb->scsi_cmd_length - 1);
}
#endif
n = spc_dataout_pio(sc, sc->sc_cp, sc->sc_cleft);
sc->sc_cp += n;
sc->sc_cleft -= n;
sc->sc_prevphase = PH_CMD;
goto loop;
case PH_DATAOUT:
if (sc->sc_state != SPC_CONNECTED)
break;
SPC_MISC(("dataout dleft=%d ", sc->sc_dleft));
n = spc_dataout_pio(sc, sc->sc_dp, sc->sc_dleft);
sc->sc_dp += n;
sc->sc_dleft -= n;
sc->sc_prevphase = PH_DATAOUT;
goto loop;
case PH_DATAIN:
if (sc->sc_state != SPC_CONNECTED)
break;
SPC_MISC(("datain "));
n = spc_datain_pio(sc, sc->sc_dp, sc->sc_dleft);
sc->sc_dp += n;
sc->sc_dleft -= n;
sc->sc_prevphase = PH_DATAIN;
goto loop;
case PH_STAT:
if (sc->sc_state != SPC_CONNECTED)
break;
SPC_ASSERT(sc->sc_nexus != NULL);
acb = sc->sc_nexus;
spc_datain_pio(sc, &acb->target_stat, 1);
SPC_MISC(("target_stat=0x%02x ", acb->target_stat));
sc->sc_prevphase = PH_STAT;
goto loop;
}
printf("%s: unexpected bus phase; resetting\n", sc->sc_dev.dv_xname);
SPC_BREAK();
reset:
spc_init(sc);
return 1;
finish:
timeout_del(&acb->xs->stimeout);
bus_space_write_1(iot, ioh, INTS, ints);
ints = 0;
spc_done(sc, acb);
goto out;
sched:
sc->sc_state = SPC_IDLE;
spc_sched(sc);
goto out;
out:
if (ints)
bus_space_write_1(iot, ioh, INTS, ints);
bus_space_write_1(iot, ioh, SCTL,
bus_space_read_1(iot, ioh, SCTL) | SCTL_INTR_ENAB);
return 1;
}
void
spc_abort(struct spc_softc *sc, struct spc_acb *acb)
{
acb->xs->timeout = SPC_ABORT_TIMEOUT;
acb->flags |= ACB_ABORT;
if (acb == sc->sc_nexus) {
if (sc->sc_state == SPC_CONNECTED)
spc_sched_msgout(sc, SEND_ABORT);
} else {
spc_dequeue(sc, acb);
TAILQ_INSERT_HEAD(&sc->ready_list, acb, chain);
if (sc->sc_state == SPC_IDLE)
spc_sched(sc);
}
}
void
spc_timeout(void *arg)
{
struct spc_acb *acb = arg;
struct scsi_xfer *xs = acb->xs;
struct scsi_link *sc_link = xs->sc_link;
struct spc_softc *sc = sc_link->bus->sb_adapter_softc;
int s;
sc_print_addr(sc_link);
printf("timed out");
s = splbio();
if (acb->flags & ACB_ABORT) {
printf(" AGAIN\n");
} else {
printf("\n");
acb->xs->error = XS_TIMEOUT;
spc_abort(sc, acb);
}
splx(s);
}
#ifdef SPC_DEBUG
void
spc_show_scsi_cmd(spc_acb *acb)
{
u_char *b = (u_char *)&acb->scsi_cmd;
struct scsi_link *sc_link = acb->xs->sc_link;
int i;
sc_print_addr(sc_link);
if ((acb->xs->flags & SCSI_RESET) == 0) {
for (i = 0; i < acb->scsi_cmd_length; i++) {
if (i)
printf(",");
printf("%x", b[i]);
}
printf("\n");
} else
printf("RESET\n");
}
void
spc_print_acb(spc_acb *acb)
{
printf("acb@%p xs=%p flags=%x", acb, acb->xs, acb->flags);
printf(" dp=%p dleft=%d target_stat=%x\n",
acb->data_addr, acb->data_length, acb->target_stat);
spc_show_scsi_cmd(acb);
}
void
spc_print_active_acb()
{
struct spc_acb *acb;
struct spc_softc *sc = spc_cd.cd_devs[0];
printf("ready list:\n");
TAILQ_FOREACH(acb, &sc->ready_list, chain)
spc_print_acb(acb);
printf("nexus:\n");
if (sc->sc_nexus != NULL)
spc_print_acb(sc->sc_nexus);
printf("nexus list:\n");
TAILQ_FOREACH(acb, &sc->nexus_list, chain)
spc_print_acb(acb);
}
void
spc_dump89352(struct spc_softc *sc)
{
bus_space_tag_t iot = sc->sc_iot;
bus_space_handle_t ioh = sc->sc_ioh;
printf("mb89352: BDID=%x SCTL=%x SCMD=%x TMOD=%x\n",
bus_space_read_1(iot, ioh, BDID),
bus_space_read_1(iot, ioh, SCTL),
bus_space_read_1(iot, ioh, SCMD),
bus_space_read_1(iot, ioh, TMOD));
printf(" INTS=%x PSNS=%x SSTS=%x SERR=%x PCTL=%x\n",
bus_space_read_1(iot, ioh, INTS),
bus_space_read_1(iot, ioh, PSNS),
bus_space_read_1(iot, ioh, SSTS),
bus_space_read_1(iot, ioh, SERR),
bus_space_read_1(iot, ioh, PCTL));
printf(" MBC=%x DREG=%x TEMP=%x TCH=%x TCM=%x\n",
bus_space_read_1(iot, ioh, MBC),
#if 0
bus_space_read_1(iot, ioh, DREG),
#else
0,
#endif
bus_space_read_1(iot, ioh, TEMP),
bus_space_read_1(iot, ioh, TCH),
bus_space_read_1(iot, ioh, TCM));
printf(" TCL=%x EXBF=%x\n",
bus_space_read_1(iot, ioh, TCL),
bus_space_read_1(iot, ioh, EXBF));
}
void
spc_dump_driver(struct spc_softc *sc)
{
struct spc_tinfo *ti;
int i;
printf("nexus=%p prevphase=%x\n", sc->sc_nexus, sc->sc_prevphase);
printf("state=%x msgin=%x msgpriq=%x msgoutq=%x lastmsg=%x "
"currmsg=%x\n", sc->sc_state, sc->sc_imess[0],
sc->sc_msgpriq, sc->sc_msgoutq, sc->sc_lastmsg, sc->sc_currmsg);
for (i = 0; i < 7; i++) {
ti = &sc->sc_tinfo[i];
printf("tinfo%d: %d cmds %d disconnects %d timeouts",
i, ti->cmds, ti->dconns, ti->touts);
printf(" %d senses flags=%x\n", ti->senses, ti->flags);
}
}
#endif