diff options
Diffstat (limited to 'sys/arch/hp300/stand/common/scsi.c')
-rw-r--r-- | sys/arch/hp300/stand/common/scsi.c | 469 |
1 files changed, 469 insertions, 0 deletions
diff --git a/sys/arch/hp300/stand/common/scsi.c b/sys/arch/hp300/stand/common/scsi.c new file mode 100644 index 00000000000..986dc38777d --- /dev/null +++ b/sys/arch/hp300/stand/common/scsi.c @@ -0,0 +1,469 @@ +/* $OpenBSD: scsi.c,v 1.1 1997/07/14 08:14:28 downsj Exp $ */ +/* $NetBSD: scsi.c,v 1.7 1997/01/30 10:32:57 thorpej Exp $ */ + +/* + * This is reported to fix some odd failures when disklabeling + * SCSI disks in SYS_INST. + */ +#define SLOWSCSI + +/* + * Copyright (c) 1988 University of Utah. + * Copyright (c) 1990, 1993 + * The Regents of the University of California. All rights reserved. + * + * This code is derived from software contributed to Berkeley by + * Van Jacobson of Lawrence Berkeley Laboratory and the Systems + * Programming Group of the University of Utah Computer Science Department. + * + * 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 University of + * California, Berkeley and its contributors. + * 4. Neither the name of the University nor the names of its contributors + * may be used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE REGENTS 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 REGENTS 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. + * + * from: Utah $Hdr: scsi.c 1.3 90/01/27$ + * + * @(#)scsi.c 8.1 (Berkeley) 6/10/93 + */ + +/* + * SCSI bus driver for standalone programs. + */ + +#include <sys/param.h> +#include <sys/reboot.h> + +#define _IOCTL_ +#include <hp300/dev/scsireg.h> + +#include "device.h" +#include "scsivar.h" + +#include <lib/libsa/stand.h> + +#include "samachdep.h" + +struct scsi_softc scsi_softc[NSCSI]; + +void scsireset(); +int scsi_cmd_wait = 50000; /* use the "real" driver init_wait value */ +int scsi_data_wait = 50000; /* use the "real" driver init_wait value */ + +scsiinit() +{ + extern struct hp_hw sc_table[]; + register struct hp_hw *hw; + register struct scsi_softc *hs; + register int i, addr; + static int waitset = 0; + + i = 0; + for (hw = sc_table; i < NSCSI && hw < &sc_table[MAXCTLRS]; hw++) { + if (!HW_ISSCSI(hw)) + continue; + hs = &scsi_softc[i]; + hs->sc_addr = hw->hw_kva; + scsireset(i); + if (howto & RB_ASKNAME) + printf("scsi%d at sc%d\n", i, hw->hw_sc); + hw->hw_pa = (caddr_t) i; /* XXX for autoconfig */ + hs->sc_alive = 1; + i++; + } + /* + * Adjust the wait values + */ + if (!waitset) { + scsi_cmd_wait *= cpuspeed; + scsi_data_wait *= cpuspeed; + waitset = 1; + } +} + +scsialive(unit) + register int unit; +{ + if (unit >= NSCSI || scsi_softc[unit].sc_alive == 0) + return (0); + return (1); +} + +void +scsireset(unit) + register int unit; +{ + volatile register struct scsidevice *hd; + register struct scsi_softc *hs; + u_int i; + + hs = &scsi_softc[unit]; + hd = (struct scsidevice *)hs->sc_addr; + hd->scsi_id = 0xFF; + DELAY(100); + /* + * Disable interrupts then reset the FUJI chip. + */ + hd->scsi_csr = 0; + hd->scsi_sctl = SCTL_DISABLE | SCTL_CTRLRST; + hd->scsi_scmd = 0; + hd->scsi_tmod = 0; + hd->scsi_pctl = 0; + hd->scsi_temp = 0; + hd->scsi_tch = 0; + hd->scsi_tcm = 0; + hd->scsi_tcl = 0; + hd->scsi_ints = 0; + + /* + * Configure the FUJI chip with its SCSI address, all + * interrupts enabled & appropriate parity. + */ + i = (~hd->scsi_hconf) & 0x7; + hs->sc_scsi_addr = 1 << i; + hd->scsi_bdid = i; + if (hd->scsi_hconf & HCONF_PARITY) + hd->scsi_sctl = SCTL_DISABLE | SCTL_ABRT_ENAB | + SCTL_SEL_ENAB | SCTL_RESEL_ENAB | + SCTL_INTR_ENAB | SCTL_PARITY_ENAB; + else + hd->scsi_sctl = SCTL_DISABLE | SCTL_ABRT_ENAB | + SCTL_SEL_ENAB | SCTL_RESEL_ENAB | + SCTL_INTR_ENAB; + hd->scsi_sctl &=~ SCTL_DISABLE; +} + + +int +scsiabort(hs, hd) + register struct scsi_softc *hs; + volatile register struct scsidevice *hd; +{ + printf("scsi%d error: scsiabort\n", hs - scsi_softc); + + scsireset(hs - scsi_softc); + DELAY(1000000); +} + +static int +issue_select(hd, target, our_addr) + volatile register struct scsidevice *hd; + u_char target, our_addr; +{ + if (hd->scsi_ssts & (SSTS_INITIATOR|SSTS_TARGET|SSTS_BUSY)) + return (1); + + if (hd->scsi_ints & INTS_DISCON) + hd->scsi_ints = INTS_DISCON; + + hd->scsi_pctl = 0; + hd->scsi_temp = (1 << target) | our_addr; + /* select timeout is hardcoded to 2ms */ + hd->scsi_tch = 0; + hd->scsi_tcm = 32; + hd->scsi_tcl = 4; + + hd->scsi_scmd = SCMD_SELECT; + return (0); +} + +static int +wait_for_select(hd) + volatile register struct scsidevice *hd; +{ + register int wait; + u_char ints; + + wait = scsi_data_wait; + while ((ints = hd->scsi_ints) == 0) { + if (--wait < 0) + return (1); + DELAY(1); + } + hd->scsi_ints = ints; + return (!(hd->scsi_ssts & SSTS_INITIATOR)); +} + +static int +ixfer_start(hd, len, phase, wait) + volatile register struct scsidevice *hd; + int len; + u_char phase; + register int wait; +{ + + hd->scsi_tch = len >> 16; + hd->scsi_tcm = len >> 8; + hd->scsi_tcl = len; + hd->scsi_pctl = phase; + hd->scsi_tmod = 0; /*XXX*/ + hd->scsi_scmd = SCMD_XFR | SCMD_PROG_XFR; + + /* wait for xfer to start or svc_req interrupt */ + while ((hd->scsi_ssts & SSTS_BUSY) == 0) { + if (hd->scsi_ints || --wait < 0) + return (0); + DELAY(1); + } + return (1); +} + +static int +ixfer_out(hd, len, buf) + volatile register struct scsidevice *hd; + int len; + register u_char *buf; +{ + register int wait = scsi_data_wait; + + for (; len > 0; --len) { + while (hd->scsi_ssts & SSTS_DREG_FULL) { + if (hd->scsi_ints || --wait < 0) + return (len); + DELAY(1); + } + hd->scsi_dreg = *buf++; + } + return (0); +} + +static int +ixfer_in(hd, len, buf) + volatile register struct scsidevice *hd; + int len; + register u_char *buf; +{ + register int wait = scsi_data_wait; + + for (; len > 0; --len) { + while (hd->scsi_ssts & SSTS_DREG_EMPTY) { + if (hd->scsi_ints || --wait < 0) { + while (! (hd->scsi_ssts & SSTS_DREG_EMPTY)) { + *buf++ = hd->scsi_dreg; + --len; + } + return (len); + } + DELAY(1); + } + *buf++ = hd->scsi_dreg; + } + return (len); +} + +static int +scsiicmd(hs, target, cbuf, clen, buf, len, xferphase) + struct scsi_softc *hs; + int target; + u_char *cbuf; + int clen; + u_char *buf; + int len; + u_char xferphase; +{ + volatile register struct scsidevice *hd = + (struct scsidevice *)hs->sc_addr; + u_char phase, ints; + register int wait; + + /* select the SCSI bus (it's an error if bus isn't free) */ + if (issue_select(hd, target, hs->sc_scsi_addr)) + return (-2); + if (wait_for_select(hd)) + return (-2); + /* + * Wait for a phase change (or error) then let the device + * sequence us through the various SCSI phases. + */ + hs->sc_stat = -1; + phase = CMD_PHASE; + while (1) { + wait = scsi_cmd_wait; + switch (phase) { + + case CMD_PHASE: + if (ixfer_start(hd, clen, phase, wait)) + if (ixfer_out(hd, clen, cbuf)) + goto abort; + phase = xferphase; + break; + + case DATA_IN_PHASE: + if (len <= 0) + goto abort; + wait = scsi_data_wait; + if (ixfer_start(hd, len, phase, wait) || + !(hd->scsi_ssts & SSTS_DREG_EMPTY)) + ixfer_in(hd, len, buf); + phase = STATUS_PHASE; + break; + + case DATA_OUT_PHASE: + if (len <= 0) + goto abort; + wait = scsi_data_wait; + if (ixfer_start(hd, len, phase, wait)) + if (ixfer_out(hd, len, buf)) + goto abort; + phase = STATUS_PHASE; + break; + + case STATUS_PHASE: + wait = scsi_data_wait; + if (ixfer_start(hd, sizeof(hs->sc_stat), phase, wait) || + !(hd->scsi_ssts & SSTS_DREG_EMPTY)) + ixfer_in(hd, sizeof(hs->sc_stat), &hs->sc_stat); + phase = MESG_IN_PHASE; + break; + + case MESG_IN_PHASE: + if (ixfer_start(hd, sizeof(hs->sc_msg), phase, wait) || + !(hd->scsi_ssts & SSTS_DREG_EMPTY)) { + ixfer_in(hd, sizeof(hs->sc_msg), &hs->sc_msg); + hd->scsi_scmd = SCMD_RST_ACK; + } + phase = BUS_FREE_PHASE; + break; + + case BUS_FREE_PHASE: + goto out; + + default: + printf("scsi%d: unexpected scsi phase %d\n", + hs - scsi_softc, phase); + goto abort; + } +#ifdef SLOWSCSI + /* + * XXX we have wierd transient problems with booting from + * slow scsi disks on fast machines. I have never been + * able to pin the problem down, but a large delay here + * seems to always work. + */ + DELAY(1000); +#endif + /* wait for last command to complete */ + while ((ints = hd->scsi_ints) == 0) { + if (--wait < 0) + goto abort; + DELAY(1); + } + hd->scsi_ints = ints; + if (ints & INTS_SRV_REQ) + phase = hd->scsi_psns & PHASE; + else if (ints & INTS_DISCON) + goto out; + else if ((ints & INTS_CMD_DONE) == 0) + goto abort; + } +abort: + scsiabort(hs, hd); +out: + return (hs->sc_stat); +} + +int +scsi_test_unit_rdy(ctlr, slave) + int ctlr, slave; +{ + register struct scsi_softc *hs = &scsi_softc[ctlr]; + static struct scsi_cdb6 cdb = { CMD_TEST_UNIT_READY }; + + return (scsiicmd(hs, slave, &cdb, sizeof(cdb), (u_char *)0, 0, + STATUS_PHASE)); +} + +int +scsi_request_sense(ctlr, slave, buf, len) + int ctlr, slave; + u_char *buf; + unsigned len; +{ + register struct scsi_softc *hs = &scsi_softc[ctlr]; + static struct scsi_cdb6 cdb = { CMD_REQUEST_SENSE }; + + cdb.len = len; + return (scsiicmd(hs, slave, &cdb, sizeof(cdb), buf, len, + DATA_IN_PHASE)); +} + +int +scsi_read_capacity(ctlr, slave, buf, len) + int ctlr, slave; + u_char *buf; + unsigned len; +{ + register struct scsi_softc *hs = &scsi_softc[ctlr]; + static struct scsi_cdb10 cdb = { CMD_READ_CAPACITY }; + + return (scsiicmd(hs, slave, &cdb, sizeof(cdb), buf, len, + DATA_IN_PHASE)); +} + +int +scsi_tt_read(ctlr, slave, buf, len, blk, nblk) + int ctlr, slave; + u_char *buf; + u_int len; + daddr_t blk; + u_int nblk; +{ + register struct scsi_softc *hs = &scsi_softc[ctlr]; + struct scsi_cdb10 cdb; + + bzero(&cdb, sizeof(cdb)); + cdb.cmd = CMD_READ_EXT; + cdb.lbah = blk >> 24; + cdb.lbahm = blk >> 16; + cdb.lbalm = blk >> 8; + cdb.lbal = blk; + cdb.lenh = nblk >> (8 + DEV_BSHIFT); + cdb.lenl = nblk >> DEV_BSHIFT; + return (scsiicmd(hs, slave, &cdb, sizeof(cdb), buf, len, + DATA_IN_PHASE)); +} + +int +scsi_tt_write(ctlr, slave, buf, len, blk, nblk) + int ctlr, slave; + u_char *buf; + u_int len; + daddr_t blk; + u_int nblk; +{ + register struct scsi_softc *hs = &scsi_softc[ctlr]; + struct scsi_cdb10 cdb; + + bzero(&cdb, sizeof(cdb)); + cdb.cmd = CMD_WRITE_EXT; + cdb.lbah = blk >> 24; + cdb.lbahm = blk >> 16; + cdb.lbalm = blk >> 8; + cdb.lbal = blk; + cdb.lenh = nblk >> (8 + DEV_BSHIFT); + cdb.lenl = nblk >> DEV_BSHIFT; + return (scsiicmd(hs, slave, &cdb, sizeof(cdb), buf, len, + DATA_OUT_PHASE)); +} |