diff options
author | Theo de Raadt <deraadt@cvs.openbsd.org> | 1995-10-18 08:53:40 +0000 |
---|---|---|
committer | Theo de Raadt <deraadt@cvs.openbsd.org> | 1995-10-18 08:53:40 +0000 |
commit | d6583bb2a13f329cf0332ef2570eb8bb8fc0e39c (patch) | |
tree | ece253b876159b39c620e62b6c9b1174642e070e /sys/dev/ic/ncr5380.c |
initial import of NetBSD tree
Diffstat (limited to 'sys/dev/ic/ncr5380.c')
-rw-r--r-- | sys/dev/ic/ncr5380.c | 729 |
1 files changed, 729 insertions, 0 deletions
diff --git a/sys/dev/ic/ncr5380.c b/sys/dev/ic/ncr5380.c new file mode 100644 index 00000000000..bbc68d483f4 --- /dev/null +++ b/sys/dev/ic/ncr5380.c @@ -0,0 +1,729 @@ +/* $NetBSD: ncr5380.c,v 1.3 1995/09/26 21:04:27 pk Exp $ */ + +/* + * Copyright (C) 1994 Adam Glass, Gordon W. Ross + * Copyright (C) 1993 Allen K. Briggs, Chris P. Caputo, + * Michael L. Finch, Bradley A. Grantham, and + * Lawrence A. Kesteloot + * 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, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * 3. All advertising materials mentioning features or use of this software + * must display the following acknowledgement: + * This product includes software developed by the Alice Group. + * 4. The names of the Alice Group or any of its members may not be used + * to endorse or promote products derived from this software without + * specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE ALICE GROUP ``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 ALICE GROUP 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. + */ + +/* + * NOTE: This is 99% Sun 3 `si' driver. I've made the probe and attach + * routines a little minimalistic, and adjusted them slightly for the Sun 4. + * Also, fixed some logic in ncr5380_select_target(). + * Jason R. Thorpe <thorpej@nas.nasa.gov> + */ + +static int +ncr5380_scsi_cmd(xs) + struct scsi_xfer *xs; +{ + int flags, s, r; + + flags = xs->flags; + if (xs->bp) flags |= (SCSI_NOSLEEP); + if ( flags & ITSDONE ) { + printf("Already done?"); + xs->flags &= ~ITSDONE; + } + if ( ! ( flags & INUSE ) ) { + printf("Not in use?"); + xs->flags |= INUSE; + } + + s = splbio(); + + if ( flags & SCSI_RESET ) { + printf("flags & SCSIRESET.\n"); + ncr5380_reset_scsibus(xs->sc_link->adapter_softc); + r = COMPLETE; + } else { + r = ncr5380_send_cmd(xs); + xs->flags |= ITSDONE; + scsi_done(xs); + } + + splx(s); + + switch(r) { + case COMPLETE: + case SUCCESSFULLY_QUEUED: + r = SUCCESSFULLY_QUEUED; + if (xs->flags & SCSI_POLL) + r = COMPLETE; + break; + default: + break; + } + return r; +} + +#ifdef DEBUG +static int +ncr5380_show_scsi_cmd(xs) + struct scsi_xfer *xs; +{ + u_char *b = (u_char *) xs->cmd; + int i = 0; + + if ( ! ( xs->flags & SCSI_RESET ) ) { + printf("si(%d:%d:%d)-", + xs->sc_link->scsibus, + xs->sc_link->target, + xs->sc_link->lun); + while (i < xs->cmdlen) { + if (i) printf(","); + printf("%x",b[i++]); + } + printf("-\n"); + } else { + printf("si(%d:%d:%d)-RESET-\n", + xs->sc_link->scsibus, + xs->sc_link->target, + xs->sc_link->lun); + } +} +#endif + +/* + * Actual chip control. + */ + +static void +ncr5380_sbc_intr(ncr5380) + struct ncr5380_softc *ncr5380; +{ + volatile sci_regmap_t *regs = ncr5380->sc_regs; + + if ((regs->sci_csr & SCI_CSR_INT) == 0) { +#ifdef DEBUG + printf (" ncr5380_sbc_intr: spurious\n"); +#endif + return; + } + + SCI_CLR_INTR(regs); +#ifdef DEBUG + printf (" ncr5380_sbc_intr\n"); +#endif +} + +static int +ncr5380_reset_scsibus(ncr5380) + struct ncr5380_softc *ncr5380; +{ + volatile sci_regmap_t *regs = ncr5380->sc_regs; + +#ifdef DEBUG + if (ncr5380_debug) { + printf("ncr5380_reset_scsibus\n"); + } +#endif + + regs->sci_icmd = SCI_ICMD_RST; + delay(100); + regs->sci_icmd = 0; + + regs->sci_mode = 0; + regs->sci_tcmd = SCI_PHASE_DISC; + regs->sci_sel_enb = 0; + + SCI_CLR_INTR(regs); + /* XXX - Need long delay here! */ +} + +static int +ncr5380_poll(adapter, timeout) + int adapter, timeout; +{ +} + +static int +ncr5380_send_cmd(xs) + struct scsi_xfer *xs; +{ + int sense; + +#ifdef DEBUG + if (ncr5380_debug & 2) + ncr5380_show_scsi_cmd(xs); +#endif + + sense = ncr5380_generic( xs->sc_link->adapter_softc, + xs->sc_link->target, xs->sc_link->lun, xs->cmd, + xs->cmdlen, xs->data, xs->datalen ); + + switch (sense) { + case 0: /* success */ + xs->resid = 0; + xs->error = XS_NOERROR; + break; + + case 0x02: /* Check condition */ +#ifdef DEBUG + if (ncr5380_debug) + printf("check cond. target %d.\n", + xs->sc_link->target); +#endif + delay(10); /* Phil's fix for slow devices. */ + ncr5380_group0(xs->sc_link->adapter_softc, + xs->sc_link->target, + xs->sc_link->lun, + 0x3, 0x0, + sizeof(struct scsi_sense_data), + 0, (caddr_t) &(xs->sense), + sizeof(struct scsi_sense_data)); + xs->error = XS_SENSE; + break; + case 0x08: /* Busy - common code will delay, retry. */ + xs->error = XS_BUSY; + break; + default: /* Dead - tell common code to give up. */ + xs->error = XS_DRIVER_STUFFUP; + break; + + } + return (COMPLETE); +} + +static int +ncr5380_select_target(regs, myid, tid, with_atn) + register volatile sci_regmap_t *regs; + u_char myid, tid; + int with_atn; +{ + register u_char bid, icmd; + int ret = SCSI_RET_RETRY; + int arb_retries, arb_wait; + int i; + + /* for our purposes.. */ + myid = 1 << myid; + tid = 1 << tid; + + regs->sci_sel_enb = 0; /* we don't want any interrupts. */ + regs->sci_tcmd = 0; /* get into a harmless state */ + + arb_retries = ARBITRATION_RETRIES; + +retry_arbitration: + regs->sci_mode = 0; /* get into a harmless state */ +wait_for_bus_free: + if (--arb_retries <= 0) { +#ifdef DEBUG + if (ncr5380_debug) { + printf("ncr5380_select: arb_retries expended; resetting...\n"); + } +#endif + ret = SCSI_RET_NEED_RESET; + goto nosel; + } + + icmd = regs->sci_icmd & ~(SCI_ICMD_DIFF|SCI_ICMD_TEST); + + if (regs->sci_bus_csr & (SCI_BUS_BSY|SCI_BUS_SEL)) { + /* Something is sitting on the SCSI bus... */ +#ifdef DEBUG + /* Only complain once (the last time through). */ + if (ncr5380_debug && (arb_retries <= 1)) { + printf("si_select_target: still BSY+SEL\n"); + } +#endif + /* Give it a little time, then try again. */ + delay(10); + goto wait_for_bus_free; + } + + regs->sci_odata = myid; + regs->sci_mode = SCI_MODE_ARB; +/* regs->sci_mode |= SCI_MODE_ARB; XXX? */ + + /* AIP might not set if BSY went true after we checked */ + /* Wait up to about 100 usec. for it to appear. */ + arb_wait = 50; /* X2 */ + do { + if (regs->sci_icmd & SCI_ICMD_AIP) + goto got_aip; + delay(2); + } while (--arb_wait > 0); + /* XXX - Could have missed it? */ +#ifdef DEBUG + if (ncr5380_debug) + printf("ncr5380_select_target: API did not appear\n"); +#endif + goto retry_arbitration; + + got_aip: +#ifdef DEBUG + if (ncr5380_debug & 4) { + printf("ncr5380_select_target: API after %d tries (last wait %d)\n", + ARBITRATION_RETRIES - arb_retries, + (50 - arb_wait)); + } +#endif + + delay(3); /* 2.2 uSec. arbitration delay */ + + if (regs->sci_icmd & SCI_ICMD_LST) { +#ifdef DEBUG + if (ncr5380_debug) + printf ("lost 1\n"); +#endif + goto retry_arbitration; /* XXX */ + } + + regs->sci_mode &= ~SCI_MODE_PAR_CHK; + bid = regs->sci_data; + + if ((bid & ~myid) > myid) { +#ifdef DEBUG + if (ncr5380_debug) + printf ("lost 2\n"); +#endif + /* Trying again will not help. */ + goto lost; + } + if (regs->sci_icmd & SCI_ICMD_LST) { +#ifdef DEBUG + if (ncr5380_debug) + printf ("lost 3\n"); +#endif + goto lost; + } + + /* Won arbitration, enter selection phase now */ + icmd = regs->sci_icmd & ~(SCI_ICMD_DIFF|SCI_ICMD_TEST); + icmd |= (with_atn ? (SCI_ICMD_SEL|SCI_ICMD_ATN) : SCI_ICMD_SEL); + regs->sci_icmd = icmd; + + if (regs->sci_icmd & SCI_ICMD_LST) { +#ifdef DEBUG + if (ncr5380_debug) + printf ("nosel\n"); +#endif + goto nosel; + } + + /* XXX a target that violates specs might still drive the bus XXX */ + /* XXX should put our id out, and after the delay check nothi XXX */ + /* XXX ng else is out there. XXX */ + + delay(3); + + regs->sci_sel_enb = 0; + + regs->sci_odata = myid | tid; + + icmd |= SCI_ICMD_BSY|SCI_ICMD_DATA; + regs->sci_icmd = icmd; + +/* regs->sci_mode &= ~SCI_MODE_ARB; 2 deskew delays, too */ + regs->sci_mode = 0; /* 2 deskew delays, too */ + + icmd &= ~SCI_ICMD_BSY; + regs->sci_icmd = icmd; + + /* bus settle delay, 400ns */ + delay(3); + + regs->sci_mode |= SCI_MODE_PAR_CHK; + + { + register int timeo = 2500;/* 250 msecs in 100 usecs chunks */ + while ((regs->sci_bus_csr & SCI_BUS_BSY) == 0) { + if (--timeo > 0) { + delay(100); + } else { + /* This is the "normal" no-such-device select error. */ +#ifdef DEBUG + if (ncr5380_debug) + printf("ncr5380_select: not BSY (nothing there)\n"); +#endif + goto nodev; + } + } + } + + icmd &= ~(SCI_ICMD_DATA|SCI_ICMD_SEL); + regs->sci_icmd = icmd; +/* regs->sci_sel_enb = myid;*/ /* looks like we should NOT have it */ + /* XXX - SCI_MODE_PAR_CHK ? */ + return SCSI_RET_SUCCESS; + +nodev: + ret = SCSI_RET_DEVICE_DOWN; + regs->sci_sel_enb = myid; +nosel: + regs->sci_icmd = 0; + regs->sci_mode = 0; + return ret; + +lost: + regs->sci_icmd = 0; + regs->sci_mode = 0; +#ifdef DEBUG + if (ncr5380_debug) { + printf("ncr5380_select: lost arbitration\n"); + } +#endif + return ret; +} + +sci_data_out(regs, phase, count, data) + register volatile sci_regmap_t *regs; + unsigned char *data; +{ + register unsigned char icmd; + register int cnt=0; + + /* ..checks.. */ + + icmd = regs->sci_icmd & ~(SCI_ICMD_DIFF|SCI_ICMD_TEST); +loop: + /* SCSI bus phase not valid until REQ is true. */ + WAIT_FOR_REQ(regs); + if (SCI_CUR_PHASE(regs->sci_bus_csr) != phase) + return cnt; + + icmd |= SCI_ICMD_DATA; + regs->sci_icmd = icmd; + regs->sci_odata = *data++; + icmd |= SCI_ICMD_ACK; + regs->sci_icmd = icmd; + + icmd &= ~(SCI_ICMD_DATA|SCI_ICMD_ACK); + WAIT_FOR_NOT_REQ(regs); + regs->sci_icmd = icmd; + ++cnt; + if (--count > 0) + goto loop; +scsi_timeout_error: + return cnt; +} + +static int +sci_data_in(regs, phase, count, data) + register volatile sci_regmap_t *regs; + unsigned char *data; +{ + register unsigned char icmd; + register int cnt=0; + + /* ..checks.. */ + + icmd = regs->sci_icmd & ~(SCI_ICMD_DIFF|SCI_ICMD_TEST); + +loop: + /* SCSI bus phase not valid until REQ is true. */ + WAIT_FOR_REQ(regs); + if (SCI_CUR_PHASE(regs->sci_bus_csr) != phase) + return cnt; + + *data++ = regs->sci_data; + icmd |= SCI_ICMD_ACK; + regs->sci_icmd = icmd; + + icmd &= ~SCI_ICMD_ACK; + WAIT_FOR_NOT_REQ(regs); + regs->sci_icmd = icmd; + ++cnt; + if (--count > 0) + goto loop; + +scsi_timeout_error: + return cnt; +} + +/* + * Return -1 (error) or number of bytes sent (>=0). + */ +static int +ncr5380_command_transfer(regs, maxlen, data, status, msg) + register volatile sci_regmap_t *regs; + int maxlen; + u_char *data; + u_char *status; + u_char *msg; +{ + int xfer, phase; + + xfer = 0; + regs->sci_icmd = 0; + + while (1) { + + WAIT_FOR_REQ(regs); + + phase = SCI_CUR_PHASE(regs->sci_bus_csr); + + switch (phase) { + case SCSI_PHASE_CMD: + SCI_ACK(regs,SCSI_PHASE_CMD); + xfer += sci_data_out(regs, SCSI_PHASE_CMD, + maxlen, data); + goto out; + + case SCSI_PHASE_DATA_IN: + printf("command_transfer: Data in phase?\n"); + goto err; + + case SCSI_PHASE_DATA_OUT: + printf("command_transfer: Data out phase?\n"); + goto err; + + case SCSI_PHASE_STATUS: + SCI_ACK(regs,SCSI_PHASE_STATUS); + printf("command_transfer: status in...\n"); + sci_data_in(regs, SCSI_PHASE_STATUS, + 1, status); + printf("command_transfer: status=0x%x\n", *status); + goto err; + + case SCSI_PHASE_MESSAGE_IN: + SCI_ACK(regs,SCSI_PHASE_MESSAGE_IN); + printf("command_transfer: msg in?\n"); + sci_data_in(regs, SCSI_PHASE_MESSAGE_IN, + 1, msg); + break; + + case SCSI_PHASE_MESSAGE_OUT: + SCI_ACK(regs,SCSI_PHASE_MESSAGE_OUT); + sci_data_out(regs, SCSI_PHASE_MESSAGE_OUT, + 1, msg); + break; + + default: + printf("command_transfer: Unexpected phase 0x%x\n", phase); + goto err; + } + } +scsi_timeout_error: +err: + xfer = -1; +out: + return xfer; +} + +static int +ncr5380_data_transfer(regs, maxlen, data, status, msg) + register volatile sci_regmap_t *regs; + int maxlen; + u_char *data, *status, *msg; +{ + int retlen = 0, xfer, phase; + + regs->sci_icmd = 0; + + *status = 0; + + while (1) { + + WAIT_FOR_REQ(regs); + + phase = SCI_CUR_PHASE(regs->sci_bus_csr); + + switch (phase) { + case SCSI_PHASE_CMD: + printf("Command phase in data_transfer().\n"); + return retlen; + case SCSI_PHASE_DATA_IN: + SCI_ACK(regs,SCSI_PHASE_DATA_IN); +#if PSEUDO_DMA + xfer = sci_pdma_in(regs, SCSI_PHASE_DATA_IN, + maxlen, data); +#else + xfer = sci_data_in(regs, SCSI_PHASE_DATA_IN, + maxlen, data); +#endif + retlen += xfer; + maxlen -= xfer; + break; + case SCSI_PHASE_DATA_OUT: + SCI_ACK(regs,SCSI_PHASE_DATA_OUT); +#if PSEUDO_DMA + xfer = sci_pdma_out(regs, SCSI_PHASE_DATA_OUT, + maxlen, data); +#else + xfer = sci_data_out(regs, SCSI_PHASE_DATA_OUT, + maxlen, data); +#endif + retlen += xfer; + maxlen -= xfer; + break; + case SCSI_PHASE_STATUS: + SCI_ACK(regs,SCSI_PHASE_STATUS); + sci_data_in(regs, SCSI_PHASE_STATUS, + 1, status); + break; + case SCSI_PHASE_MESSAGE_IN: + SCI_ACK(regs,SCSI_PHASE_MESSAGE_IN); + sci_data_in(regs, SCSI_PHASE_MESSAGE_IN, + 1, msg); + if (*msg == 0) { + return retlen; + } else { + printf( "message 0x%x in " + "data_transfer.\n", *msg); + } + break; + case SCSI_PHASE_MESSAGE_OUT: + SCI_ACK(regs,SCSI_PHASE_MESSAGE_OUT); + sci_data_out(regs, SCSI_PHASE_MESSAGE_OUT, + 1, msg); + break; + default: + printf( "Unexpected phase 0x%x in " + "data_transfer().\n", phase); +scsi_timeout_error: + return retlen; + break; + } + } +} + +/* + * Returns 0 on success, -1 on internal error, or the status byte + */ +static int +ncr5380_dorequest(sc, target, lun, cmd, cmdlen, databuf, datalen, sent) + struct ncr5380_softc *sc; + int target, lun; + u_char *cmd; + int cmdlen; + char *databuf; + int datalen, *sent; +{ + register volatile sci_regmap_t *regs = sc->sc_regs; + int cmd_bytes_sent, r; + u_char stat, msg, c; + +#ifdef DEBUG + if (ncr5380_debug) { + printf("ncr5380_dorequest: target=%d, lun=%d\n", target, lun); + } +#endif + + *sent = 0; + + if ( ( r = ncr5380_select_target(regs, 7, target, 1) ) != SCSI_RET_SUCCESS) { +#ifdef DEBUG + if (ncr5380_debug) { + printf("ncr5380_dorequest: select returned %d\n", r); + } +#endif + + SCI_CLR_INTR(regs); + switch (r) { + + case SCSI_RET_NEED_RESET: + printf("ncr5380_dorequest: target=%d, lun=%d, r=%d resetting...\n", + target, lun, r); + reset_adapter(sc); + ncr5380_reset_scsibus(sc); + /* fall through */ + case SCSI_RET_RETRY: + return 0x08; /* Busy - tell common code to retry. */ + + default: + printf("ncr5380_dorequest: target=%d, lun=%d, error=%d.\n", + target, lun, r); + /* fall through */ + case SCSI_RET_DEVICE_DOWN: + return -1; /* Dead - tell common code to give up. */ + } + } + + c = 0x80 | lun; + + if ((cmd_bytes_sent = ncr5380_command_transfer(regs, cmdlen, + (u_char *) cmd, &stat, &c)) != cmdlen) + { + SCI_CLR_INTR(regs); + if (cmd_bytes_sent >= 0) { + printf("Data underrun sending CCB (%d bytes of %d, sent).\n", + cmd_bytes_sent, cmdlen); + } + return -1; + } + + *sent = ncr5380_data_transfer(regs, datalen, (u_char *)databuf, + &stat, &msg); +#ifdef DEBUG + if (ncr5380_debug) { + printf("ncr5380_dorequest: data transfered = %d\n", *sent); + } +#endif + + return stat; +} + +static int +ncr5380_generic(adapter, id, lun, cmd, cmdlen, databuf, datalen) + void *adapter; + int id, lun; + struct scsi_generic *cmd; + int cmdlen; + void *databuf; + int datalen; +{ + register struct ncr5380_softc *sc = adapter; + int i, j, sent; + + if (cmd->opcode == TEST_UNIT_READY) /* XXX */ + cmd->bytes[0] = ((u_char) lun << 5); + + i = ncr5380_dorequest(sc, id, lun, (u_char *) cmd, cmdlen, + databuf, datalen, &sent); + + return i; +} + +static int +ncr5380_group0(adapter, id, lun, opcode, addr, len, flags, databuf, datalen) + void *adapter; + int id, lun, opcode, addr, len, flags; + caddr_t databuf; + int datalen; +{ + register struct ncr5380_softc *sc = adapter; + unsigned char cmd[6]; + int i, j, sent; + + cmd[0] = opcode; /* Operation code */ + cmd[1] = (lun << 5) | ((addr >> 16) & 0x1F); /* Lun & MSB of addr */ + cmd[2] = (addr >> 8) & 0xFF; /* addr */ + cmd[3] = addr & 0xFF; /* LSB of addr */ + cmd[4] = len; /* Allocation length */ + cmd[5] = flags; /* Link/Flag */ + + i = ncr5380_dorequest(sc, id, lun, cmd, 6, databuf, datalen, &sent); + + return i; +} |