summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/dev/ic/isp.c213
1 files changed, 142 insertions, 71 deletions
diff --git a/sys/dev/ic/isp.c b/sys/dev/ic/isp.c
index ef4c101a4dd..727e1d6e654 100644
--- a/sys/dev/ic/isp.c
+++ b/sys/dev/ic/isp.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: isp.c,v 1.18 2000/12/06 01:07:21 mjacob Exp $ */
+/* $OpenBSD: isp.c,v 1.19 2001/01/09 03:26:18 mjacob Exp $ */
/*
* Machine and OS Independent (well, as best as possible)
* code for the Qlogic ISP SCSI adapters.
@@ -64,43 +64,43 @@
/*
* Local static data
*/
-static char *warnlun =
+static const char warnlun[] =
"WARNING- cannot determine Expanded LUN capability- limiting to one LUN";
-static char *portshift =
+static const char portshift[] =
"Target %d Loop ID 0x%x (Port 0x%x) => Loop 0x%x (Port 0x%x)";
-static char *portdup =
+static const char portdup[] =
"Target %d duplicates Target %d- killing off both";
-static char *retained =
+static const char retained[] =
"Retaining Loop ID 0x%x for Target %d (Port 0x%x)";
#ifdef ISP2100_FABRIC
-static char *lretained =
+static const char lretained[] =
"Retained login of Target %d (Loop ID 0x%x) Port 0x%x";
-static char *plogout =
+static const char plogout[] =
"Logging out Target %d at Loop ID 0x%x (Port 0x%x)";
-static char *plogierr =
+static const char plogierr[] =
"Command Error in PLOGI for Port 0x%x (0x%x)";
-static char *nopdb =
+static const char nopdb[] =
"Could not get PDB for Device @ Port 0x%x";
-static char *pdbmfail1 =
+static const char pdbmfail1[] =
"PDB Loop ID info for Device @ Port 0x%x does not match up (0x%x)";
-static char *pdbmfail2 =
+static const char pdbmfail2[] =
"PDB Port info for Device @ Port 0x%x does not match up (0x%x)";
-static char *ldumped =
+static const char ldumped[] =
"Target %d (Loop ID 0x%x) Port 0x%x dumped after login info mismatch";
#endif
-static char *notresp =
+static const char notresp[] =
"Not RESPONSE in RESPONSE Queue (type 0x%x) @ idx %d (next %d) nlooked %d";
-static char *xact1 =
+static const char xact1[] =
"HBA attempted queued transaction with disconnect not set for %d.%d.%d";
-static char *xact2 =
+static const char xact2[] =
"HBA attempted queued transaction to target routine %d on target %d bus %d";
-static char *xact3 =
+static const char xact3[] =
"HBA attempted queued cmd for %d.%d.%d when queueing disabled";
-static char *pskip =
+static const char pskip[] =
"SCSI phase skipped for target %d.%d.%d";
-static char *topology =
+static const char topology[] =
"Loop ID %d, AL_PA 0x%x, Port ID 0x%x, Loop State 0x%x, Topology '%s'";
-static char *finmsg =
+static const char finmsg[] =
"(%d.%d.%d): FIN dl%d resid%d STS 0x%x SKEY %c XS_ERR=0x%x";
/*
* Local function prototypes.
@@ -123,6 +123,7 @@ static int isp_same_lportdb __P((struct lportdb *, struct lportdb *));
static int isp_pdb_sync __P((struct ispsoftc *, int));
#ifdef ISP2100_FABRIC
static int isp_scan_fabric __P((struct ispsoftc *));
+static void isp_register_fc4_type __P((struct ispsoftc *));
#endif
static void isp_fw_state __P((struct ispsoftc *));
static void isp_mboxcmd __P((struct ispsoftc *, mbreg_t *, int));
@@ -150,7 +151,7 @@ isp_reset(isp)
{
mbreg_t mbs;
int loops, i, touched, dodnld = 1;
- char *revname;
+ char *revname = "????";
isp->isp_state = ISP_NILSTATE;
@@ -186,7 +187,7 @@ isp_reset(isp)
*/
ISP_WRITE(isp, HCCR, HCCR_CMD_RELEASE);
mbs.param[0] = MBOX_ABOUT_FIRMWARE;
- isp_mboxcmd(isp, &mbs, MBLOGNONE);
+ isp_mboxcmd(isp, &mbs, MBOX_COMMAND_ERROR);
/*
* This *shouldn't* fail.....
*/
@@ -207,13 +208,12 @@ isp_reset(isp)
ISP_WRITE(isp, HCCR, HCCR_CMD_PAUSE);
if (IS_FC(isp)) {
- revname = "2X00";
switch (isp->isp_type) {
case ISP_HA_FC_2100:
- revname[1] = '1';
+ revname = "2100";
break;
case ISP_HA_FC_2200:
- revname[1] = '2';
+ revname = "2200";
break;
default:
break;
@@ -229,7 +229,7 @@ isp_reset(isp)
* XXX: Should probably do some bus sensing.
*/
} else if (IS_ULTRA2(isp)) {
- static char *m = "bus %d is in %s Mode";
+ static const char m[] = "bus %d is in %s Mode";
u_int16_t l;
sdparam *sdp = isp->isp_param;
@@ -729,10 +729,12 @@ isp_init(isp)
if (IS_DUALBUS(isp)) {
isp_setdfltparm(isp, 1);
}
- if (IS_FC(isp)) {
- isp_fibre_init(isp);
- } else {
- isp_scsi_init(isp);
+ if ((isp->isp_confopts & ISP_CFG_NOINIT) == 0) {
+ if (IS_FC(isp)) {
+ isp_fibre_init(isp);
+ } else {
+ isp_scsi_init(isp);
+ }
}
}
@@ -1288,16 +1290,16 @@ isp_fclink_test(isp, usdelay)
*/
enano = NANOTIME_SUB(&hrb, &hra);
+ isp_prt(isp, ISP_LOGDEBUG3, "usec%d: 0x%lx->0x%lx enano %lu",
+ count, (long) GET_NANOSEC(&hra), (long) GET_NANOSEC(&hrb),
+ (enano > ((u_int64_t)0xffffffff))? 0xffffffff :
+ (unsigned long) (enano & 0xffffffff));
+
/*
* If the elapsed time is less than 1 millisecond,
* delay a period of time up to that millisecond of
* waiting.
- */
- isp_prt(isp, ISP_LOGDEBUG3, "usec%d: 0x%lx->0x%lx enano %u",
- count, (long) GET_NANOSEC(&hra), (long) GET_NANOSEC(&hrb),
- enano);
-
- /*
+ *
* This peculiar code is an attempt to try and avoid
* invoking u_int64_t math support functions for some
* platforms where linkage is a problem.
@@ -1385,19 +1387,12 @@ isp_fclink_test(isp, usdelay)
lp->portid = BITS2WORD(pdb.pdb_portid_bits);
lp->loopid = pdb.pdb_loopid;
lp->loggedin = lp->valid = 1;
-#if 0
- if (isp->isp_rfabric == 0) {
- isp_i_register_fc4_type(isp);
- }
-#endif
+ isp_register_fc4_type(isp);
} else
#endif
{
fcp->isp_portid = mbs.param[2];
fcp->isp_onfabric = 0;
-#if 0
- isp->isp_rfabric = 0;
-#endif
fcp->portdb[FL_PORT_ID].valid = 0;
}
@@ -1454,7 +1449,7 @@ isp_pdb_sync(isp, target)
struct lportdb *lp, *tport;
fcparam *fcp = isp->isp_param;
isp_pdb_t pdb;
- int loopid, prange, lim;
+ int loopid, frange, prange, lim;
#ifdef ISP2100_FABRIC
/*
@@ -1472,12 +1467,13 @@ isp_pdb_sync(isp, target)
switch (fcp->isp_topo) {
case TOPO_F_PORT:
case TOPO_PTP_STUB:
- prange = 0;
+ frange = prange = 0;
break;
case TOPO_N_PORT:
prange = 2;
break;
default:
+ frange = FC_SNS_ID+1;
prange = FL_PORT_ID;
break;
}
@@ -1712,7 +1708,7 @@ isp_pdb_sync(isp, target)
/*
* Tell the outside world we've arrived.
*/
- (void) isp_async(isp, ISPASYNC_PDB_CHANGED, &i);
+ (void) isp_async(isp, ISPASYNC_LOGGED_INOUT, &i);
}
/*
@@ -1727,7 +1723,7 @@ isp_pdb_sync(isp, target)
* Tell the outside world we've gone away.
*/
loopid = lp - fcp->portdb;
- (void) isp_async(isp, ISPASYNC_PDB_CHANGED, &loopid);
+ (void) isp_async(isp, ISPASYNC_LOGGED_INOUT, &loopid);
MEMZERO((void *) lp, sizeof (*lp));
}
@@ -1735,27 +1731,39 @@ isp_pdb_sync(isp, target)
/*
* Now log in any fabric devices
*/
- for (lp = &fcp->portdb[FC_SNS_ID+1];
+ for (lp = &fcp->portdb[frange];
lp < &fcp->portdb[MAX_FC_TARG]; lp++) {
u_int32_t portid;
mbreg_t mbs;
+ loopid = lp - fcp->portdb;
+ if (loopid >= FL_PORT_ID && loopid <= FC_SNS_ID) {
+ continue;
+ }
+
/*
* Anything here?
*/
- if (lp->port_wwn == 0)
+ if (lp->port_wwn == 0) {
continue;
+ }
/*
* Don't try to log into yourself.
*/
- if ((portid = lp->portid) == fcp->isp_portid)
+ if ((portid = lp->portid) == fcp->isp_portid) {
continue;
+ }
/*
* If we'd been logged in- see if we still are and we haven't
* changed. If so, no need to log ourselves out, etc..
+ *
+ * Unfortunately, our charming Qlogic f/w has decided to
+ * return a valid port database entry for a fabric device
+ * that has, in fact, gone away. And it hangs trying to
+ * log it out.
*/
if (lp->loggedin &&
isp_getpdb(isp, lp->loopid, &pdb) == 0) {
@@ -1916,7 +1924,7 @@ isp_pdb_sync(isp, target)
if (lp->node_wwn && lp->port_wwn) {
lp->valid = 1;
loopid = lp - fcp->portdb;
- (void) isp_async(isp, ISPASYNC_PDB_CHANGED, &loopid);
+ (void) isp_async(isp, ISPASYNC_LOGGED_INOUT, &loopid);
continue;
}
dump_em:
@@ -1996,6 +2004,39 @@ isp_scan_fabric(isp)
*/
return (0);
}
+
+static void
+isp_register_fc4_type(struct ispsoftc *isp)
+{
+ fcparam *fcp = isp->isp_param;
+ sns_screq_t *reqp;
+ mbreg_t mbs;
+
+ reqp = (sns_screq_t *) fcp->isp_scratch;
+ MEMZERO((void *) reqp, SNS_RFT_REQ_SIZE);
+ reqp->snscb_rblen = SNS_RFT_RESP_SIZE >> 1;
+ reqp->snscb_addr[RQRSP_ADDR0015] = DMA_LSW(fcp->isp_scdma + 0x100);
+ reqp->snscb_addr[RQRSP_ADDR1631] = DMA_MSW(fcp->isp_scdma + 0x100);
+ reqp->snscb_sblen = 22;
+ reqp->snscb_data[0] = SNS_RFT;
+ reqp->snscb_data[4] = fcp->isp_portid & 0xffff;
+ reqp->snscb_data[5] = (fcp->isp_portid >> 16) & 0xff;
+ reqp->snscb_data[6] = 0x100; /* SCS - FCP */
+#if 0
+ reqp->snscb_data[6] |= 20; /* ISO/IEC 8802-2 LLC/SNAP */
+#endif
+ ISP_SWIZZLE_SNS_REQ(isp, reqp);
+ mbs.param[0] = MBOX_SEND_SNS;
+ mbs.param[1] = SNS_RFT_REQ_SIZE >> 1;
+ mbs.param[2] = DMA_MSW(fcp->isp_scdma);
+ mbs.param[3] = DMA_LSW(fcp->isp_scdma);
+ mbs.param[6] = 0;
+ mbs.param[7] = 0;
+ isp_mboxcmd(isp, &mbs, MBLOGALL);
+ if (mbs.param[0] == MBOX_COMMAND_COMPLETE) {
+ isp_prt(isp, ISP_LOGINFO, "Register FC4 types succeeded");
+ }
+}
#endif
/*
* Start a command. Locking is assumed done in the caller.
@@ -2051,9 +2092,13 @@ isp_start(xs)
#if defined(ISP2100_FABRIC)
/*
* If we're not on a Fabric, we can't have a target
- * above FL_PORT_ID-1. If we're on a fabric and
- * connected as an F-port, we can't have a target
- * less than FC_SNS_ID+1.
+ * above FL_PORT_ID-1.
+ *
+ * If we're on a fabric and *not* connected as an F-port,
+ * we can't have a target less than FC_SNS_ID+1. This
+ * keeps us from having to sort out the difference between
+ * local public loop devices and those which we might get
+ * from a switch's database.
*/
if (fcp->isp_onfabric == 0) {
if (target >= FL_PORT_ID) {
@@ -2065,7 +2110,7 @@ isp_start(xs)
XS_SETERR(xs, HBA_SELTIMEOUT);
return (CMD_COMPLETE);
}
- if (fcp->isp_topo == TOPO_F_PORT &&
+ if (fcp->isp_topo != TOPO_F_PORT &&
target < FL_PORT_ID) {
XS_SETERR(xs, HBA_SELTIMEOUT);
return (CMD_COMPLETE);
@@ -2442,10 +2487,11 @@ isp_control(isp, ctl, arg)
break;
}
isp_init(isp);
- if (isp->isp_state != ISP_INITSTATE) {
- break;
+ if ((isp->isp_confopts & ISP_CFG_NOINIT) == 0) {
+ if (isp->isp_state == ISP_INITSTATE) {
+ isp->isp_state = ISP_RUNSTATE;
+ }
}
- isp->isp_state = ISP_RUNSTATE;
}
return (0);
}
@@ -2531,7 +2577,7 @@ isp_intr(arg)
int obits, i = 0;
if ((obits = isp->isp_mboxbsy) != 0) {
isp->isp_mboxtmp[i++] = mbox;
- for (i = 1; i < 8; i++) {
+ for (i = 1; i < MAX_MAILBOX; i++) {
if ((obits & (1 << i)) == 0) {
continue;
}
@@ -3036,16 +3082,16 @@ isp_parse_async(isp, mbox)
isp->isp_sendmarker = 1;
FCPARAM(isp)->isp_loopstate = LOOP_PDB_RCVD;
isp_mark_getpdb_all(isp);
- isp_prt(isp, ISP_LOGINFO, "Port Database Changed");
+ isp_async(isp, ISPASYNC_CHANGE_NOTIFY, (void *) 0);
break;
case ASYNC_CHANGE_NOTIFY:
- isp_mark_getpdb_all(isp);
/*
* Not correct, but it will force us to rescan the loop.
*/
FCPARAM(isp)->isp_loopstate = LOOP_PDB_RCVD;
- isp_async(isp, ISPASYNC_CHANGE_NOTIFY, NULL);
+ isp_mark_getpdb_all(isp);
+ isp_async(isp, ISPASYNC_CHANGE_NOTIFY, (void *) 1);
break;
case ASYNC_PTPMODE:
@@ -4263,6 +4309,8 @@ isp_setdfltparm(isp, channel)
if (IS_FC(isp)) {
fcparam *fcp = (fcparam *) isp->isp_param;
+ int nvfail;
+
fcp += channel;
if (fcp->isp_gotdparms) {
return;
@@ -4294,16 +4342,39 @@ isp_setdfltparm(isp, channel)
fcp->isp_fwoptions &= ~ICBOPT_EXTENDED;
/*
- * Now try and read NVRAM
+ * Now try and read NVRAM unless told to not do so.
+ * This will set fcparam's isp_nodewwn && isp_portwwn.
*/
- if ((isp->isp_confopts & (ISP_CFG_NONVRAM|ISP_CFG_OWNWWN)) ||
- (isp_read_nvram(isp))) {
- isp_prt(isp, ISP_LOGINFO,
- "Node WWN 0x%08x%08x, Port WWN 0x%08x%08x",
- (u_int32_t) (fcp->isp_nodewwn >> 32),
- (u_int32_t) (fcp->isp_nodewwn & 0xffffffff),
- (u_int32_t) (fcp->isp_portwwn >> 32),
- (u_int32_t) (fcp->isp_portwwn & 0xffffffff));
+ if ((isp->isp_confopts & ISP_CFG_NONVRAM) == 0) {
+ nvfail = isp_read_nvram(isp);
+ if (nvfail)
+ isp->isp_confopts |= ISP_CFG_NONVRAM;
+ } else {
+ nvfail = 1;
+ }
+ /*
+ * Set node && port to override platform set defaults
+ * unless the nvram read failed (or none was done),
+ * or the platform code wants to use what had been
+ * set in the defaults.
+ */
+ if (nvfail || (isp->isp_confopts & ISP_CFG_OWNWWN)) {
+ isp_prt(isp, ISP_LOGCONFIG,
+ "Using Node WWN 0x%08x%08x, Port WWN 0x%08x%08x",
+ (u_int32_t) (DEFAULT_NODEWWN(isp) >> 32),
+ (u_int32_t) (DEFAULT_NODEWWN(isp) & 0xffffffff),
+ (u_int32_t) (DEFAULT_PORTWWN(isp) >> 32),
+ (u_int32_t) (DEFAULT_PORTWWN(isp) & 0xffffffff));
+ isp->isp_confopts |= ISP_CFG_OWNWWN;
+ ISP_NODEWWN(isp) = DEFAULT_NODEWWN(isp);
+ ISP_PORTWWN(isp) = DEFAULT_PORTWWN(isp);
+ } else {
+ /*
+ * We always start out with values derived
+ * from NVRAM or our platform default.
+ */
+ ISP_NODEWWN(isp) = fcp->isp_nodewwn;
+ ISP_PORTWWN(isp) = fcp->isp_portwwn;
}
return;
}