summaryrefslogtreecommitdiffstats
path: root/sys/dev/isp/isp.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/isp/isp.c')
-rw-r--r--sys/dev/isp/isp.c1841
1 files changed, 1841 insertions, 0 deletions
diff --git a/sys/dev/isp/isp.c b/sys/dev/isp/isp.c
new file mode 100644
index 0000000..f50bfcc
--- /dev/null
+++ b/sys/dev/isp/isp.c
@@ -0,0 +1,1841 @@
+/* $Id: isp.c,v 1.6 1998/04/14 17:43:45 mjacob Exp $ */
+/*
+ * Machine Independent (well, as best as possible)
+ * code for the Qlogic ISP SCSI adapters.
+ *
+ *---------------------------------------
+ * Copyright (c) 1997, 1998 by Matthew Jacob
+ * NASA/Ames Research Center
+ * All rights reserved.
+ *---------------------------------------
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice immediately at the beginning of the file, without modification,
+ * this list of conditions, and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. The name of the author may not be used to endorse or promote products
+ * derived from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR
+ * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ */
+
+/*
+ * Inspiration and ideas about this driver are from Erik Moe's Linux driver
+ * (qlogicisp.c) and Dave Miller's SBus version of same (qlogicisp.c). Some
+ * ideas dredged from the Solaris driver.
+ */
+
+/*
+ * Include header file appropriate for platform we're building on.
+ */
+
+#ifdef __NetBSD__
+#include <dev/ic/isp_netbsd.h>
+#endif
+#ifdef __FreeBSD__
+#include <dev/isp/isp_freebsd.h>
+#endif
+#ifdef __linux__
+#include <isp_linux.h>
+#endif
+
+/*
+ * General defines
+ */
+
+#define MBOX_DELAY_COUNT 1000000 / 100
+
+/*
+ * Function prototypes.
+ */
+static int isp_poll __P((struct ispsoftc *, ISP_SCSI_XFER_T *, int));
+static void isp_parse_status
+__P((struct ispsoftc *, ispstatusreq_t *, ISP_SCSI_XFER_T *));
+static void isp_lostcmd
+__P((struct ispsoftc *, ISP_SCSI_XFER_T *, ispreq_t *));
+static void isp_fibre_init __P((struct ispsoftc *));
+static void isp_fw_state __P((struct ispsoftc *));
+static void isp_dumpregs __P((struct ispsoftc *, const char *));
+static void isp_setdparm __P((struct ispsoftc *));
+static void isp_prtstst __P((ispstatusreq_t *));
+static void isp_mboxcmd __P((struct ispsoftc *, mbreg_t *));
+
+/*
+ * Reset Hardware.
+ */
+void
+isp_reset(isp)
+ struct ispsoftc *isp;
+{
+ mbreg_t mbs;
+ int loops, i, dodnld = 1;
+ char *revname;
+ ISP_LOCKVAL_DECL;
+
+ revname = "(unknown)";
+
+ isp->isp_state = ISP_NILSTATE;
+
+ /*
+ * Basic types have been set in the MD code.
+ * See if we can't figure out more here.
+ */
+ isp->isp_dblev = DFLT_DBLEVEL;
+ if (isp->isp_type & ISP_HA_FC) {
+ revname = "2100";
+ } else {
+ i = ISP_READ(isp, BIU_CONF0) & BIU_CONF0_HW_MASK;
+ switch (i) {
+ default:
+ PRINTF("%s: unknown ISP type %x- assuming 1020\n",
+ isp->isp_name, i);
+ isp->isp_type = ISP_HA_SCSI_1020;
+ break;
+ case 1:
+ case 2:
+ revname = "1020";
+ isp->isp_type = ISP_HA_SCSI_1020;
+ break;
+ case 3:
+ revname = "1040A";
+ isp->isp_type = ISP_HA_SCSI_1040A;
+ break;
+ case 5:
+ revname = "1040B";
+ isp->isp_type = ISP_HA_SCSI_1040B;
+ break;
+ }
+ }
+
+ /*
+ * Do MD specific pre initialization
+ */
+ ISP_LOCK;
+ ISP_RESET0(isp);
+ isp_setdparm(isp);
+
+ /*
+ * Hit the chip over the head with hammer,
+ * and give the ISP a chance to recover.
+ */
+
+ if (isp->isp_type & ISP_HA_SCSI) {
+ ISP_WRITE(isp, BIU_ICR, BIU_ICR_SOFT_RESET);
+ /*
+ * A slight delay...
+ */
+ SYS_DELAY(100);
+
+ /*
+ * Clear data && control DMA engines.
+ */
+ ISP_WRITE(isp, CDMA_CONTROL,
+ DMA_CNTRL_CLEAR_CHAN | DMA_CNTRL_RESET_INT);
+ ISP_WRITE(isp, DDMA_CONTROL,
+ DMA_CNTRL_CLEAR_CHAN | DMA_CNTRL_RESET_INT);
+ } else {
+ ISP_WRITE(isp, BIU2100_CSR, BIU2100_SOFT_RESET);
+ /*
+ * A slight delay...
+ */
+ SYS_DELAY(100);
+ ISP_WRITE(isp, CDMA2100_CONTROL,
+ DMA_CNTRL2100_CLEAR_CHAN | DMA_CNTRL2100_RESET_INT);
+ ISP_WRITE(isp, TDMA2100_CONTROL,
+ DMA_CNTRL2100_CLEAR_CHAN | DMA_CNTRL2100_RESET_INT);
+ ISP_WRITE(isp, RDMA2100_CONTROL,
+ DMA_CNTRL2100_CLEAR_CHAN | DMA_CNTRL2100_RESET_INT);
+ }
+
+ /*
+ * Wait for ISP to be ready to go...
+ */
+ loops = MBOX_DELAY_COUNT;
+ for (;;) {
+ if (isp->isp_type & ISP_HA_SCSI) {
+ if (!(ISP_READ(isp, BIU_ICR) & BIU_ICR_SOFT_RESET))
+ break;
+ } else {
+ if (!(ISP_READ(isp, BIU2100_CSR) & BIU2100_SOFT_RESET))
+ break;
+ }
+ SYS_DELAY(100);
+ if (--loops < 0) {
+ isp_dumpregs(isp, "chip reset timed out");
+ ISP_UNLOCK;
+ return;
+ }
+ }
+ /*
+ * More initialization
+ */
+ if (isp->isp_type & ISP_HA_SCSI) {
+ ISP_WRITE(isp, BIU_CONF1, 0);
+ } else {
+ ISP_WRITE(isp, BIU2100_CSR, 0);
+ ISP_WRITE(isp, RISC_MTR2100, 0x1212); /* FM */
+ }
+
+ ISP_WRITE(isp, HCCR, HCCR_CMD_RESET);
+ SYS_DELAY(100);
+
+ if (isp->isp_type & ISP_HA_SCSI) {
+ ISP_SETBITS(isp, BIU_CONF1, isp->isp_mdvec->dv_conf1);
+ if (isp->isp_mdvec->dv_conf1 & BIU_BURST_ENABLE) {
+ ISP_SETBITS(isp, CDMA_CONF, DMA_ENABLE_BURST);
+ ISP_SETBITS(isp, DDMA_CONF, DMA_ENABLE_BURST);
+ }
+ }
+ ISP_WRITE(isp, HCCR, HCCR_CMD_RELEASE); /* release paused processor */
+
+ /*
+ * Do MD specific post initialization
+ */
+ ISP_RESET1(isp);
+
+ /*
+ * Enable interrupts
+ */
+ ENABLE_INTS(isp);
+
+ /*
+ * Do some sanity checking.
+ */
+ mbs.param[0] = MBOX_NO_OP;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ isp_dumpregs(isp, "NOP test failed");
+ ISP_UNLOCK;
+ return;
+ }
+
+ if (isp->isp_type & ISP_HA_SCSI) {
+ mbs.param[0] = MBOX_MAILBOX_REG_TEST;
+ mbs.param[1] = 0xdead;
+ mbs.param[2] = 0xbeef;
+ mbs.param[3] = 0xffff;
+ mbs.param[4] = 0x1111;
+ mbs.param[5] = 0xa5a5;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ isp_dumpregs(isp,
+ "Mailbox Register test didn't complete");
+ ISP_UNLOCK;
+ return;
+ }
+ if (mbs.param[1] != 0xdead || mbs.param[2] != 0xbeef ||
+ mbs.param[3] != 0xffff || mbs.param[4] != 0x1111 ||
+ mbs.param[5] != 0xa5a5) {
+ isp_dumpregs(isp, "Register Test Failed");
+ ISP_UNLOCK;
+ return;
+ }
+
+ }
+
+ /*
+ * Download new Firmware, unless requested not to do so.
+ * This is made slightly trickier in some cases where the
+ * firmware of the ROM revision is newer than the revision
+ * compiled into the driver. So, where we used to compare
+ * versions of our f/w and the ROM f/w, now we just see
+ * whether we have f/w at all and whether a config flag
+ * has disabled our download.
+ */
+ if ((isp->isp_mdvec->dv_fwlen == 0) ||
+ (isp->isp_confopts & ISP_CFG_NORELOAD)) {
+ dodnld = 0;
+ }
+
+ if (dodnld) {
+ for (i = 0; i < isp->isp_mdvec->dv_fwlen; i++) {
+ mbs.param[0] = MBOX_WRITE_RAM_WORD;
+ mbs.param[1] = isp->isp_mdvec->dv_codeorg + i;
+ mbs.param[2] = isp->isp_mdvec->dv_ispfw[i];
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ isp_dumpregs(isp, "f/w download failed");
+
+ ISP_UNLOCK;
+ return;
+ }
+ }
+
+ if (isp->isp_mdvec->dv_fwlen) {
+ /*
+ * Verify that it downloaded correctly.
+ */
+ mbs.param[0] = MBOX_VERIFY_CHECKSUM;
+ mbs.param[1] = isp->isp_mdvec->dv_codeorg;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ isp_dumpregs(isp, "ram checksum failure");
+ ISP_UNLOCK;
+ return;
+ }
+ }
+ } else {
+ IDPRINTF(3, ("%s: skipping f/w download\n", isp->isp_name));
+ }
+
+ /*
+ * Now start it rolling.
+ *
+ * If we didn't actually download f/w,
+ * we still need to (re)start it.
+ */
+
+ mbs.param[0] = MBOX_EXEC_FIRMWARE;
+ mbs.param[1] = isp->isp_mdvec->dv_codeorg;
+ isp_mboxcmd(isp, &mbs);
+
+ if (isp->isp_type & ISP_HA_SCSI) {
+ /*
+ * Set CLOCK RATE
+ */
+ if (((sdparam *)isp->isp_param)->isp_clock) {
+ mbs.param[0] = MBOX_SET_CLOCK_RATE;
+ mbs.param[1] = ((sdparam *)isp->isp_param)->isp_clock;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ isp_dumpregs(isp, "failed to set CLOCKRATE");
+ ISP_UNLOCK;
+ return;
+ }
+ }
+ }
+ mbs.param[0] = MBOX_ABOUT_FIRMWARE;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ isp_dumpregs(isp, "ABOUT FIRMWARE command failed");
+ ISP_UNLOCK;
+ return;
+ }
+ PRINTF("%s: Board Revision %s, %s F/W Revision %d.%d\n",
+ isp->isp_name, revname, dodnld? "loaded" : "ROM",
+ mbs.param[1], mbs.param[2]);
+ isp_fw_state(isp);
+ isp->isp_state = ISP_RESETSTATE;
+ ISP_UNLOCK;
+}
+
+/*
+ * Abort an executing command.
+ * Locks (ints blocked) assumed held.
+ */
+
+int
+isp_abortcmd(isp, xidx)
+ struct ispsoftc *isp;
+ int xidx;
+{
+ mbreg_t mbs;
+ ISP_SCSI_XFER_T *xs;
+
+ xs = (ISP_SCSI_XFER_T *) isp->isp_xflist[xidx];
+ if (xs == NULL) {
+ PRINTF("%s: isp_abortcmd - NULL xs\n", isp->isp_name);
+ return (0);
+ }
+ mbs.param[0] = MBOX_ABORT;
+ mbs.param[1] = XS_TGT(xs) | XS_LUN(xs);
+ mbs.param[2] = (xidx+1) >> 16;
+ mbs.param[3] = (xidx+1) & 0xffff;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ PRINTF("%s: isp_abort failure (code %x)\n",
+ isp->isp_name, mbs.param[0]);
+ return (0);
+ } else {
+ return (1);
+ }
+}
+
+/*
+ * Initialize Hardware to known state
+ */
+void
+isp_init(isp)
+ struct ispsoftc *isp;
+{
+ sdparam *sdp;
+ mbreg_t mbs;
+ int i, l;
+ ISP_LOCKVAL_DECL;
+
+ if (isp->isp_type & ISP_HA_FC) {
+ isp_fibre_init(isp);
+ return;
+ }
+
+ sdp = isp->isp_param;
+
+ /*
+ * Try and figure out if we're connected to a differential bus.
+ * You have to pause the RISC processor to read SXP registers.
+ *
+ * This, by the way, is likely broken in that it should be
+ * getting this info from NVRAM settings too.
+ */
+ ISP_LOCK;
+ ISP_WRITE(isp, HCCR, HCCR_CMD_PAUSE);
+ if (ISP_READ(isp, SXP_PINS_DIFF) & SXP_PINS_DIFF_SENSE) {
+ sdp->isp_diffmode = 1;
+ PRINTF("%s: Differential Mode\n", isp->isp_name);
+ } else {
+ /*
+ * Force pullups on.
+ */
+ sdp->isp_req_ack_active_neg = 1;
+ sdp->isp_data_line_active_neg = 1;
+ sdp->isp_diffmode = 0;
+ }
+ ISP_WRITE(isp, HCCR, HCCR_CMD_RELEASE); /* release paused processor */
+
+ mbs.param[0] = MBOX_GET_INIT_SCSI_ID;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ ISP_UNLOCK;
+ isp_dumpregs(isp, "failed to get initiator id");
+ return;
+ }
+ if (mbs.param[1] != sdp->isp_initiator_id) {
+ PRINTF("%s: setting Initiator ID to %d\n", isp->isp_name,
+ sdp->isp_initiator_id);
+ mbs.param[0] = MBOX_SET_INIT_SCSI_ID;
+ mbs.param[1] = sdp->isp_initiator_id;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ ISP_UNLOCK;
+ isp_dumpregs(isp, "failed to set initiator id");
+ return;
+ }
+ } else {
+ IDPRINTF(3, ("%s: leaving Initiator ID at %d\n", isp->isp_name,
+ sdp->isp_initiator_id));
+ }
+
+ mbs.param[0] = MBOX_SET_RETRY_COUNT;
+ mbs.param[1] = sdp->isp_retry_count;
+ mbs.param[2] = sdp->isp_retry_delay;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ ISP_UNLOCK;
+ isp_dumpregs(isp, "failed to set retry count and delay");
+ return;
+ }
+
+ mbs.param[0] = MBOX_SET_ASYNC_DATA_SETUP_TIME;
+ mbs.param[1] = sdp->isp_async_data_setup;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ ISP_UNLOCK;
+ isp_dumpregs(isp, "failed to set async data setup time");
+ return;
+ }
+
+ mbs.param[0] = MBOX_SET_ACTIVE_NEG_STATE;
+ mbs.param[1] = (sdp->isp_req_ack_active_neg << 4) |
+ (sdp->isp_data_line_active_neg << 5);
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ ISP_UNLOCK;
+ isp_dumpregs(isp, "failed to set active neg state");
+ return;
+ }
+
+ mbs.param[0] = MBOX_SET_TAG_AGE_LIMIT;
+ mbs.param[1] = sdp->isp_tag_aging;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ ISP_UNLOCK;
+ isp_dumpregs(isp, "failed to set tag age limit");
+ return;
+ }
+
+ mbs.param[0] = MBOX_SET_SELECT_TIMEOUT;
+ mbs.param[1] = sdp->isp_selection_timeout;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ ISP_UNLOCK;
+ isp_dumpregs(isp, "failed to set selection timeout");
+ return;
+ }
+
+ IDPRINTF(2, ("%s: devparm, W=wide, S=sync, T=Tag\n", isp->isp_name));
+ for (i = 0; i < MAX_TARGETS; i++) {
+ char bz[9];
+ u_int16_t cj = sdp->isp_devparam[i].sync_period;
+
+ if (sdp->isp_devparam[i].dev_flags & DPARM_SYNC) {
+ u_int16_t x;
+ if (cj == (ISP_20M_SYNCPARMS & 0xff)) {
+ x = 20;
+ } else if (cj == (ISP_10M_SYNCPARMS & 0xff)) {
+ x = 10;
+ } else if (cj == (ISP_08M_SYNCPARMS & 0xff)) {
+ x = 8;
+ } else if (cj == (ISP_05M_SYNCPARMS & 0xff)) {
+ x = 5;
+ } else if (cj == (ISP_04M_SYNCPARMS & 0xff)) {
+ x = 4;
+ } else {
+ x = 0;
+ }
+ if (x)
+ sprintf(bz, "%02dMHz:", x);
+ else
+ sprintf(bz, "?%04x:", cj);
+ } else {
+ sprintf(bz, "Async:");
+ }
+ if (sdp->isp_devparam[i].dev_flags & DPARM_WIDE)
+ bz[6] = 'W';
+ else
+ bz[6] = ' ';
+ if (sdp->isp_devparam[i].dev_flags & DPARM_TQING)
+ bz[7] = 'T';
+ else
+ bz[7] = ' ';
+ bz[8] = 0;
+ IDPRINTF(2, (" id%x:%s", i, bz));
+ if (((i+1) & 0x3) == 0)
+ IDPRINTF(2, ("\n"));
+ if (sdp->isp_devparam[i].dev_enable == 0)
+ continue;
+
+ /*
+ * It is not safe to run the 1020 in ultra mode.
+ */
+ if (isp->isp_type == ISP_HA_SCSI_1020 &&
+ cj == (ISP_20M_SYNCPARMS & 0xff)) {
+ PRINTF("%s: an ISP1020 set to Ultra Speed- derating.\n",
+ isp->isp_name);
+ sdp->isp_devparam[i].sync_offset =
+ ISP_10M_SYNCPARMS >> 8;
+ sdp->isp_devparam[i].sync_period =
+ ISP_10M_SYNCPARMS & 0xff;
+ }
+ mbs.param[0] = MBOX_SET_TARGET_PARAMS;
+ mbs.param[1] = i << 8;
+ mbs.param[2] = sdp->isp_devparam[i].dev_flags << 8;
+ mbs.param[3] =
+ (sdp->isp_devparam[i].sync_offset << 8) |
+ (sdp->isp_devparam[i].sync_period);
+
+ IDPRINTF(3, ("\n%s: target %d flags %x offset %x period %x\n",
+ isp->isp_name, i, sdp->isp_devparam[i].dev_flags,
+ sdp->isp_devparam[i].sync_offset,
+ sdp->isp_devparam[i].sync_period));
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ PRINTF("%s: failed to set parameters for target %d\n",
+ isp->isp_name, i);
+ PRINTF("%s: flags %x offset %x period %x\n",
+ isp->isp_name, sdp->isp_devparam[i].dev_flags,
+ sdp->isp_devparam[i].sync_offset,
+ sdp->isp_devparam[i].sync_period);
+ mbs.param[0] = MBOX_SET_TARGET_PARAMS;
+ mbs.param[1] = i << 8;
+ mbs.param[2] = DPARM_DEFAULT << 8;
+ mbs.param[3] = ISP_10M_SYNCPARMS;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ ISP_UNLOCK;
+ PRINTF("%s: failed even to set defaults\n",
+ isp->isp_name);
+ return;
+ }
+ }
+ for (l = 0; l < MAX_LUNS; l++) {
+ mbs.param[0] = MBOX_SET_DEV_QUEUE_PARAMS;
+ mbs.param[1] = (i << 8) | l;
+ mbs.param[2] = sdp->isp_max_queue_depth;
+ mbs.param[3] = sdp->isp_devparam[i].exc_throttle;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ ISP_UNLOCK;
+ isp_dumpregs(isp, "failed to set device queue "
+ "parameters");
+ return;
+ }
+ }
+ }
+
+ /*
+ * Set up DMA for the request and result mailboxes.
+ */
+ if (ISP_MBOXDMASETUP(isp)) {
+ ISP_UNLOCK;
+ PRINTF("%s: can't setup dma mailboxes\n", isp->isp_name);
+ return;
+ }
+
+ mbs.param[0] = MBOX_INIT_RES_QUEUE;
+ mbs.param[1] = RESULT_QUEUE_LEN(isp);
+ mbs.param[2] = (u_int16_t) (isp->isp_result_dma >> 16);
+ mbs.param[3] = (u_int16_t) (isp->isp_result_dma & 0xffff);
+ mbs.param[4] = 0;
+ mbs.param[5] = 0;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ ISP_UNLOCK;
+ isp_dumpregs(isp, "set of response queue failed");
+ return;
+ }
+ isp->isp_residx = 0;
+
+ mbs.param[0] = MBOX_INIT_REQ_QUEUE;
+ mbs.param[1] = RQUEST_QUEUE_LEN(isp);
+ mbs.param[2] = (u_int16_t) (isp->isp_rquest_dma >> 16);
+ mbs.param[3] = (u_int16_t) (isp->isp_rquest_dma & 0xffff);
+ mbs.param[4] = 0;
+ mbs.param[5] = 0;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ ISP_UNLOCK;
+ isp_dumpregs(isp, "set of request queue failed");
+ return;
+ }
+ isp->isp_reqidx = 0;
+
+ /*
+ * Unfortunately, this is the only way right now for
+ * forcing a sync renegotiation. If we boot off of
+ * an Alpha, it's put the chip in SYNC mode, but we
+ * haven't necessarily set up the parameters the
+ * same, so we'll have to yank the reset line to
+ * get everyone to renegotiate.
+ */
+
+ mbs.param[0] = MBOX_BUS_RESET;
+ mbs.param[1] = 2;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ ISP_UNLOCK;
+ isp_dumpregs(isp, "SCSI bus reset failed");
+ }
+ /*
+ * This is really important to have set after a bus reset.
+ */
+ isp->isp_sendmarker = 1;
+ ISP_UNLOCK;
+ isp->isp_state = ISP_INITSTATE;
+}
+
+static void
+isp_fibre_init(isp)
+ struct ispsoftc *isp;
+{
+ fcparam *fcp;
+ isp_icb_t *icbp;
+ mbreg_t mbs;
+ int count;
+ u_int8_t lwfs;
+ ISP_LOCKVAL_DECL;
+
+ fcp = isp->isp_param;
+
+ fcp->isp_retry_count = 0;
+ fcp->isp_retry_delay = 1;
+
+ ISP_LOCK;
+ mbs.param[0] = MBOX_SET_RETRY_COUNT;
+ mbs.param[1] = fcp->isp_retry_count;
+ mbs.param[2] = fcp->isp_retry_delay;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ ISP_UNLOCK;
+ isp_dumpregs(isp, "failed to set retry count and delay");
+ return;
+ }
+
+ if (ISP_MBOXDMASETUP(isp)) {
+ ISP_UNLOCK;
+ PRINTF("%s: can't setup DMA for mailboxes\n", isp->isp_name);
+ return;
+ }
+
+ icbp = (isp_icb_t *) fcp->isp_scratch;
+ bzero(icbp, sizeof (*icbp));
+#if 0
+ icbp->icb_maxfrmlen = ICB_DFLT_FRMLEN;
+ MAKE_NODE_NAME(isp, icbp);
+ icbp->icb_rqstqlen = RQUEST_QUEUE_LEN(isp);
+ icbp->icb_rsltqlen = RESULT_QUEUE_LEN(isp);
+ icbp->icb_rqstaddr[0] = (u_int16_t) (isp->isp_rquest_dma & 0xffff);
+ icbp->icb_rqstaddr[1] = (u_int16_t) (isp->isp_rquest_dma >> 16);
+ icbp->icb_respaddr[0] = (u_int16_t) (isp->isp_result_dma & 0xffff);
+ icbp->icb_respaddr[1] = (u_int16_t) (isp->isp_result_dma >> 16);
+#endif
+ icbp->icb_version = 1;
+ icbp->icb_maxfrmlen = ICB_DFLT_FRMLEN;
+ icbp->icb_maxalloc = 256;
+ icbp->icb_execthrottle = 16;
+ icbp->icb_retry_delay = 5;
+ icbp->icb_retry_count = 0;
+ MAKE_NODE_NAME(isp, icbp);
+ icbp->icb_rqstqlen = RQUEST_QUEUE_LEN(isp);
+ icbp->icb_rsltqlen = RESULT_QUEUE_LEN(isp);
+ icbp->icb_rqstaddr[0] = (u_int16_t) (isp->isp_rquest_dma & 0xffff);
+ icbp->icb_rqstaddr[1] = (u_int16_t) (isp->isp_rquest_dma >> 16);
+ icbp->icb_respaddr[0] = (u_int16_t) (isp->isp_result_dma & 0xffff);
+ icbp->icb_respaddr[1] = (u_int16_t) (isp->isp_result_dma >> 16);
+
+ mbs.param[0] = MBOX_INIT_FIRMWARE;
+ mbs.param[1] = 0;
+ mbs.param[2] = (u_int16_t) (fcp->isp_scdma >> 16);
+ mbs.param[3] = (u_int16_t) (fcp->isp_scdma & 0xffff);
+ mbs.param[4] = 0;
+ mbs.param[5] = 0;
+ mbs.param[6] = 0;
+ mbs.param[7] = 0;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ ISP_UNLOCK;
+ isp_dumpregs(isp, "INIT FIRMWARE failed");
+ return;
+ }
+ isp->isp_reqidx = 0;
+ isp->isp_residx = 0;
+
+ /*
+ * Wait up to 3 seconds for FW to go to READY state.
+ *
+ * This is all very much not right. The problem here
+ * is that the cable may not be plugged in, or there
+ * may be many many members of the loop that haven't
+ * been logged into.
+ *
+ * This model of doing things doesn't support dynamic
+ * attachment, so we just plain lose (for now).
+ */
+ lwfs = FW_CONFIG_WAIT;
+ for (count = 0; count < 3000; count++) {
+ isp_fw_state(isp);
+ if (lwfs != fcp->isp_fwstate) {
+ PRINTF("%s: Firmware State %s -> %s\n", isp->isp_name,
+ fw_statename(lwfs), fw_statename(fcp->isp_fwstate));
+ lwfs = fcp->isp_fwstate;
+ }
+ if (fcp->isp_fwstate == FW_READY) {
+ break;
+ }
+ SYS_DELAY(1000); /* wait one millisecond */
+ }
+ isp->isp_sendmarker = 1;
+
+ /*
+ * Get our Loop ID
+ * (if possible)
+ */
+ if (fcp->isp_fwstate == FW_READY) {
+ mbs.param[0] = MBOX_GET_LOOP_ID;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ isp_dumpregs(isp, "GET LOOP ID failed");
+ ISP_UNLOCK;
+ return;
+ }
+ fcp->isp_loopid = mbs.param[1];
+ if (fcp->isp_loopid) {
+ PRINTF("%s: Loop ID 0x%x\n", isp->isp_name,
+ fcp->isp_loopid);
+ }
+ isp->isp_state = ISP_INITSTATE;
+ }
+ ISP_UNLOCK;
+}
+
+/*
+ * Free any associated resources prior to decommissioning.
+ */
+void
+isp_uninit(isp)
+ struct ispsoftc *isp;
+{
+ STOP_WATCHDOG(isp_watch, isp);
+}
+
+
+/*
+ * start an xfer
+ */
+int32_t
+ispscsicmd(xs)
+ ISP_SCSI_XFER_T *xs;
+{
+ struct ispsoftc *isp;
+ u_int8_t iptr, optr;
+ union {
+ ispreq_t *_reqp;
+ ispreqt2_t *_t2reqp;
+ } _u;
+#define reqp _u._reqp
+#define t2reqp _u._t2reqp
+#define UZSIZE max(sizeof (ispreq_t), sizeof (ispreqt2_t))
+ int i;
+ ISP_LOCKVAL_DECL;
+
+ XS_INITERR(xs);
+ isp = XS_ISP(xs);
+
+ if (isp->isp_type & ISP_HA_FC) {
+ if (XS_CDBLEN(xs) > 12) {
+ PRINTF("%s: unsupported cdb length for fibre (%d)\n",
+ isp->isp_name, XS_CDBLEN(xs));
+ XS_SETERR(xs, HBA_BOTCH);
+ return (CMD_COMPLETE);
+ }
+ }
+ optr = ISP_READ(isp, OUTMAILBOX4);
+ iptr = isp->isp_reqidx;
+
+ reqp = (ispreq_t *) ISP_QUEUE_ENTRY(isp->isp_rquest, iptr);
+ iptr = (iptr + 1) & (RQUEST_QUEUE_LEN(isp) - 1);
+ if (iptr == optr) {
+ PRINTF("%s: Request Queue Overflow\n", isp->isp_name);
+ XS_SETERR(xs, HBA_BOTCH);
+ return (CMD_EAGAIN);
+ }
+
+ ISP_LOCK;
+ if (isp->isp_type & ISP_HA_FC)
+ DISABLE_INTS(isp);
+
+ if (isp->isp_sendmarker) {
+ ispmarkreq_t *marker = (ispmarkreq_t *) reqp;
+
+ bzero((void *) marker, sizeof (*marker));
+ marker->req_header.rqs_entry_count = 1;
+ marker->req_header.rqs_entry_type = RQSTYPE_MARKER;
+ marker->req_modifier = SYNC_ALL;
+
+ isp->isp_sendmarker = 0;
+
+ if (((iptr + 1) & (RQUEST_QUEUE_LEN(isp) - 1)) == optr) {
+ ISP_WRITE(isp, INMAILBOX4, iptr);
+ isp->isp_reqidx = iptr;
+
+ if (isp->isp_type & ISP_HA_FC)
+ ENABLE_INTS(isp);
+ ISP_UNLOCK;
+ PRINTF("%s: Request Queue Overflow+\n", isp->isp_name);
+ XS_SETERR(xs, HBA_BOTCH);
+ return (CMD_EAGAIN);
+ }
+ reqp = (ispreq_t *) ISP_QUEUE_ENTRY(isp->isp_rquest, iptr);
+ iptr = (iptr + 1) & (RQUEST_QUEUE_LEN(isp) - 1);
+ }
+
+ bzero((void *) reqp, UZSIZE);
+ reqp->req_header.rqs_entry_count = 1;
+ if (isp->isp_type & ISP_HA_FC) {
+ reqp->req_header.rqs_entry_type = RQSTYPE_T2RQS;
+ } else {
+ reqp->req_header.rqs_entry_type = RQSTYPE_REQUEST;
+ }
+ reqp->req_header.rqs_flags = 0;
+ reqp->req_header.rqs_seqno = isp->isp_seqno++;
+
+ for (i = 0; i < RQUEST_QUEUE_LEN(isp); i++) {
+ if (isp->isp_xflist[i] == NULL)
+ break;
+ }
+ if (i == RQUEST_QUEUE_LEN(isp)) {
+ if (isp->isp_type & ISP_HA_FC)
+ ENABLE_INTS(isp);
+ ISP_UNLOCK;
+ PRINTF("%s: ran out of xflist pointers?????\n", isp->isp_name);
+ XS_SETERR(xs, HBA_BOTCH);
+ return (CMD_EAGAIN);
+ } else {
+ /*
+ * Never have a handle that is zero, so
+ * set req_handle off by one.
+ */
+ isp->isp_xflist[i] = xs;
+ reqp->req_handle = i+1;
+ }
+
+ if (isp->isp_type & ISP_HA_FC) {
+ /*
+ * See comment in isp_intr
+ */
+ XS_RESID(xs) = 0;
+ /*
+ * Fibre Channel always requires some kind of tag.
+ */
+ if (XS_POLLDCMD(xs)) {
+ t2reqp->req_flags = REQFLAG_STAG;
+ } else {
+ t2reqp->req_flags = REQFLAG_OTAG;
+ }
+ } else {
+ sdparam *sdp = (sdparam *)isp->isp_param;
+ if ((sdp->isp_devparam[XS_TGT(xs)].dev_flags & DPARM_TQING) &&
+ (XS_POLLDCMD(xs) == 0)) {
+ reqp->req_flags = REQFLAG_OTAG;
+ } else {
+ reqp->req_flags = 0;
+ }
+ }
+ reqp->req_lun_trn = XS_LUN(xs);
+ reqp->req_target = XS_TGT(xs);
+ if (isp->isp_type & ISP_HA_SCSI) {
+ reqp->req_cdblen = XS_CDBLEN(xs);
+ }
+ bcopy((void *)XS_CDBP(xs), reqp->req_cdb, XS_CDBLEN(xs));
+
+ IDPRINTF(6, ("%s(%d.%d): START%d cmd 0x%x datalen %d\n", isp->isp_name,
+ XS_TGT(xs), XS_LUN(xs), reqp->req_header.rqs_seqno,
+ *(u_char *) XS_CDBP(xs), XS_XFRLEN(xs)));
+
+ reqp->req_time = XS_TIME(xs) / 1000;
+ if (reqp->req_time == 0 && XS_TIME(xs))
+ reqp->req_time = 1;
+ if (ISP_DMASETUP(isp, xs, reqp, &iptr, optr)) {
+ if (isp->isp_type & ISP_HA_FC)
+ ENABLE_INTS(isp);
+ ISP_UNLOCK;
+ XS_SETERR(xs, HBA_BOTCH);
+ return (CMD_COMPLETE);
+ }
+ XS_SETERR(xs, HBA_NOERROR);
+ ISP_WRITE(isp, INMAILBOX4, iptr);
+ isp->isp_reqidx = iptr;
+ if (isp->isp_type & ISP_HA_FC)
+ ENABLE_INTS(isp);
+ isp->isp_nactive++;
+ if (XS_POLLDCMD(xs) == 0) {
+ ISP_UNLOCK;
+ return (CMD_QUEUED);
+ }
+
+ /*
+ * If we can't use interrupts, poll on completion.
+ */
+ if (isp_poll(isp, xs, XS_TIME(xs))) {
+ /*
+ * If no other error occurred but we didn't finish,
+ * something bad happened.
+ */
+ if (XS_IS_CMD_DONE(xs) == 0) {
+ isp->isp_nactive--;
+ if (isp->isp_nactive < 0)
+ isp->isp_nactive = 0;
+ if (XS_NOERR(xs)) {
+ isp_lostcmd(isp, xs, reqp);
+ XS_SETERR(xs, HBA_BOTCH);
+ }
+ }
+ }
+ ISP_UNLOCK;
+ return (CMD_COMPLETE);
+#undef reqp
+#undef t2reqp
+}
+
+/*
+ * Interrupt Service Routine(s)
+ */
+
+int
+isp_poll(isp, xs, mswait)
+ struct ispsoftc *isp;
+ ISP_SCSI_XFER_T *xs;
+ int mswait;
+{
+
+ while (mswait) {
+ /* Try the interrupt handling routine */
+ (void)isp_intr((void *)isp);
+
+ /* See if the xs is now done */
+ if (XS_IS_CMD_DONE(xs))
+ return (0);
+ SYS_DELAY(1000); /* wait one millisecond */
+ mswait--;
+ }
+ return (1);
+}
+
+int
+isp_intr(arg)
+ void *arg;
+{
+ ISP_SCSI_XFER_T *xs;
+ struct ispsoftc *isp = arg;
+ u_int16_t iptr, optr, isr;
+
+ isr = ISP_READ(isp, BIU_ISR);
+ if (isp->isp_type & ISP_HA_FC) {
+ if (isr == 0 || (isr & BIU2100_ISR_RISC_INT) == 0) {
+ if (isr) {
+ IDPRINTF(4, ("%s: isp_intr isr=%x\n",
+ isp->isp_name, isr));
+ }
+ return (0);
+ }
+ } else {
+ if (isr == 0 || (isr & BIU_ISR_RISC_INT) == 0) {
+ if (isr) {
+ IDPRINTF(4, ("%s: isp_intr isr=%x\n",
+ isp->isp_name, isr));
+ }
+ return (0);
+ }
+ }
+
+ optr = isp->isp_residx;
+ if (ISP_READ(isp, BIU_SEMA) & 1) {
+ u_int16_t mbox0 = ISP_READ(isp, OUTMAILBOX0);
+ switch (mbox0) {
+ case ASYNC_BUS_RESET:
+ case ASYNC_TIMEOUT_RESET:
+ PRINTF("%s: bus or timeout reset\n", isp->isp_name);
+ isp->isp_sendmarker = 1;
+ break;
+ case ASYNC_LIP_OCCURRED:
+ PRINTF("%s: LIP occurred\n", isp->isp_name);
+ break;
+ case ASYNC_LOOP_UP:
+ PRINTF("%s: Loop UP\n", isp->isp_name);
+ break;
+ case ASYNC_LOOP_DOWN:
+ PRINTF("%s: Loop DOWN\n", isp->isp_name);
+ break;
+ case ASYNC_LOOP_RESET:
+ PRINTF("%s: Loop RESET\n", isp->isp_name);
+ break;
+ default:
+ PRINTF("%s: async %x\n", isp->isp_name, mbox0);
+ break;
+ }
+ ISP_WRITE(isp, BIU_SEMA, 0);
+ }
+
+ ISP_WRITE(isp, HCCR, HCCR_CMD_CLEAR_RISC_INT);
+ iptr = ISP_READ(isp, OUTMAILBOX5);
+ if (optr == iptr) {
+ IDPRINTF(4, ("why intr? isr %x iptr %x optr %x\n",
+ isr, optr, iptr));
+ }
+ ENABLE_INTS(isp);
+
+ while (optr != iptr) {
+ ispstatusreq_t *sp;
+ int buddaboom = 0;
+
+ sp = (ispstatusreq_t *) ISP_QUEUE_ENTRY(isp->isp_result, optr);
+
+ optr = (optr + 1) & (RESULT_QUEUE_LEN(isp)-1);
+ if (sp->req_header.rqs_entry_type != RQSTYPE_RESPONSE) {
+ PRINTF("%s: not RESPONSE in RESPONSE Queue (0x%x)\n",
+ isp->isp_name, sp->req_header.rqs_entry_type);
+ if (sp->req_header.rqs_entry_type != RQSTYPE_REQUEST) {
+ ISP_WRITE(isp, INMAILBOX5, optr);
+ continue;
+ }
+ buddaboom = 1;
+ }
+
+ if (sp->req_header.rqs_flags & 0xf) {
+ if (sp->req_header.rqs_flags & RQSFLAG_CONTINUATION) {
+ ISP_WRITE(isp, INMAILBOX5, optr);
+ continue;
+ }
+ PRINTF("%s: rqs_flags=%x\n", isp->isp_name,
+ sp->req_header.rqs_flags & 0xf);
+ }
+ if (sp->req_handle > RQUEST_QUEUE_LEN(isp) ||
+ sp->req_handle < 1) {
+ PRINTF("%s: bad request handle %d\n", isp->isp_name,
+ sp->req_handle);
+ ISP_WRITE(isp, INMAILBOX5, optr);
+ continue;
+ }
+ xs = (ISP_SCSI_XFER_T *) isp->isp_xflist[sp->req_handle - 1];
+ if (xs == NULL) {
+ PRINTF("%s: NULL xs in xflist\n", isp->isp_name);
+ ISP_WRITE(isp, INMAILBOX5, optr);
+ continue;
+ }
+ isp->isp_xflist[sp->req_handle - 1] = NULL;
+ if (sp->req_status_flags & RQSTF_BUS_RESET) {
+ isp->isp_sendmarker = 1;
+ }
+ if (buddaboom) {
+ XS_SETERR(xs, HBA_BOTCH);
+ }
+ XS_STS(xs) = sp->req_scsi_status & 0xff;
+ if (isp->isp_type & ISP_HA_SCSI) {
+ if (sp->req_state_flags & RQSF_GOT_SENSE) {
+ bcopy(sp->req_sense_data, XS_SNSP(xs),
+ XS_SNSLEN(xs));
+ XS_SNS_IS_VALID(xs);
+ }
+ } else {
+ if (XS_STS(xs) == SCSI_CHECK) {
+ XS_SNS_IS_VALID(xs);
+ bcopy(sp->req_sense_data, XS_SNSP(xs),
+ XS_SNSLEN(xs));
+ sp->req_state_flags |= RQSF_GOT_SENSE;
+ }
+ }
+ if (XS_NOERR(xs) && XS_STS(xs) == SCSI_BUSY) {
+ XS_SETERR(xs, HBA_TGTBSY);
+ }
+
+ if (sp->req_header.rqs_entry_type == RQSTYPE_RESPONSE) {
+ if (XS_NOERR(xs) && sp->req_completion_status)
+ isp_parse_status(isp, sp, xs);
+ } else {
+ PRINTF("%s: unknown return %x\n", isp->isp_name,
+ sp->req_header.rqs_entry_type);
+ if (XS_NOERR(xs))
+ XS_SETERR(xs, HBA_BOTCH);
+ }
+ if (isp->isp_type & ISP_HA_SCSI) {
+ XS_RESID(xs) = sp->req_resid;
+ } else if (sp->req_scsi_status & RQCS_RU) {
+ XS_RESID(xs) = sp->req_resid;
+ IDPRINTF(4, ("%s: cnt %d rsd %d\n", isp->isp_name,
+ XS_XFRLEN(xs), sp->req_resid));
+ }
+ if (XS_XFRLEN(xs)) {
+ ISP_DMAFREE(isp, xs, sp->req_handle - 1);
+ }
+ if ((isp->isp_dblev >= 5) ||
+ (isp->isp_dblev > 2 && !XS_NOERR(xs))) {
+ PRINTF("%s(%d.%d): FIN%d cmd0x%x len%d resid%d STS %x",
+ isp->isp_name, XS_TGT(xs), XS_LUN(xs),
+ sp->req_header.rqs_seqno, *(u_char *) XS_CDBP(xs),
+ XS_XFRLEN(xs), XS_RESID(xs), XS_STS(xs));
+ if (sp->req_state_flags & RQSF_GOT_SENSE) {
+ PRINTF(" Skey: %x", XS_SNSKEY(xs));
+ if (!(XS_IS_SNS_VALID(xs))) {
+ PRINTF(" BUT NOT SET");
+ }
+ }
+ PRINTF(" XS_ERR(xs) %d\n", XS_ERR(xs));
+ }
+ ISP_WRITE(isp, INMAILBOX5, optr);
+ isp->isp_nactive--;
+ if (isp->isp_nactive < 0)
+ isp->isp_nactive = 0;
+ XS_CMD_DONE(xs);
+ }
+ isp->isp_residx = optr;
+ return (1);
+}
+
+/*
+ * Support routines.
+ */
+
+static void
+isp_parse_status(isp, sp, xs)
+ struct ispsoftc *isp;
+ ispstatusreq_t *sp;
+ ISP_SCSI_XFER_T *xs;
+{
+ switch (sp->req_completion_status) {
+ case RQCS_COMPLETE:
+ XS_SETERR(xs, HBA_NOERROR);
+ return;
+
+ case RQCS_INCOMPLETE:
+ if ((sp->req_state_flags & RQSF_GOT_TARGET) == 0) {
+ XS_SETERR(xs, HBA_SELTIMEOUT);
+ return;
+ }
+ PRINTF("%s: incomplete, state %x\n",
+ isp->isp_name, sp->req_state_flags);
+ break;
+
+ case RQCS_TRANSPORT_ERROR:
+ PRINTF("%s: transport error\n", isp->isp_name);
+ isp_prtstst(sp);
+ break;
+
+ case RQCS_DATA_OVERRUN:
+ if (isp->isp_type & ISP_HA_FC) {
+ XS_RESID(xs) = sp->req_resid;
+ break;
+ }
+ XS_SETERR(xs, HBA_NOERROR);
+ return;
+
+ case RQCS_DATA_UNDERRUN:
+ if (isp->isp_type & ISP_HA_FC) {
+ XS_RESID(xs) = sp->req_resid;
+ /* an UNDERRUN is not a botch ??? */
+ }
+ XS_SETERR(xs, HBA_NOERROR);
+ return;
+
+ case RQCS_TIMEOUT:
+ XS_SETERR(xs, HBA_CMDTIMEOUT);
+ return;
+
+ case RQCS_RESET_OCCURRED:
+ PRINTF("%s: reset occurred, %d active\n", isp->isp_name,
+ isp->isp_nactive);
+ isp->isp_sendmarker = 1;
+ XS_SETERR(xs, HBA_BUSRESET);
+ return;
+
+ case RQCS_ABORTED:
+ PRINTF("%s: command aborted\n", isp->isp_name);
+ isp->isp_sendmarker = 1;
+ XS_SETERR(xs, HBA_ABORTED);
+ return;
+
+ case RQCS_PORT_UNAVAILABLE:
+ /*
+ * No such port on the loop. Moral equivalent of SELTIMEO
+ */
+ XS_SETERR(xs, HBA_SELTIMEOUT);
+ return;
+
+ case RQCS_PORT_LOGGED_OUT:
+ PRINTF("%s: port logout for target %d\n",
+ isp->isp_name, XS_TGT(xs));
+ break;
+
+ case RQCS_PORT_CHANGED:
+ PRINTF("%s: port changed for target %d\n",
+ isp->isp_name, XS_TGT(xs));
+ break;
+
+ case RQCS_PORT_BUSY:
+ PRINTF("%s: port busy for target %d\n",
+ isp->isp_name, XS_TGT(xs));
+ XS_SETERR(xs, HBA_TGTBSY);
+ return;
+
+ default:
+ PRINTF("%s: comp status %x\n", isp->isp_name,
+ sp->req_completion_status);
+ break;
+ }
+ XS_SETERR(xs, HBA_BOTCH);
+}
+
+#define HINIB(x) ((x) >> 0x4)
+#define LONIB(x) ((x) & 0xf)
+#define MAKNIB(a, b) (((a) << 4) | (b))
+static u_int8_t mbpcnt[] = {
+ MAKNIB(1, 1), /* 0x00: MBOX_NO_OP */
+ MAKNIB(5, 5), /* 0x01: MBOX_LOAD_RAM */
+ MAKNIB(2, 0), /* 0x02: MBOX_EXEC_FIRMWARE */
+ MAKNIB(5, 5), /* 0x03: MBOX_DUMP_RAM */
+ MAKNIB(3, 3), /* 0x04: MBOX_WRITE_RAM_WORD */
+ MAKNIB(2, 3), /* 0x05: MBOX_READ_RAM_WORD */
+ MAKNIB(6, 6), /* 0x06: MBOX_MAILBOX_REG_TEST */
+ MAKNIB(2, 3), /* 0x07: MBOX_VERIFY_CHECKSUM */
+ MAKNIB(1, 3), /* 0x08: MBOX_ABOUT_FIRMWARE */
+ MAKNIB(0, 0), /* 0x09: */
+ MAKNIB(0, 0), /* 0x0a: */
+ MAKNIB(0, 0), /* 0x0b: */
+ MAKNIB(0, 0), /* 0x0c: */
+ MAKNIB(0, 0), /* 0x0d: */
+ MAKNIB(1, 2), /* 0x0e: MBOX_CHECK_FIRMWARE */
+ MAKNIB(0, 0), /* 0x0f: */
+ MAKNIB(5, 5), /* 0x10: MBOX_INIT_REQ_QUEUE */
+ MAKNIB(6, 6), /* 0x11: MBOX_INIT_RES_QUEUE */
+ MAKNIB(4, 4), /* 0x12: MBOX_EXECUTE_IOCB */
+ MAKNIB(2, 2), /* 0x13: MBOX_WAKE_UP */
+ MAKNIB(1, 6), /* 0x14: MBOX_STOP_FIRMWARE */
+ MAKNIB(4, 4), /* 0x15: MBOX_ABORT */
+ MAKNIB(2, 2), /* 0x16: MBOX_ABORT_DEVICE */
+ MAKNIB(3, 3), /* 0x17: MBOX_ABORT_TARGET */
+ MAKNIB(2, 2), /* 0x18: MBOX_BUS_RESET */
+ MAKNIB(2, 3), /* 0x19: MBOX_STOP_QUEUE */
+ MAKNIB(2, 3), /* 0x1a: MBOX_START_QUEUE */
+ MAKNIB(2, 3), /* 0x1b: MBOX_SINGLE_STEP_QUEUE */
+ MAKNIB(2, 3), /* 0x1c: MBOX_ABORT_QUEUE */
+ MAKNIB(2, 4), /* 0x1d: MBOX_GET_DEV_QUEUE_STATUS */
+ MAKNIB(0, 0), /* 0x1e: */
+ MAKNIB(1, 3), /* 0x1f: MBOX_GET_FIRMWARE_STATUS */
+ MAKNIB(1, 2), /* 0x20: MBOX_GET_INIT_SCSI_ID */
+ MAKNIB(1, 2), /* 0x21: MBOX_GET_SELECT_TIMEOUT */
+ MAKNIB(1, 3), /* 0x22: MBOX_GET_RETRY_COUNT */
+ MAKNIB(1, 2), /* 0x23: MBOX_GET_TAG_AGE_LIMIT */
+ MAKNIB(1, 2), /* 0x24: MBOX_GET_CLOCK_RATE */
+ MAKNIB(1, 2), /* 0x25: MBOX_GET_ACT_NEG_STATE */
+ MAKNIB(1, 2), /* 0x26: MBOX_GET_ASYNC_DATA_SETUP_TIME */
+ MAKNIB(1, 3), /* 0x27: MBOX_GET_PCI_PARAMS */
+ MAKNIB(2, 4), /* 0x28: MBOX_GET_TARGET_PARAMS */
+ MAKNIB(2, 4), /* 0x29: MBOX_GET_DEV_QUEUE_PARAMS */
+ MAKNIB(0, 0), /* 0x2a: */
+ MAKNIB(0, 0), /* 0x2b: */
+ MAKNIB(0, 0), /* 0x2c: */
+ MAKNIB(0, 0), /* 0x2d: */
+ MAKNIB(0, 0), /* 0x2e: */
+ MAKNIB(0, 0), /* 0x2f: */
+ MAKNIB(2, 2), /* 0x30: MBOX_SET_INIT_SCSI_ID */
+ MAKNIB(2, 2), /* 0x31: MBOX_SET_SELECT_TIMEOUT */
+ MAKNIB(3, 3), /* 0x32: MBOX_SET_RETRY_COUNT */
+ MAKNIB(2, 2), /* 0x33: MBOX_SET_TAG_AGE_LIMIT */
+ MAKNIB(2, 2), /* 0x34: MBOX_SET_CLOCK_RATE */
+ MAKNIB(2, 2), /* 0x35: MBOX_SET_ACTIVE_NEG_STATE */
+ MAKNIB(2, 2), /* 0x36: MBOX_SET_ASYNC_DATA_SETUP_TIME */
+ MAKNIB(3, 3), /* 0x37: MBOX_SET_PCI_CONTROL_PARAMS */
+ MAKNIB(4, 4), /* 0x38: MBOX_SET_TARGET_PARAMS */
+ MAKNIB(4, 4), /* 0x39: MBOX_SET_DEV_QUEUE_PARAMS */
+ MAKNIB(0, 0), /* 0x3a: */
+ MAKNIB(0, 0), /* 0x3b: */
+ MAKNIB(0, 0), /* 0x3c: */
+ MAKNIB(0, 0), /* 0x3d: */
+ MAKNIB(0, 0), /* 0x3e: */
+ MAKNIB(0, 0), /* 0x3f: */
+ MAKNIB(1, 2), /* 0x40: MBOX_RETURN_BIOS_BLOCK_ADDR */
+ MAKNIB(6, 1), /* 0x41: MBOX_WRITE_FOUR_RAM_WORDS */
+ MAKNIB(2, 3), /* 0x42: MBOX_EXEC_BIOS_IOCB */
+ MAKNIB(0, 0), /* 0x43: */
+ MAKNIB(0, 0), /* 0x44: */
+ MAKNIB(0, 0), /* 0x45: */
+ MAKNIB(0, 0), /* 0x46: */
+ MAKNIB(0, 0), /* 0x47: */
+ MAKNIB(0, 0), /* 0x48: */
+ MAKNIB(0, 0), /* 0x49: */
+ MAKNIB(0, 0), /* 0x4a: */
+ MAKNIB(0, 0), /* 0x4b: */
+ MAKNIB(0, 0), /* 0x4c: */
+ MAKNIB(0, 0), /* 0x4d: */
+ MAKNIB(0, 0), /* 0x4e: */
+ MAKNIB(0, 0), /* 0x4f: */
+ MAKNIB(0, 0), /* 0x50: */
+ MAKNIB(0, 0), /* 0x51: */
+ MAKNIB(0, 0), /* 0x52: */
+ MAKNIB(0, 0), /* 0x53: */
+ MAKNIB(8, 0), /* 0x54: MBOX_EXEC_COMMAND_IOCB_A64 */
+ MAKNIB(0, 0), /* 0x55: */
+ MAKNIB(0, 0), /* 0x56: */
+ MAKNIB(0, 0), /* 0x57: */
+ MAKNIB(0, 0), /* 0x58: */
+ MAKNIB(0, 0), /* 0x59: */
+ MAKNIB(0, 0), /* 0x5a: */
+ MAKNIB(0, 0), /* 0x5b: */
+ MAKNIB(0, 0), /* 0x5c: */
+ MAKNIB(0, 0), /* 0x5d: */
+ MAKNIB(0, 0), /* 0x5e: */
+ MAKNIB(0, 0), /* 0x5f: */
+ MAKNIB(8, 6), /* 0x60: MBOX_INIT_FIRMWARE */
+ MAKNIB(0, 0), /* 0x60: MBOX_GET_INIT_CONTROL_BLOCK (FORMAT?) */
+ MAKNIB(2, 1), /* 0x62: MBOX_INIT_LIP */
+ MAKNIB(8, 1), /* 0x63: MBOX_GET_FC_AL_POSITION_MAP */
+ MAKNIB(8, 1), /* 0x64: MBOX_GET_PORT_DB */
+ MAKNIB(3, 1), /* 0x65: MBOX_CLEAR_ACA */
+ MAKNIB(3, 1), /* 0x66: MBOX_TARGET_RESET */
+ MAKNIB(3, 1), /* 0x67: MBOX_CLEAR_TASK_SET */
+ MAKNIB(3, 1), /* 0x69: MBOX_ABORT_TASK_SET */
+ MAKNIB(1, 2) /* 0x69: MBOX_GET_FW_STATE */
+};
+#define NMBCOM (sizeof (mbpcnt) / sizeof (mbpcnt[0]))
+
+static void
+isp_mboxcmd(isp, mbp)
+ struct ispsoftc *isp;
+ mbreg_t *mbp;
+{
+ int outparam, inparam;
+ int loops;
+ u_int8_t opcode;
+
+ if (mbp->param[0] == ISP2100_SET_PCI_PARAM) {
+ opcode = mbp->param[0] = MBOX_SET_PCI_PARAMETERS;
+ inparam = 4;
+ outparam = 4;
+ goto command_known;
+ } else if (mbp->param[0] > NMBCOM) {
+ PRINTF("%s: bad command %x\n", isp->isp_name, mbp->param[0]);
+ return;
+ }
+
+ opcode = mbp->param[0];
+ inparam = HINIB(mbpcnt[mbp->param[0]]);
+ outparam = LONIB(mbpcnt[mbp->param[0]]);
+
+ if (inparam == 0 && outparam == 0) {
+ PRINTF("%s: no parameters for %x\n", isp->isp_name,
+ mbp->param[0]);
+ return;
+ }
+
+
+command_known:
+ /*
+ * Make sure we can send some words..
+ */
+
+ loops = MBOX_DELAY_COUNT;
+ while ((ISP_READ(isp, HCCR) & HCCR_HOST_INT) != 0) {
+ SYS_DELAY(100);
+ if (--loops < 0) {
+ PRINTF("%s: isp_mboxcmd timeout #1\n", isp->isp_name);
+ return;
+ }
+ }
+
+ /*
+ * Write input parameters
+ */
+ switch (inparam) {
+ case 8: ISP_WRITE(isp, INMAILBOX7, mbp->param[7]); mbp->param[7] = 0;
+ case 7: ISP_WRITE(isp, INMAILBOX6, mbp->param[6]); mbp->param[6] = 0;
+ case 6: ISP_WRITE(isp, INMAILBOX5, mbp->param[5]); mbp->param[5] = 0;
+ case 5: ISP_WRITE(isp, INMAILBOX4, mbp->param[4]); mbp->param[4] = 0;
+ case 4: ISP_WRITE(isp, INMAILBOX3, mbp->param[3]); mbp->param[3] = 0;
+ case 3: ISP_WRITE(isp, INMAILBOX2, mbp->param[2]); mbp->param[2] = 0;
+ case 2: ISP_WRITE(isp, INMAILBOX1, mbp->param[1]); mbp->param[1] = 0;
+ case 1: ISP_WRITE(isp, INMAILBOX0, mbp->param[0]); mbp->param[0] = 0;
+ }
+
+ /*
+ * Clear semaphore on mailbox registers
+ */
+ ISP_WRITE(isp, BIU_SEMA, 0);
+
+ /*
+ * Clear RISC int condition.
+ */
+ ISP_WRITE(isp, HCCR, HCCR_CMD_CLEAR_RISC_INT);
+
+ /*
+ * Set Host Interrupt condition so that RISC will pick up mailbox regs.
+ */
+ ISP_WRITE(isp, HCCR, HCCR_CMD_SET_HOST_INT);
+
+ /*
+ * Wait until RISC int is set, except 2100
+ */
+ if ((isp->isp_type & ISP_HA_FC) == 0) {
+ loops = MBOX_DELAY_COUNT;
+ while ((ISP_READ(isp, BIU_ISR) & BIU_ISR_RISC_INT) == 0) {
+ SYS_DELAY(100);
+ if (--loops < 0) {
+ PRINTF("%s: isp_mboxcmd timeout #2\n",
+ isp->isp_name);
+ return;
+ }
+ }
+ }
+
+ /*
+ * Check to make sure that the semaphore has been set.
+ */
+ loops = MBOX_DELAY_COUNT;
+ while ((ISP_READ(isp, BIU_SEMA) & 1) == 0) {
+ SYS_DELAY(100);
+ if (--loops < 0) {
+ PRINTF("%s: isp_mboxcmd timeout #3\n", isp->isp_name);
+ return;
+ }
+ }
+
+ /*
+ * Make sure that the MBOX_BUSY has gone away
+ */
+ loops = MBOX_DELAY_COUNT;
+ while (ISP_READ(isp, OUTMAILBOX0) == MBOX_BUSY) {
+ SYS_DELAY(100);
+ if (--loops < 0) {
+ PRINTF("%s: isp_mboxcmd timeout #4\n", isp->isp_name);
+ return;
+ }
+ }
+
+
+ /*
+ * Pick up output parameters.
+ */
+ switch (outparam) {
+ case 8: mbp->param[7] = ISP_READ(isp, OUTMAILBOX7);
+ case 7: mbp->param[6] = ISP_READ(isp, OUTMAILBOX6);
+ case 6: mbp->param[5] = ISP_READ(isp, OUTMAILBOX5);
+ case 5: mbp->param[4] = ISP_READ(isp, OUTMAILBOX4);
+ case 4: mbp->param[3] = ISP_READ(isp, OUTMAILBOX3);
+ case 3: mbp->param[2] = ISP_READ(isp, OUTMAILBOX2);
+ case 2: mbp->param[1] = ISP_READ(isp, OUTMAILBOX1);
+ case 1: mbp->param[0] = ISP_READ(isp, OUTMAILBOX0);
+ }
+
+ /*
+ * Clear RISC int.
+ */
+ ISP_WRITE(isp, HCCR, HCCR_CMD_CLEAR_RISC_INT);
+
+ /*
+ * Release semaphore on mailbox registers
+ */
+ ISP_WRITE(isp, BIU_SEMA, 0);
+
+ /*
+ * Just to be chatty here...
+ */
+ switch(mbp->param[0]) {
+ case MBOX_COMMAND_COMPLETE:
+ break;
+ case MBOX_INVALID_COMMAND:
+ /*
+ * GET_CLOCK_RATE can fail a lot
+ * So can a couple of other commands.
+ */
+ if (isp->isp_dblev > 2 && opcode != MBOX_GET_CLOCK_RATE) {
+ PRINTF("%s: mbox cmd %x failed with INVALID_COMMAND\n",
+ isp->isp_name, opcode);
+ }
+ break;
+ case MBOX_HOST_INTERFACE_ERROR:
+ PRINTF("%s: mbox cmd %x failed with HOST_INTERFACE_ERROR\n",
+ isp->isp_name, opcode);
+ break;
+ case MBOX_TEST_FAILED:
+ PRINTF("%s: mbox cmd %x failed with TEST_FAILED\n",
+ isp->isp_name, opcode);
+ break;
+ case MBOX_COMMAND_ERROR:
+ PRINTF("%s: mbox cmd %x failed with COMMAND_ERROR\n",
+ isp->isp_name, opcode);
+ break;
+ case MBOX_COMMAND_PARAM_ERROR:
+ PRINTF("%s: mbox cmd %x failed with COMMAND_PARAM_ERROR\n",
+ isp->isp_name, opcode);
+ break;
+
+ case ASYNC_LIP_OCCURRED:
+ break;
+
+ default:
+ /*
+ * The expected return of EXEC_FIRMWARE is zero.
+ */
+ if ((opcode == MBOX_EXEC_FIRMWARE && mbp->param[0] != 0) ||
+ (opcode != MBOX_EXEC_FIRMWARE)) {
+ PRINTF("%s: mbox cmd %x failed with error %x\n",
+ isp->isp_name, opcode, mbp->param[0]);
+ }
+ break;
+ }
+}
+
+static void
+isp_lostcmd(struct ispsoftc *isp, ISP_SCSI_XFER_T *xs, ispreq_t *req)
+{
+ mbreg_t mbs;
+
+ mbs.param[0] = MBOX_GET_FIRMWARE_STATUS;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ isp_dumpregs(isp, "couldn't GET FIRMWARE STATUS");
+ return;
+ }
+ if (mbs.param[1]) {
+ PRINTF("%s: %d commands on completion queue\n",
+ isp->isp_name, mbs.param[1]);
+ }
+ if (XS_NULL(xs))
+ return;
+
+ mbs.param[0] = MBOX_GET_DEV_QUEUE_STATUS;
+ mbs.param[1] = XS_TGT(xs) << 8 | XS_LUN(xs);
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ isp_dumpregs(isp, "couldn't GET DEVICE QUEUE STATUS");
+ return;
+ }
+ PRINTF("%s: lost command for target %d lun %d, %d active of %d, "
+ "Queue State: %x\n", isp->isp_name, XS_TGT(xs),
+ XS_LUN(xs), mbs.param[2], mbs.param[3], mbs.param[1]);
+
+ isp_dumpregs(isp, "lost command");
+ /*
+ * XXX: Need to try and do something to recover.
+ */
+}
+
+static void
+isp_dumpregs(struct ispsoftc *isp, const char *msg)
+{
+ PRINTF("%s: %s\n", isp->isp_name, msg);
+ if (isp->isp_type & ISP_HA_SCSI)
+ PRINTF(" biu_conf1=%x", ISP_READ(isp, BIU_CONF1));
+ else
+ PRINTF(" biu_csr=%x", ISP_READ(isp, BIU2100_CSR));
+ PRINTF(" biu_icr=%x biu_isr=%x biu_sema=%x ", ISP_READ(isp, BIU_ICR),
+ ISP_READ(isp, BIU_ISR), ISP_READ(isp, BIU_SEMA));
+ PRINTF("risc_hccr=%x\n", ISP_READ(isp, HCCR));
+
+ if (isp->isp_type & ISP_HA_SCSI) {
+ ISP_WRITE(isp, HCCR, HCCR_CMD_PAUSE);
+ PRINTF(" cdma_conf=%x cdma_sts=%x cdma_fifostat=%x\n",
+ ISP_READ(isp, CDMA_CONF), ISP_READ(isp, CDMA_STATUS),
+ ISP_READ(isp, CDMA_FIFO_STS));
+ PRINTF(" ddma_conf=%x ddma_sts=%x ddma_fifostat=%x\n",
+ ISP_READ(isp, DDMA_CONF), ISP_READ(isp, DDMA_STATUS),
+ ISP_READ(isp, DDMA_FIFO_STS));
+ PRINTF(" sxp_int=%x sxp_gross=%x sxp(scsi_ctrl)=%x\n",
+ ISP_READ(isp, SXP_INTERRUPT),
+ ISP_READ(isp, SXP_GROSS_ERR),
+ ISP_READ(isp, SXP_PINS_CONTROL));
+ ISP_WRITE(isp, HCCR, HCCR_CMD_RELEASE);
+ }
+ ISP_DUMPREGS(isp);
+}
+
+static void
+isp_fw_state(struct ispsoftc *isp)
+{
+ mbreg_t mbs;
+ if (isp->isp_type & ISP_HA_FC) {
+ int once = 0;
+ fcparam *fcp = isp->isp_param;
+again:
+ mbs.param[0] = MBOX_GET_FW_STATE;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ if (mbs.param[0] == ASYNC_LIP_OCCURRED) {
+ if (!once++) {
+ goto again;
+ }
+ }
+ isp_dumpregs(isp, "GET FIRMWARE STATE failed");
+ return;
+ }
+ fcp->isp_fwstate = mbs.param[1];
+ }
+}
+
+static void
+isp_setdparm(struct ispsoftc *isp)
+{
+ int i;
+ mbreg_t mbs;
+ sdparam *sdp;
+
+ isp->isp_fwrev = 0;
+ if (isp->isp_type & ISP_HA_FC) {
+ /*
+ * ROM in 2100 doesn't appear to support ABOUT_FIRMWARE
+ */
+ return;
+ }
+
+ mbs.param[0] = MBOX_ABOUT_FIRMWARE;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ IDPRINTF(3, ("1st ABOUT FIRMWARE command failed"));
+ } else {
+ isp->isp_fwrev =
+ (((u_int16_t) mbs.param[1]) << 10) + mbs.param[2];
+ }
+
+
+ sdp = (sdparam *) isp->isp_param;
+ /*
+ * Try and get old clock rate out before we hit the
+ * chip over the head- but if and only if we don't
+ * know our desired clock rate.
+ */
+ if (isp->isp_mdvec->dv_clock == 0) {
+ mbs.param[0] = MBOX_GET_CLOCK_RATE;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] == MBOX_COMMAND_COMPLETE) {
+ sdp->isp_clock = mbs.param[1];
+ PRINTF("%s: using board clock 0x%x\n",
+ isp->isp_name, sdp->isp_clock);
+ }
+ } else {
+ sdp->isp_clock = isp->isp_mdvec->dv_clock;
+ }
+
+ mbs.param[0] = MBOX_GET_ACT_NEG_STATE;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ IDPRINTF(2, ("could not GET ACT NEG STATE\n"));
+ sdp->isp_req_ack_active_neg = 1;
+ sdp->isp_data_line_active_neg = 1;
+ } else {
+ sdp->isp_req_ack_active_neg = (mbs.param[1] >> 4) & 0x1;
+ sdp->isp_data_line_active_neg = (mbs.param[1] >> 5) & 0x1;
+ }
+ for (i = 0; i < MAX_TARGETS; i++) {
+ mbs.param[0] = MBOX_GET_TARGET_PARAMS;
+ mbs.param[1] = i << 8;
+ isp_mboxcmd(isp, &mbs);
+ if (mbs.param[0] != MBOX_COMMAND_COMPLETE) {
+ IDPRINTF(2, ("cannot get params for target %d\n", i));
+ sdp->isp_devparam[i].sync_period =
+ ISP_10M_SYNCPARMS & 0xff;
+ sdp->isp_devparam[i].sync_offset =
+ ISP_10M_SYNCPARMS >> 8;
+ sdp->isp_devparam[i].dev_flags = DPARM_DEFAULT;
+ } else {
+ IDPRINTF(3, ("\%s: target %d - flags 0x%x, sync %x\n",
+ isp->isp_name, i, mbs.param[2], mbs.param[3]));
+ sdp->isp_devparam[i].dev_flags = mbs.param[2] >> 8;
+ /*
+ * The maximum period we can really see
+ * here is 100 (decimal), or 400 ns.
+ * For some unknown reason we sometimes
+ * get back wildass numbers from the
+ * boot device's paramaters.
+ */
+ if ((mbs.param[3] & 0xff) <= 0x64) {
+ sdp->isp_devparam[i].sync_period =
+ mbs.param[3] & 0xff;
+ sdp->isp_devparam[i].sync_offset =
+ mbs.param[3] >> 8;
+ }
+ }
+ }
+
+ /*
+ * Set Default Host Adapter Parameters
+ * XXX: Should try and get them out of NVRAM
+ */
+ sdp->isp_adapter_enabled = 1;
+ sdp->isp_cmd_dma_burst_enable = 1;
+ sdp->isp_data_dma_burst_enabl = 1;
+ sdp->isp_fifo_threshold = 2;
+ sdp->isp_initiator_id = 7;
+ sdp->isp_async_data_setup = 6;
+ sdp->isp_selection_timeout = 250;
+ sdp->isp_max_queue_depth = 128;
+ sdp->isp_tag_aging = 8;
+ sdp->isp_bus_reset_delay = 3;
+ sdp->isp_retry_count = 0;
+ sdp->isp_retry_delay = 1;
+
+ for (i = 0; i < MAX_TARGETS; i++) {
+ sdp->isp_devparam[i].exc_throttle = 16;
+ sdp->isp_devparam[i].dev_enable = 1;
+ }
+}
+
+static void
+isp_phoenix(struct ispsoftc *isp)
+{
+ ISP_SCSI_XFER_T *tlist[MAXISPREQUEST], *xs;
+ int i;
+
+ for (i = 0; i < RQUEST_QUEUE_LEN(isp); i++) {
+ tlist[i] = (ISP_SCSI_XFER_T *) isp->isp_xflist[i];
+ }
+ isp_reset(isp);
+ isp_init(isp);
+ isp->isp_state = ISP_RUNSTATE;
+
+ for (i = 0; i < RQUEST_QUEUE_LEN(isp); i++) {
+ xs = tlist[i];
+ if (XS_NULL(xs))
+ continue;
+ isp->isp_nactive--;
+ if (isp->isp_nactive < 0)
+ isp->isp_nactive = 0;
+ XS_RESID(xs) = XS_XFRLEN(xs);
+ XS_SETERR(xs, HBA_BOTCH);
+ XS_CMD_DONE(xs);
+ }
+}
+
+void
+isp_watch(void *arg)
+{
+ int i;
+ struct ispsoftc *isp = arg;
+ ISP_SCSI_XFER_T *xs;
+ ISP_LOCKVAL_DECL;
+
+ /*
+ * Look for completely dead commands (but not polled ones).
+ */
+ ISP_LOCK;
+ for (i = 0; i < RQUEST_QUEUE_LEN(isp); i++) {
+ if ((xs = (ISP_SCSI_XFER_T *) isp->isp_xflist[i]) == NULL) {
+ continue;
+ }
+ if (XS_POLLDCMD(xs))
+ continue;
+ if (XS_TIME(xs) == 0) {
+ continue;
+ }
+ XS_TIME(xs) -= (WATCH_INTERVAL * 1000);
+ /*
+ * Avoid later thinking that this
+ * transaction is not being timed.
+ * Then give ourselves to watchdog
+ * periods of grace.
+ */
+ if (XS_TIME(xs) == 0)
+ XS_TIME(xs) = 1;
+ else if (XS_TIME(xs) > -(2 * WATCH_INTERVAL * 1000)) {
+ continue;
+ }
+ if (isp_abortcmd(isp, i)) {
+ PRINTF("%s: isp_watch failed to abort command\n",
+ isp->isp_name);
+ isp_phoenix(isp);
+ break;
+ }
+ }
+ ISP_UNLOCK;
+ RESTART_WATCHDOG(isp_watch, isp);
+}
+
+static void
+isp_prtstst(ispstatusreq_t *sp)
+{
+ PRINTF("states->");
+ if (sp->req_state_flags & RQSF_GOT_BUS)
+ PRINTF("GOT_BUS ");
+ if (sp->req_state_flags & RQSF_GOT_TARGET)
+ PRINTF("GOT_TGT ");
+ if (sp->req_state_flags & RQSF_SENT_CDB)
+ PRINTF("SENT_CDB ");
+ if (sp->req_state_flags & RQSF_XFRD_DATA)
+ PRINTF("XFRD_DATA ");
+ if (sp->req_state_flags & RQSF_GOT_STATUS)
+ PRINTF("GOT_STS ");
+ if (sp->req_state_flags & RQSF_GOT_SENSE)
+ PRINTF("GOT_SNS ");
+ if (sp->req_state_flags & RQSF_XFER_COMPLETE)
+ PRINTF("XFR_CMPLT ");
+ PRINTF("\n");
+ PRINTF("status->");
+ if (sp->req_status_flags & RQSTF_DISCONNECT)
+ PRINTF("Disconnect ");
+ if (sp->req_status_flags & RQSTF_SYNCHRONOUS)
+ PRINTF("Sync_xfr ");
+ if (sp->req_status_flags & RQSTF_PARITY_ERROR)
+ PRINTF("Parity ");
+ if (sp->req_status_flags & RQSTF_BUS_RESET)
+ PRINTF("Bus_Reset ");
+ if (sp->req_status_flags & RQSTF_DEVICE_RESET)
+ PRINTF("Device_Reset ");
+ if (sp->req_status_flags & RQSTF_ABORTED)
+ PRINTF("Aborted ");
+ if (sp->req_status_flags & RQSTF_TIMEOUT)
+ PRINTF("Timeout ");
+ if (sp->req_status_flags & RQSTF_NEGOTIATION)
+ PRINTF("Negotiation ");
+ PRINTF("\n");
+}
OpenPOWER on IntegriCloud