diff options
Diffstat (limited to 'sys/arch/macppc/dev/mesh.c')
-rw-r--r-- | sys/arch/macppc/dev/mesh.c | 1181 |
1 files changed, 1181 insertions, 0 deletions
diff --git a/sys/arch/macppc/dev/mesh.c b/sys/arch/macppc/dev/mesh.c new file mode 100644 index 00000000000..3b08fdde4e0 --- /dev/null +++ b/sys/arch/macppc/dev/mesh.c @@ -0,0 +1,1181 @@ +/* $OpenBSD: mesh.c,v 1.1 2001/09/01 15:50:00 drahn Exp $ */ +/* $NetBSD: mesh.c,v 1.1 1999/02/19 13:06:03 tsubai Exp $ */ + +/*- + * Copyright (C) 1999 Internet Research Institute, Inc. + * 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 + * Internet Research Institute, Inc. + * 4. 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 ``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 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. + */ + +#include <sys/param.h> +#include <sys/buf.h> +#include <sys/device.h> +#include <sys/errno.h> +#include <sys/kernel.h> +#include <sys/malloc.h> +#include <sys/queue.h> +#include <sys/systm.h> + +#include <vm/vm.h> + + +#include <scsi/scsi_all.h> +#include <scsi/scsiconf.h> +#include <scsi/scsi_message.h> +/* +#include "scsipi/scsi_all.h" +#include "scsipi/scsipi_all.h" +#include "scsipi/scsiconf.h" +#include "scsipi/scsi_message.h"*/ + + + +#include <dev/ofw/openfirm.h> + +#include <machine/autoconf.h> +#include <machine/cpu.h> +#include <machine/pio.h> + +#include "dbdma.h" +#include "meshreg.h" + +#define T_SYNCMODE 0x01 /* target uses sync mode */ +#define T_SYNCNEGO 0x02 /* sync negotiation done */ + + + +struct mesh_tinfo { + int flags; + int period; + int offset; +}; + +/* scb flags */ +#define MESH_POLL 0x01 +#define MESH_CHECK 0x02 +#define MESH_SENSE 0x04 +#define MESH_READ 0x80 + +struct mesh_scb { + TAILQ_ENTRY(mesh_scb) chain; + int flags; + struct scsi_xfer *xs; + struct scsi_generic cmd; + int cmdlen; + int target; /* target SCSI ID */ + int resid; + vaddr_t daddr; + vsize_t dlen; + int status; +}; + +/* sc_flags value */ +#define MESH_DMA_ACTIVE 0x01 + +struct mesh_softc { + struct device sc_dev; /* us as a device */ + struct scsi_link sc_link; + struct scsi_adapter sc_adapter; + + u_char *sc_reg; /* MESH base address */ + dbdma_regmap_t *sc_dmareg; /* DMA register address */ + dbdma_command_t *sc_dmacmd; /* DMA command area */ + + int sc_flags; + int sc_cfflags; /* copy of config flags */ + int sc_meshid; /* MESH version */ + int sc_minsync; /* minimum sync period */ + int sc_irq; + int sc_freq; /* SCSI bus frequency in MHz */ + int sc_id; /* our SCSI ID */ + struct mesh_tinfo sc_tinfo[8]; /* target information */ + + int sc_nextstate; + int sc_prevphase; + struct mesh_scb *sc_nexus; /* current command */ + + int sc_msgout; + int sc_imsglen; + int sc_omsglen; + u_char sc_imsg[16]; + u_char sc_omsg[16]; + + TAILQ_HEAD(, mesh_scb) free_scb; + TAILQ_HEAD(, mesh_scb) ready_scb; + struct mesh_scb sc_scb[16]; + + struct timeout sc_tmo; +}; + +/* mesh_msgout() values */ +#define SEND_REJECT 1 +#define SEND_IDENTIFY 2 +#define SEND_SDTR 4 + + +#ifdef __OpenBSD__ +#define scsi_print_addr sc_print_addr +#define scsipi_done scsi_done +#endif + +static __inline int mesh_read_reg __P((struct mesh_softc *, int)); +static __inline void mesh_set_reg __P((struct mesh_softc *, int, int)); + +int mesh_match __P((struct device *, struct cfdata *, void *)); +void mesh_attach __P((struct device *, struct device *, void *)); +void mesh_shutdownhook __P((void *)); +int mesh_intr __P((void *)); +void mesh_error __P((struct mesh_softc *, struct mesh_scb *, int, int)); +void mesh_select __P((struct mesh_softc *, struct mesh_scb *)); +void mesh_identify __P((struct mesh_softc *, struct mesh_scb *)); +void mesh_command __P((struct mesh_softc *, struct mesh_scb *)); +void mesh_dma_setup __P((struct mesh_softc *, struct mesh_scb *)); +void mesh_dataio __P((struct mesh_softc *, struct mesh_scb *)); +void mesh_status __P((struct mesh_softc *, struct mesh_scb *)); +void mesh_msgin __P((struct mesh_softc *, struct mesh_scb *)); +void mesh_msgout __P((struct mesh_softc *, int)); +void mesh_bus_reset __P((struct mesh_softc *)); +void mesh_reset __P((struct mesh_softc *)); +int mesh_stp __P((struct mesh_softc *, int)); +void mesh_setsync __P((struct mesh_softc *, struct mesh_tinfo *)); +struct mesh_scb *mesh_get_scb __P((struct mesh_softc *)); +void mesh_free_scb __P((struct mesh_softc *, struct mesh_scb *)); +int mesh_scsi_cmd __P((struct scsi_xfer *)); +void mesh_sched __P((struct mesh_softc *)); +int mesh_poll __P((struct mesh_softc *, struct scsi_xfer *)); +void mesh_done __P((struct mesh_softc *, struct mesh_scb *)); +void mesh_timeout __P((void *)); +void mesh_sense __P((struct mesh_softc *, struct mesh_scb *)); +void mesh_minphys __P((struct buf *)); + + +#define MESH_DATAOUT 0 +#define MESH_DATAIN MESH_STATUS0_IO +#define MESH_COMMAND MESH_STATUS0_CD +#define MESH_STATUS (MESH_STATUS0_CD | MESH_STATUS0_IO) +#define MESH_MSGOUT (MESH_STATUS0_MSG | MESH_STATUS0_CD) +#define MESH_MSGIN (MESH_STATUS0_MSG | MESH_STATUS0_CD | MESH_STATUS0_IO) + +#define MESH_SELECTING 8 +#define MESH_IDENTIFY 9 +#define MESH_COMPLETE 10 +#define MESH_BUSFREE 11 +#define MESH_UNKNOWN -1 + +#define MESH_PHASE_MASK (MESH_STATUS0_MSG | MESH_STATUS0_CD | MESH_STATUS0_IO) + +struct cfattach mesh_ca = { + sizeof(struct mesh_softc),(cfmatch_t)mesh_match, + mesh_attach +}; + + +struct cfdriver mesh_cd = { + NULL, "mesh", DV_DULL +}; + +struct scsi_device mesh_dev = { + NULL, /* Use default error handler */ + NULL, /* have a queue, served by this */ + NULL, /* have no async handler */ + NULL, /* Use default 'done' routine */ +}; + +int +mesh_match(parent, cf, aux) + struct device *parent; + struct cfdata *cf; + void *aux; +{ + struct confargs *ca = aux; + printf("MESH_MATCH called ,ca->ca_name= %s\n",ca->ca_name); + + if (strcmp(ca->ca_name, "mesh") != 0) + return 0; + + return 1; +} + +void +mesh_attach(parent, self, aux) + struct device *parent, *self; + void *aux; +{ + struct mesh_softc *sc = (void *)self; + struct confargs *ca = aux; + int i; + u_int *reg; + + printf("MESH_ATTACH called\n"); + + reg = ca->ca_reg; + reg[0] += ca->ca_baseaddr; + reg[2] += ca->ca_baseaddr; + sc->sc_reg = mapiodev(reg[0], reg[1]); + sc->sc_irq = ca->ca_intr[0]; + sc->sc_dmareg = mapiodev(reg[2], reg[3]); + + sc->sc_cfflags = self->dv_cfdata->cf_flags; + sc->sc_meshid = mesh_read_reg(sc, MESH_MESH_ID) & 0x1f; +#if 0 + if (sc->sc_meshid != (MESH_SIGNATURE & 0x1f) { + printf(": unknown MESH ID (0x%x)\n", sc->sc_meshid); + return; + } +#endif + if (OF_getprop(ca->ca_node, "clock-frequency", &sc->sc_freq, 4) != 4) { + printf(": cannot get clock-frequency\n"); + return; + } + sc->sc_freq /= 1000000; /* in MHz */ + sc->sc_minsync = 25; /* maximum sync rate = 10MB/sec */ + sc->sc_id = 7; + + TAILQ_INIT(&sc->free_scb); + TAILQ_INIT(&sc->ready_scb); + for (i = 0; i < sizeof(sc->sc_scb)/sizeof(sc->sc_scb[0]); i++) + TAILQ_INSERT_TAIL(&sc->free_scb, &sc->sc_scb[i], chain); + + sc->sc_dmacmd = dbdma_alloc(sizeof(dbdma_command_t) * 20); + timeout_set(&sc->sc_tmo, mesh_timeout, scb); + + mesh_reset(sc); + mesh_bus_reset(sc); + + printf(" irq %d: %dMHz, SCSI ID %d\n", + sc->sc_irq, sc->sc_freq, sc->sc_id); + + sc->sc_adapter.scsi_cmd = mesh_scsi_cmd; + sc->sc_adapter.scsi_minphys = mesh_minphys; + +/* sc->sc_link.scsi_scsi.channel = SCSI_CHANNEL_ONLY_ONE;*/ + sc->sc_link.adapter_softc = sc; + sc->sc_link.adapter_target = sc->sc_id; + sc->sc_link.adapter = &sc->sc_adapter; + sc->sc_link.device = &mesh_dev; + sc->sc_link.openings = 2; +/* sc->sc_link.scsi_scsi.max_target = 7; + sc->sc_link.scsi_scsi.max_lun = 7; + sc->sc_link.type = BUS_SCSI; +*/ + config_found(&sc->sc_dev, &sc->sc_link, scsiprint); + + printf("in mesh_attach, after config_found,calling mac_intr_establish,sc->sc_irq = %d\n",sc->sc_irq); + + /*intr_establish(sc->sc_irq, IST_LEVEL, IPL_BIO, mesh_intr, sc);*/ + mac_intr_establish(parent,sc->sc_irq,IST_LEVEL, IPL_BIO, mesh_intr, sc,"mesh intr"); + /* Reset SCSI bus when halt. */ + shutdownhook_establish(mesh_shutdownhook, sc); +} + +#define MESH_SET_XFER(sc, count) do { \ + mesh_set_reg(sc, MESH_XFER_COUNT0, count); \ + mesh_set_reg(sc, MESH_XFER_COUNT1, count >> 8); \ +} while (0) + +#define MESH_GET_XFER(sc) ((mesh_read_reg(sc, MESH_XFER_COUNT1) << 8) | \ + mesh_read_reg(sc, MESH_XFER_COUNT0)) + +int +mesh_read_reg(sc, reg) + struct mesh_softc *sc; + int reg; +{ + return in8(sc->sc_reg + reg); +} + +void +mesh_set_reg(sc, reg, val) + struct mesh_softc *sc; + int reg, val; +{ + out8(sc->sc_reg + reg, val); +} + +void +mesh_shutdownhook(arg) + void *arg; +{ + struct mesh_softc *sc = arg; + + /* Set to async mode. */ + mesh_set_reg(sc, MESH_SYNC_PARAM, 2); +} + +int +mesh_intr(arg) + void *arg; +{ + struct mesh_softc *sc = arg; + struct mesh_scb *scb; + u_char intr, exception, error, status0, status1; + int i; + + intr = mesh_read_reg(sc, MESH_INTERRUPT); + +#ifdef MESH_DEBUG + if (intr == 0) { + printf("mesh: stray interrupt\n"); + return 0; + } +#endif + + exception = mesh_read_reg(sc, MESH_EXCEPTION); + error = mesh_read_reg(sc, MESH_ERROR); + status0 = mesh_read_reg(sc, MESH_BUS_STATUS0); + status1 = mesh_read_reg(sc, MESH_BUS_STATUS1); + + /* clear interrupt */ + mesh_set_reg(sc, MESH_INTERRUPT, intr); + + scb = sc->sc_nexus; + if (scb == NULL) { +#ifdef MESH_DEBUG + printf("mesh: NULL nexus\n"); +#endif + return 1; + } + + if (sc->sc_flags & MESH_DMA_ACTIVE) { + dbdma_stop(sc->sc_dmareg); + + sc->sc_flags &= ~MESH_DMA_ACTIVE; + scb->resid = MESH_GET_XFER(sc); + + if (mesh_read_reg(sc, MESH_FIFO_COUNT) != 0) + panic("mesh: FIFO != 0"); /* XXX */ + } + + if (intr & MESH_INTR_ERROR) { + mesh_error(sc, scb, error, 0); + return 1; + } + + if (intr & MESH_INTR_EXCEPTION) { + /* selection timeout */ + if (exception & MESH_EXC_SELTO) { + mesh_error(sc, scb, 0, exception); + return 1; + } + + /* phase mismatch */ + if (exception & MESH_EXC_PHASEMM) { + sc->sc_nextstate = status0 & MESH_PHASE_MASK; +#if 0 + printf("mesh: PHASE MISMATCH cdb ="); + printf(" %02x", scb->cmd.opcode); + for (i = 0; i < 5; i++) { + printf(" %02x", scb->cmd.bytes[i]); + } + printf("\n"); +#endif + } + } + + if (sc->sc_nextstate == MESH_UNKNOWN) + sc->sc_nextstate = status0 & MESH_PHASE_MASK; + + switch (sc->sc_nextstate) { + + case MESH_IDENTIFY: + mesh_identify(sc, scb); + break; + case MESH_COMMAND: + mesh_command(sc, scb); + printf("mesh_intr:case MESH_COMMAND\n"); + break; + case MESH_DATAIN: + case MESH_DATAOUT: + mesh_dataio(sc, scb); + printf("mesh_intr:case MESH_DATAIN or MESH_DATAOUT\n"); + break; + case MESH_STATUS: + mesh_status(sc, scb); + printf("mesh_intr:case MESH_STATUS\n"); + break; + case MESH_MSGIN: + mesh_msgin(sc, scb); + printf("mesh_intr:case MESH_MSGIN\n"); + break; + case MESH_COMPLETE: + printf("mesh_intr:case MESH_COMPLETE\n"); + mesh_done(sc, scb); + break; + + default: + panic("mesh: unknown state (0x%x)", sc->sc_nextstate); + } + + return 1; +} + +void +mesh_error(sc, scb, error, exception) + struct mesh_softc *sc; + struct mesh_scb *scb; + int error, exception; +{ + if (error & MESH_ERR_SCSI_RESET) { + printf("mesh: SCSI RESET\n"); + + /* Wait until the RST signal is deasserted. */ + while (mesh_read_reg(sc, MESH_BUS_STATUS1) & MESH_STATUS1_RST); + mesh_reset(sc); + return; + } + + if (error & MESH_ERR_PARITY_ERR0) { + printf("mesh: parity error\n"); + scb->xs->error = XS_DRIVER_STUFFUP; + } + + if (error & MESH_ERR_DISCONNECT) { + printf("mesh: unexpected disconnect\n"); + if (sc->sc_nextstate != MESH_COMPLETE) + scb->xs->error = XS_DRIVER_STUFFUP; + } + + if (exception & MESH_EXC_SELTO) { + /* XXX should reset bus here? */ + scb->xs->error = XS_DRIVER_STUFFUP; + } + + mesh_done(sc, scb); +} + +void +mesh_select(sc, scb) + struct mesh_softc *sc; + struct mesh_scb *scb; +{ + struct mesh_tinfo *ti = &sc->sc_tinfo[scb->target]; + + mesh_setsync(sc, ti); + MESH_SET_XFER(sc, 0); + + /* arbitration */ + + /* + * MESH mistakenly asserts TARGET ID bit along with its own ID bit + * in arbitration phase (like selection). So we should load + * initiator ID to DestID register temporarily. + */ + mesh_set_reg(sc, MESH_DEST_ID, sc->sc_id); + mesh_set_reg(sc, MESH_INTR_MASK, 0); /* disable intr. */ + mesh_set_reg(sc, MESH_SEQUENCE, MESH_CMD_ARBITRATE); + + while (mesh_read_reg(sc, MESH_INTERRUPT) == 0); + mesh_set_reg(sc, MESH_INTERRUPT, 1); + mesh_set_reg(sc, MESH_INTR_MASK, 7); + + /* selection */ + mesh_set_reg(sc, MESH_DEST_ID, scb->target); + mesh_set_reg(sc, MESH_SEQUENCE, MESH_CMD_SELECT | MESH_SEQ_ATN); + + sc->sc_prevphase = MESH_SELECTING; + sc->sc_nextstate = MESH_IDENTIFY; + + timeout_add(&sc->sc_tmo, 10*hz); +} + +void +mesh_identify(sc, scb) + struct mesh_softc *sc; + struct mesh_scb *scb; +{ + mesh_set_reg(sc, MESH_SEQUENCE, MESH_CMD_FLUSH_FIFO); + mesh_msgout(sc, SEND_IDENTIFY); + + sc->sc_nextstate = MESH_COMMAND; + printf("mesh_identify called\n"); +} + +void +mesh_command(sc, scb) + struct mesh_softc *sc; + struct mesh_scb *scb; +{ + struct mesh_tinfo *ti = &sc->sc_tinfo[scb->target]; + int i; + char *cmdp; + + if ((ti->flags & T_SYNCNEGO) == 0) { + ti->period = sc->sc_minsync; + ti->offset = 15; + mesh_msgout(sc, SEND_SDTR); + sc->sc_prevphase = MESH_COMMAND; + sc->sc_nextstate = MESH_MSGIN; + return; + } + + mesh_set_reg(sc, MESH_SEQUENCE, MESH_CMD_FLUSH_FIFO); + + MESH_SET_XFER(sc, scb->cmdlen); + mesh_set_reg(sc, MESH_SEQUENCE, MESH_CMD_COMMAND); + + cmdp = (char *)&scb->cmd; + for (i = 0; i < scb->cmdlen; i++) + mesh_set_reg(sc, MESH_FIFO, *cmdp++); + + if (scb->resid == 0) + sc->sc_nextstate = MESH_STATUS; /* no data xfer */ + else + sc->sc_nextstate = MESH_DATAIN; +} + +void +mesh_dma_setup(sc, scb) + struct mesh_softc *sc; + struct mesh_scb *scb; +{ + struct scsi_xfer *xs = scb->xs; + int datain = scb->flags & MESH_READ; + dbdma_command_t *cmdp; + u_int cmd; + vaddr_t va; + int count, offset; + + cmdp = sc->sc_dmacmd; + cmd = datain ? DBDMA_CMD_IN_MORE : DBDMA_CMD_OUT_MORE; + + count = scb->dlen; + + if (count / NBPG > 32) + panic("mesh: transfer size >= 128k"); + + va = scb->daddr; + offset = va & PGOFSET; + + /* if va is not page-aligned, setup the first page */ + if (offset != 0) { + int rest = NBPG - offset; /* the rest in the page */ + + if (count > rest) { /* if continues to next page */ + DBDMA_BUILD(cmdp, cmd, 0, rest, vtophys(va), + DBDMA_INT_NEVER, DBDMA_WAIT_NEVER, + DBDMA_BRANCH_NEVER); + count -= rest; + va += rest; + cmdp++; + } + } + + /* now va is page-aligned */ + while (count > NBPG) { + DBDMA_BUILD(cmdp, cmd, 0, NBPG, vtophys(va), + DBDMA_INT_NEVER, DBDMA_WAIT_NEVER, DBDMA_BRANCH_NEVER); + count -= NBPG; + va += NBPG; + cmdp++; + } + + /* the last page (count <= NBPG here) */ + cmd = datain ? DBDMA_CMD_IN_LAST : DBDMA_CMD_OUT_LAST; + DBDMA_BUILD(cmdp, cmd , 0, count, vtophys(va), + DBDMA_INT_NEVER, DBDMA_WAIT_NEVER, DBDMA_BRANCH_NEVER); + cmdp++; + + DBDMA_BUILD(cmdp, DBDMA_CMD_STOP, 0, 0, 0, + DBDMA_INT_NEVER, DBDMA_WAIT_NEVER, DBDMA_BRANCH_NEVER); +} + +void +mesh_dataio(sc, scb) + struct mesh_softc *sc; + struct mesh_scb *scb; +{ + mesh_dma_setup(sc, scb); + + if (scb->dlen == 65536) + MESH_SET_XFER(sc, 0); /* TC = 0 means 64KB transfer */ + else + MESH_SET_XFER(sc, scb->dlen); + + if (scb->flags & MESH_READ) + mesh_set_reg(sc, MESH_SEQUENCE, MESH_CMD_DATAIN | MESH_SEQ_DMA); + else + mesh_set_reg(sc, MESH_SEQUENCE, MESH_CMD_DATAOUT | MESH_SEQ_DMA); + dbdma_start(sc->sc_dmareg, sc->sc_dmacmd); + sc->sc_flags |= MESH_DMA_ACTIVE; + sc->sc_nextstate = MESH_STATUS; +} + +void +mesh_status(sc, scb) + struct mesh_softc *sc; + struct mesh_scb *scb; +{ + if (mesh_read_reg(sc, MESH_FIFO_COUNT) == 0) { /* XXX cheat */ + MESH_SET_XFER(sc, 1); + mesh_set_reg(sc, MESH_SEQUENCE, MESH_CMD_STATUS); + sc->sc_nextstate = MESH_STATUS; + return; + } + + scb->status = mesh_read_reg(sc, MESH_FIFO); + + mesh_set_reg(sc, MESH_SEQUENCE, MESH_CMD_FLUSH_FIFO); + MESH_SET_XFER(sc, 1); + mesh_set_reg(sc, MESH_SEQUENCE, MESH_CMD_MSGIN); + + sc->sc_nextstate = MESH_MSGIN; +} + +#define IS1BYTEMSG(m) (((m) != 1 && (m) < 0x20) || (m) & 0x80) +#define IS2BYTEMSG(m) (((m) & 0xf0) == 0x20) +#define ISEXTMSG(m) ((m) == 1) + +void +mesh_msgin(sc, scb) + struct mesh_softc *sc; + struct mesh_scb *scb; +{ + int i; + + if (mesh_read_reg(sc, MESH_FIFO_COUNT) == 0) { /* XXX cheat */ + MESH_SET_XFER(sc, 1); + mesh_set_reg(sc, MESH_SEQUENCE, MESH_CMD_MSGIN); + sc->sc_imsglen = 0; + sc->sc_nextstate = MESH_MSGIN; + return; + } + + sc->sc_imsg[sc->sc_imsglen++] = mesh_read_reg(sc, MESH_FIFO); + + if (sc->sc_imsglen == 1 && IS1BYTEMSG(sc->sc_imsg[0])) + goto gotit; + if (sc->sc_imsglen == 2 && IS2BYTEMSG(sc->sc_imsg[0])) + goto gotit; + if (sc->sc_imsglen >= 3 && ISEXTMSG(sc->sc_imsg[0]) && + sc->sc_imsglen == sc->sc_imsg[1] + 2) + goto gotit; + + sc->sc_nextstate = MESH_MSGIN; + MESH_SET_XFER(sc, 1); + mesh_set_reg(sc, MESH_SEQUENCE, MESH_CMD_MSGIN); + return; + +gotit: +#ifdef DEBUG + printf("msgin:"); + for (i = 0; i < sc->sc_imsglen; i++) + printf(" 0x%02x", sc->sc_imsg[i]); + printf("\n"); +#endif + + switch (sc->sc_imsg[0]) { + case MSG_CMDCOMPLETE: + mesh_set_reg(sc, MESH_SEQUENCE, MESH_CMD_BUSFREE); + sc->sc_nextstate = MESH_COMPLETE; + sc->sc_imsglen = 0; + return; + + case MSG_MESSAGE_REJECT: + switch (sc->sc_msgout) { + case SEND_SDTR: + printf("SDTR rejected\n"); + printf("using async mode\n"); + sc->sc_tinfo[scb->target].period = 0; + sc->sc_tinfo[scb->target].offset = 0; + mesh_setsync(sc, &sc->sc_tinfo[scb->target]); + break; + } + break; + + case MSG_NOOP: + break; + + case MSG_EXTENDED: + goto extended_msg; + + default: + scsi_print_addr(scb->xs->sc_link); + printf("unrecognized MESSAGE(0x%02x); sending REJECT\n", + sc->sc_imsg[0]); + + reject: + mesh_msgout(sc, SEND_REJECT); + return; + } + goto done; + +extended_msg: + /* process an extended message */ + switch (sc->sc_imsg[2]) { + case MSG_EXT_SDTR: + { + struct mesh_tinfo *ti = &sc->sc_tinfo[scb->target]; + int period = sc->sc_imsg[3]; + int offset = sc->sc_imsg[4]; + int r = 250 / period; + int s = (100*250) / period - 100 * r; + + if (period < sc->sc_minsync) { + ti->period = sc->sc_minsync; + ti->offset = 15; + mesh_msgout(sc, SEND_SDTR); + return; + } + scsi_print_addr(scb->xs->sc_link); + /* XXX if (offset != 0) ... */ + printf("max sync rate %d.%02dMb/s\n", r, s); + ti->period = period; + ti->offset = offset; + ti->flags |= T_SYNCNEGO; + ti->flags |= T_SYNCMODE; + mesh_setsync(sc, ti); + goto done; + } + default: + printf("%s target %d: rejecting extended message 0x%x\n", + sc->sc_dev.dv_xname, scb->target, sc->sc_imsg[0]); + goto reject; + } + +done: + sc->sc_imsglen = 0; + sc->sc_nextstate = MESH_UNKNOWN; + + mesh_set_reg(sc, MESH_SEQUENCE, MESH_CMD_BUSFREE); /* XXX really? */ +} + +void +mesh_msgout(sc, msg) + struct mesh_softc *sc; + int msg; +{ + struct mesh_scb *scb = sc->sc_nexus; + struct mesh_tinfo *ti; + int lun, i; + + switch (msg) { + case SEND_REJECT: + sc->sc_omsglen = 1; + sc->sc_omsg[0] = MSG_MESSAGE_REJECT; + break; + case SEND_IDENTIFY: + lun = scb->xs->sc_link->lun; + sc->sc_omsglen = 1; + sc->sc_omsg[0] = MSG_IDENTIFY(lun, 0); + break; + case SEND_SDTR: + ti = &sc->sc_tinfo[scb->target]; + sc->sc_omsglen = 5; + sc->sc_omsg[0] = MSG_EXTENDED; + sc->sc_omsg[1] = 3; + sc->sc_omsg[2] = MSG_EXT_SDTR; + sc->sc_omsg[3] = ti->period; + sc->sc_omsg[4] = ti->offset; + break; + } + sc->sc_msgout = msg; + + MESH_SET_XFER(sc, sc->sc_omsglen); + mesh_set_reg(sc, MESH_SEQUENCE, MESH_CMD_MSGOUT | MESH_SEQ_ATN); + + for (i = 0; i < sc->sc_omsglen; i++) + mesh_set_reg(sc, MESH_FIFO, sc->sc_omsg[i]); + + sc->sc_nextstate = MESH_UNKNOWN; +} + +void +mesh_bus_reset(sc) + struct mesh_softc *sc; +{ + /* Disable interrupts. */ + mesh_set_reg(sc, MESH_INTR_MASK, 0); + + /* Assert RST line. */ + mesh_set_reg(sc, MESH_BUS_STATUS1, MESH_STATUS1_RST); + delay(50); + mesh_set_reg(sc, MESH_BUS_STATUS1, 0); + + mesh_reset(sc); +} + +void +mesh_reset(sc) + struct mesh_softc *sc; +{ + int i; + + /* Reset DMA first. */ + dbdma_reset(sc->sc_dmareg); + + /* Disable interrupts. */ + mesh_set_reg(sc, MESH_INTR_MASK, 0); + + mesh_set_reg(sc, MESH_SEQUENCE, MESH_CMD_RESET_MESH); + delay(1); + + /* Wait for reset done. */ + while (mesh_read_reg(sc, MESH_INTERRUPT) == 0); + + /* Clear interrupts */ + mesh_set_reg(sc, MESH_INTERRUPT, 0x7); + + /* Set SCSI ID */ + mesh_set_reg(sc, MESH_SOURCE_ID, sc->sc_id); + + /* Set to async mode by default. */ + mesh_set_reg(sc, MESH_SYNC_PARAM, 2); + + /* Set selection timeout to 250ms. */ + mesh_set_reg(sc, MESH_SEL_TIMEOUT, 250 * sc->sc_freq / 500); + + /* Enable parity check. */ + mesh_set_reg(sc, MESH_SEQUENCE, MESH_CMD_ENABLE_PARITY); + + /* Enable all interrupts. */ + mesh_set_reg(sc, MESH_INTR_MASK, 0x7); + + for (i = 0; i < 7; i++) { + struct mesh_tinfo *ti = &sc->sc_tinfo[i]; + + ti->flags = 0; + ti->period = ti->offset = 0; + if (sc->sc_cfflags & (1 << i)) { + ti->flags |= T_SYNCNEGO; + } + } + sc->sc_nexus = NULL; +} + +int +mesh_stp(sc, v) + struct mesh_softc *sc; + int v; +{ + /* + * stp(v) = 5 * clock_period (v == 0) + * = (v + 2) * 2 clock_period (v > 0) + */ + + if (v == 0) + return 5 * 250 / sc->sc_freq; + else + return (v + 2) * 2 * 250 / sc->sc_freq; +} + +void +mesh_setsync(sc, ti) + struct mesh_softc *sc; + struct mesh_tinfo *ti; +{ + int period = ti->period; + int offset = ti->offset; + int v; + + if ((ti->flags & T_SYNCMODE) == 0) + offset = 0; + + if (offset == 0) { /* async mode */ + mesh_set_reg(sc, MESH_SYNC_PARAM, 2); + return; + } + + v = period * sc->sc_freq / 250 / 2 - 2; + if (v < 0) + v = 0; + if (mesh_stp(sc, v) < period) + v++; + if (v > 15) + v = 15; + mesh_set_reg(sc, MESH_SYNC_PARAM, (offset << 4) | v); +} + +struct mesh_scb * +mesh_get_scb(sc) + struct mesh_softc *sc; +{ + struct mesh_scb *scb; + int s; + + s = splbio(); + while ((scb = sc->free_scb.tqh_first) == NULL) + tsleep(&sc->free_scb, PRIBIO, "meshscb", 0); + TAILQ_REMOVE(&sc->free_scb, scb, chain); + splx(s); + + return scb; +} + +void +mesh_free_scb(sc, scb) + struct mesh_softc *sc; + struct mesh_scb *scb; +{ + int s; + + s = splbio(); + TAILQ_INSERT_HEAD(&sc->free_scb, scb, chain); + if (scb->chain.tqe_next == NULL) + wakeup(&sc->free_scb); + splx(s); +} + +int +mesh_scsi_cmd(xs) + struct scsi_xfer *xs; +{ + struct scsi_link *sc_link = xs->sc_link; + struct mesh_softc *sc = sc_link->adapter_softc; + struct mesh_scb *scb; + u_int flags; + int s; + + flags = xs->flags; + + scb = mesh_get_scb(sc); + scb->xs = xs; + scb->flags = 0; + scb->status = 0; + scb->daddr = (vaddr_t)xs->data; + scb->dlen = xs->datalen; + scb->resid = xs->datalen; + bcopy(xs->cmd, &scb->cmd, xs->cmdlen); + scb->cmdlen = xs->cmdlen; + + scb->target = sc_link->target; + sc->sc_imsglen = 0; /* XXX ? */ + + printf("messh_scsi_cmd,scb->target=%d\n",scb->target); + + if (flags & SCSI_POLL){ + printf("mesh_scsi_cmd:flags=SCSI_POLL\n"); + scb->flags |= MESH_POLL; + } + +#if 0 + if (flags & SCSI_DATA_OUT) + scb->flags &= ~MESH_READ; +#endif + if (flags & SCSI_DATA_IN){ + printf("mesh_scsi_cmd:flags=SCSI_DATA_IN\n"); + scb->flags |= MESH_READ; + } + + s = splbio(); + + TAILQ_INSERT_TAIL(&sc->ready_scb, scb, chain); + + if (sc->sc_nexus == NULL){ + printf("mesh_scsi_cmd:sc->sc_nexus == NULL,calling mesh_sched\n"); /* IDLE */ + mesh_sched(sc); + } + + splx(s); + + if ((flags & SCSI_POLL) == 0){ + printf("mesh_scsi_cmd: returning SUCCESSFULLY_QUEUED\n"); + return SUCCESSFULLY_QUEUED; + } + + if (mesh_poll(sc, xs)) { + printf("mesh: timeout\n"); + if (mesh_poll(sc, xs)) + printf("mesh: timeout again\n"); + } + printf("mesh_scsi_cmd: returning COMPLETE\n"); + return COMPLETE; +} + +void +mesh_sched(sc) + struct mesh_softc *sc; +{ + struct scsi_xfer *xs; + struct scsi_link *sc_link; + struct mesh_scb *scb; + + scb = sc->ready_scb.tqh_first; +start: + if (scb == NULL) + return; + + xs = scb->xs; + sc_link = xs->sc_link; + + if (sc->sc_nexus == NULL) { + TAILQ_REMOVE(&sc->ready_scb, scb, chain); + sc->sc_nexus = scb; + mesh_select(sc, scb); + return; + } + + scb = scb->chain.tqe_next; + goto start; +} + +int +mesh_poll(sc, xs) + struct mesh_softc *sc; + struct scsi_xfer *xs; +{ + int count = xs->timeout; + printf("in mesh_poll,timeout=%d\n",xs->timeout); + + + while (count) { + if (mesh_read_reg(sc, MESH_INTERRUPT)) + mesh_intr(sc); + + if (xs->flags & ITSDONE) + return 0; + DELAY(1000); + count--; + }; + return 1; +} + +void +mesh_done(sc, scb) + struct mesh_softc *sc; + struct mesh_scb *scb; +{ + struct scsi_xfer *xs = scb->xs; + +#ifdef MESH_SHOWSTATE + printf("mesh_done\n"); +#endif + + sc->sc_nextstate = MESH_BUSFREE; + sc->sc_nexus = NULL; + + timeout_del(&sc->sc_tmo); + + if (scb->status == SCSI_BUSY) { + xs->error = XS_BUSY; + printf("Target busy\n"); + } + + if (scb->status == SCSI_CHECK) { + if (scb->flags & MESH_SENSE) + panic("SCSI_CHECK && MESH_SENSE?"); + xs->resid = scb->resid; + mesh_sense(sc, scb); + return; + } + + if (xs->error == XS_NOERROR) { + xs->status = scb->status; + if (scb->flags & MESH_SENSE) + xs->error = XS_SENSE; + else + xs->resid = scb->resid; + } + + xs->flags |= ITSDONE; + + mesh_set_reg(sc, MESH_SYNC_PARAM, 2); + + if ((xs->flags & SCSI_POLL) == 0) + mesh_sched(sc); + + scsi_done(xs); + mesh_free_scb(sc, scb); +} + +void +mesh_timeout(arg) + void *arg; +{ + struct mesh_scb *scb = arg; + struct mesh_softc *sc = scb->xs->sc_link->adapter_softc; + int s; + int status0, status1; + int intr, error, exception; + + printf("mesh: timeout state=%x\n", sc->sc_nextstate); + intr = mesh_read_reg(sc, MESH_INTERRUPT); + + + exception = mesh_read_reg(sc, MESH_EXCEPTION); + + error = mesh_read_reg(sc, MESH_ERROR); + + status0 = mesh_read_reg(sc, MESH_BUS_STATUS0); + + status1 = mesh_read_reg(sc, MESH_BUS_STATUS1); + +#if 1 +printf("intr 0x%02x, except 0x%02x, err 0x%02x\n", intr, exception, error); +#endif + + s = splbio(); + + if (sc->sc_flags & MESH_DMA_ACTIVE) { + printf("mesh: resetting dma\n"); + dbdma_reset(sc->sc_dmareg); + } + scb->xs->error = XS_TIMEOUT; + + mesh_set_reg(sc, MESH_SEQUENCE, MESH_CMD_BUSFREE); + + sc->sc_nextstate = MESH_COMPLETE; + + splx(s); + printf("rerturning from mesh_timeout\n"); +} + +void +mesh_sense(sc, scb) + struct mesh_softc *sc; + struct mesh_scb *scb; +{ + struct scsi_xfer *xs = scb->xs; + struct scsi_link *sc_link = xs->sc_link; + struct scsi_sense *ss = (void *)&scb->cmd; + + bzero(ss, sizeof(*ss)); + ss->opcode = REQUEST_SENSE; + ss->byte2 = sc_link->lun << 5; + ss->length = sizeof(struct scsi_sense_data); + scb->cmdlen = sizeof(*ss); + scb->daddr = (vaddr_t)&xs->sense; + scb->dlen = sizeof(struct scsi_sense_data); + scb->resid = scb->dlen; + bzero((void *)scb->daddr, scb->dlen); + + scb->flags |= MESH_SENSE | MESH_READ; + + TAILQ_INSERT_HEAD(&sc->ready_scb, scb, chain); + if (sc->sc_nexus == NULL) + mesh_sched(sc); +} + +void +mesh_minphys(bp) + struct buf *bp; +{ + if (bp->b_bcount > 64*1024) + bp->b_bcount = 64*1024; + + minphys(bp); +} |