diff options
author | Miod Vallat <miod@cvs.openbsd.org> | 2004-09-06 06:25:29 +0000 |
---|---|---|
committer | Miod Vallat <miod@cvs.openbsd.org> | 2004-09-06 06:25:29 +0000 |
commit | e25eb6048c5c5ab1481e24c5595905b483564bcc (patch) | |
tree | 5a3c601eee6f4d71f87c6b3af1c9962682dda6e0 /sys/arch/mvme88k/dev | |
parent | 68a72e1690ba2cd8ba8b785a70c770b3bd7ad9c0 (diff) |
Jumbo pack of fixes:
- do not leak memory when polling;
- bring LUN support back - Motorola documentation says LUNs are not
supported, but it's a SysV/m88k limitation, not a hardware one.
- honour request timeout while polling (instead of using a fixed value)
- do not program the scsi command length if the hardware knows it from
the scsi command group (as advised in the manual)
- various minor fixes, especially better error recovery.
tested by nick@ and I; ok deraadt@.
Diffstat (limited to 'sys/arch/mvme88k/dev')
-rw-r--r-- | sys/arch/mvme88k/dev/vs.c | 288 | ||||
-rw-r--r-- | sys/arch/mvme88k/dev/vsreg.h | 4 | ||||
-rw-r--r-- | sys/arch/mvme88k/dev/vsvar.h | 4 |
3 files changed, 152 insertions, 144 deletions
diff --git a/sys/arch/mvme88k/dev/vs.c b/sys/arch/mvme88k/dev/vs.c index 2873e22acc5..5c3ec56f599 100644 --- a/sys/arch/mvme88k/dev/vs.c +++ b/sys/arch/mvme88k/dev/vs.c @@ -1,4 +1,4 @@ -/* $OpenBSD: vs.c,v 1.52 2004/07/30 19:02:06 miod Exp $ */ +/* $OpenBSD: vs.c,v 1.53 2004/09/06 06:25:28 miod Exp $ */ /* * Copyright (c) 2004, Miodrag Vallat. @@ -88,16 +88,16 @@ struct cfdriver vs_cd = { NULL, "vs", DV_DULL, }; -int do_vspoll(struct vs_softc *, struct scsi_xfer *, int, int); +int do_vspoll(struct vs_softc *, struct scsi_xfer *, int); void thaw_queue(struct vs_softc *, int); void thaw_all_queues(struct vs_softc *); M328_SG vs_alloc_scatter_gather(void); M328_SG vs_build_memory_structure(struct vs_softc *, struct scsi_xfer *, bus_addr_t); -int vs_checkintr(struct vs_softc *, struct scsi_xfer *, int *); void vs_chksense(struct scsi_xfer *); void vs_dealloc_scatter_gather(M328_SG); int vs_eintr(void *); +void vs_free(M328_CMD *); bus_addr_t vs_getcqe(struct vs_softc *); bus_addr_t vs_getiopb(struct vs_softc *); int vs_initialize(struct vs_softc *); @@ -105,11 +105,12 @@ int vs_intr(struct vs_softc *); void vs_link_sg_element(sg_list_element_t *, vaddr_t, int); void vs_link_sg_list(sg_list_element_t *, vaddr_t, int); int vs_nintr(void *); -int vs_poll(struct vs_softc *, struct scsi_xfer *); +int vs_poll(struct vs_softc *, M328_CMD *); +void vs_print_addr(struct vs_softc *, struct scsi_xfer *); int vs_queue_number(struct scsi_link *, struct vs_softc *); void vs_reset(struct vs_softc *, int); void vs_resync(struct vs_softc *); -void vs_scsidone(struct vs_softc *, struct scsi_xfer *, int); +void vs_scsidone(struct vs_softc *, struct scsi_xfer *); static __inline__ void vs_clear_return_info(struct vs_softc *); static __inline__ paddr_t kvtop(vaddr_t); @@ -193,7 +194,9 @@ vsattach(struct device *parent, struct device *self, void *args) sc_link->adapter_softc = sc; sc_link->adapter_target = sc->sc_id[bus]; sc_link->device = &vs_scsidev; +#if 0 sc_link->luns = 1; +#endif sc_link->openings = NUM_IOPB / 8; if (bus != 0) sc_link->flags = SDEV_2NDBUS; @@ -223,71 +226,76 @@ vsattach(struct device *parent, struct device *self, void *args) bootbus = 0; } +void +vs_print_addr(struct vs_softc *sc, struct scsi_xfer *xs) +{ + if (xs == NULL) + printf("%s: ", sc->sc_dev.dv_xname); + else { + sc_print_addr(xs->sc_link); + + /* print bus number too if appropriate */ + if (sc->sc_id[1] >= 0) + printf("(bus %d) ", + !!(xs->sc_link->flags & SDEV_2NDBUS)); + } +} + int -do_vspoll(struct vs_softc *sc, struct scsi_xfer *xs, int to, int canreset) +do_vspoll(struct vs_softc *sc, struct scsi_xfer *xs, int canreset) { - int i; + int to; int crsw, bus; - if (xs != NULL) + if (xs != NULL) { bus = !!(xs->sc_link->flags & SDEV_2NDBUS); - else + to = xs->timeout; + if (to == 0) + to = 2000; + } else { bus = -1; - - /* XXX use cmd_wait values */ - if (to <= 0) - to = 50000; - i = 10000; + to = 2000; + } while (((crsw = CRSW) & (M_CRSW_CRBV | M_CRSW_CC)) == 0) { - if (--i <= 0) { - i = 50000; - --to; - if (to <= 0) { - if (canreset) { - vs_reset(sc, bus); - vs_resync(sc); - } - if (xs == NULL) - printf("%s: ", sc->sc_dev.dv_xname); - else - sc_print_addr(xs->sc_link); - printf("timeout %d crsw 0x%x\n", to, crsw); - return 1; + if (to-- <= 0) { + vs_print_addr(sc, xs); + printf("command timeout, crsw 0x%x\n", crsw); + + if (canreset) { + vs_reset(sc, bus); + vs_resync(sc); } + return 1; } + delay(1000); } return 0; } int -vs_poll(struct vs_softc *sc, struct scsi_xfer *xs) +vs_poll(struct vs_softc *sc, M328_CMD *cmd) { - int status; - int to; + struct scsi_xfer *xs; + int rc; - to = xs->timeout / 1000; - for (;;) { - if (do_vspoll(sc, xs, to, 1)) { - xs->error = XS_SELTIMEOUT; - xs->status = -1; - xs->flags |= ITSDONE; - vs_clear_return_info(sc); - if (xs->flags & SCSI_POLL) - return (COMPLETE); - break; - } - if (vs_checkintr(sc, xs, &status)) { - vs_scsidone(sc, xs, status); - } + xs = cmd->xs; + rc = do_vspoll(sc, xs, 1); + vs_free(cmd); - if (CRSW & M_CRSW_ER) - CRB_CLR_ER; - CRB_CLR_DONE; + if (rc != 0) { + xs->error = XS_SELTIMEOUT; + xs->status = -1; + xs->flags |= ITSDONE; +#if 0 + scsi_done(xs); +#endif + } else + vs_scsidone(sc, xs); - if (xs->flags & ITSDONE) - break; - } + if (CRSW & M_CRSW_ER) + CRB_CLR_ER; + CRB_CLR_DONE; vs_clear_return_info(sc); return (COMPLETE); @@ -308,18 +316,28 @@ thaw_all_queues(struct vs_softc *sc) { int i; - for (i = 1; i < NUM_WQ ; i++) + for (i = 1; i < NUM_WQ; i++) thaw_queue(sc, i); } void -vs_scsidone(struct vs_softc *sc, struct scsi_xfer *xs, int stat) +vs_scsidone(struct vs_softc *sc, struct scsi_xfer *xs) { + u_int32_t len; + int error; int tgt; - tgt = vs_queue_number(xs->sc_link, sc); - xs->status = stat; + len = vs_read(4, sh_RET_IOPB + IOPB_LENGTH); + xs->resid = xs->datalen - len; + error = vs_read(2, sh_RET_IOPB + IOPB_STATUS); + if ((error & 0xff) == SCSI_SELECTION_TO) { + xs->error = XS_SELTIMEOUT; + xs->status = -1; + } else + xs->status = error >> 8; + + tgt = vs_queue_number(xs->sc_link, sc); while (xs->status == SCSI_CHECK) { vs_chksense(xs); thaw_queue(sc, tgt); @@ -345,6 +363,10 @@ vs_scsicmd(struct scsi_xfer *xs) if (flags & SCSI_POLL) { cqep = sh_MCE; iopb = sh_MCE_IOPB; + + /* Wait until we can use the command queue entry. */ + while (mce_read(2, CQE_QECR) & M_QECR_GO) + ; } else { cqep = vs_getcqe(sc); if (cqep == 0) { @@ -356,7 +378,22 @@ vs_scsicmd(struct scsi_xfer *xs) vs_bzero(iopb, IOPB_LONG_SIZE); - iopb_len = IOPB_SHORT_SIZE + xs->cmdlen; + /* + * We should only provide the iopb len if the controller is not + * able to compute it from the SCSI command group. + * Note that it has no knowledge of group 2. + */ + switch ((xs->cmd->opcode) >> 5) { + case 0: + case 1: + case 5: + iopb_len = 0; + break; + default: + iopb_len = IOPB_SHORT_SIZE + xs->cmdlen; + break; + } + bus_space_write_region_1(sc->sc_iot, sc->sc_ioh, iopb + IOPB_SCSI_DATA, (u_int8_t *)xs->cmd, xs->cmdlen); @@ -387,14 +424,6 @@ vs_scsicmd(struct scsi_xfer *xs) } vs_write(2, iopb + IOPB_ADDR, ADDR_MOD); - /* - * Wait until we can use the command queue entry. - * Should only have to wait if the master command - * queue entry is busy and we are polling. - */ - while (vs_read(2, cqep + CQE_QECR) & M_QECR_GO) - ; - vs_write(2, cqep + CQE_IOPB_ADDR, iopb); vs_write(1, cqep + CQE_IOPB_LENGTH, iopb_len); vs_write(1, cqep + CQE_WORK_QUEUE, @@ -417,7 +446,7 @@ vs_scsicmd(struct scsi_xfer *xs) if (flags & SCSI_POLL) { /* poll for the command to complete */ - return vs_poll(sc, xs); + return vs_poll(sc, m328_cmd); } return (SUCCESSFULLY_QUEUED); @@ -437,6 +466,10 @@ vs_chksense(struct scsi_xfer *xs) CRB_CLR_DONE; xs->status = 0; + /* Wait until we can use the command queue entry. */ + while (mce_read(2, CQE_QECR) & M_QECR_GO) + ; + vs_bzero(sh_MCE_IOPB, IOPB_LONG_SIZE); /* This is a command, so point to it */ ss = (void *)(bus_space_vaddr(sc->sc_iot, sc->sc_ioh) + @@ -458,14 +491,13 @@ vs_chksense(struct scsi_xfer *xs) vs_bzero(sh_MCE, CQE_SIZE); mce_write(2, CQE_IOPB_ADDR, sh_MCE_IOPB); - mce_write(1, CQE_IOPB_LENGTH, - IOPB_SHORT_SIZE + sizeof(struct scsi_sense)); + mce_write(1, CQE_IOPB_LENGTH, 0); mce_write(1, CQE_WORK_QUEUE, 0); mce_write(2, CQE_QECR, M_QECR_GO); /* poll for the command to complete */ s = splbio(); - do_vspoll(sc, xs, 0, 1); + do_vspoll(sc, xs, 1); xs->status = vs_read(2, sh_RET_IOPB + IOPB_STATUS) >> 8; splx(s); } @@ -593,15 +625,19 @@ vs_initialize(struct vs_softc *sc) vs_bzero(sh_MCE, CQE_SIZE); mce_write(2, CQE_IOPB_ADDR, sh_MCE_IOPB); - mce_write(1, CQE_IOPB_LENGTH, IOPB_LONG_SIZE); + mce_write(1, CQE_IOPB_LENGTH, 0); mce_write(1, CQE_WORK_QUEUE, 0); mce_write(2, CQE_QECR, M_QECR_GO); /* poll for the command to complete */ - do_vspoll(sc, NULL, 0, 1); + do_vspoll(sc, NULL, 1); /* initialize work queues */ for (i = 1; i < NUM_WQ; i++) { + /* Wait until we can use the command queue entry. */ + while (mce_read(2, CQE_QECR) & M_QECR_GO) + ; + vs_bzero(sh_MCE_IOPB, IOPB_LONG_SIZE); mce_iopb_write(2, WQCF_CMD, CNTR_INIT_WORKQ); mce_iopb_write(2, WQCF_OPTION, 0); @@ -615,32 +651,23 @@ vs_initialize(struct vs_softc *sc) vs_bzero(sh_MCE, CQE_SIZE); mce_write(2, CQE_IOPB_ADDR, sh_MCE_IOPB); - mce_write(1, CQE_IOPB_LENGTH, IOPB_LONG_SIZE); + mce_write(1, CQE_IOPB_LENGTH, 0); mce_write(1, CQE_WORK_QUEUE, 0); mce_write(2, CQE_QECR, M_QECR_GO); /* poll for the command to complete */ - do_vspoll(sc, NULL, 0, 1); - if (CRSW & M_CRSW_ER) - CRB_CLR_ER; + do_vspoll(sc, NULL, 1); + if (CRSW & M_CRSW_ER) { + printf("work queue %d initialization error 0x%x\n", + i, vs_read(2, sh_RET_IOPB + IOPB_STATUS)); + return 1; + } CRB_CLR_DONE; -#if 0 - delay(500); -#endif } /* start queue mode */ mcsb_write(2, MCSB_MCR, mcsb_read(2, MCSB_MCR) | M_MCR_SQM); - do_vspoll(sc, NULL, 0, 1); - if (CRSW & M_CRSW_ER) { - printf("initialization error, status = 0x%x\n", - vs_read(2, sh_RET_IOPB + IOPB_STATUS)); - CRB_CLR_DONE; - return 1; - } - CRB_CLR_DONE; - /* reset all SCSI buses */ vs_reset(sc, -1); /* sync all devices */ @@ -662,6 +689,10 @@ vs_resync(struct vs_softc *sc) if (target == sc->sc_id[bus]) continue; + /* Wait until we can use the command queue entry. */ + while (mce_read(2, CQE_QECR) & M_QECR_GO) + ; + vs_bzero(sh_MCE_IOPB, IOPB_SHORT_SIZE); mce_iopb_write(2, DRCF_CMD, CNTR_DEV_REINIT); mce_iopb_write(2, DRCF_OPTION, 0); /* prefer polling */ @@ -673,12 +704,12 @@ vs_resync(struct vs_softc *sc) vs_bzero(sh_MCE, CQE_SIZE); mce_write(2, CQE_IOPB_ADDR, sh_MCE_IOPB); - mce_write(1, CQE_IOPB_LENGTH, IOPB_SHORT_SIZE); + mce_write(1, CQE_IOPB_LENGTH, 0); mce_write(1, CQE_WORK_QUEUE, 0); mce_write(2, CQE_QECR, M_QECR_GO); /* poll for the command to complete */ - do_vspoll(sc, NULL, 0, 0); + do_vspoll(sc, NULL, 0); if (CRSW & M_CRSW_ER) CRB_CLR_ER; CRB_CLR_DONE; @@ -697,6 +728,10 @@ vs_reset(struct vs_softc *sc, int bus) if (bus >= 0 && b != bus) continue; + /* Wait until we can use the command queue entry. */ + while (mce_read(2, CQE_QECR) & M_QECR_GO) + ; + vs_bzero(sh_MCE_IOPB, IOPB_SHORT_SIZE); mce_iopb_write(2, SRCF_CMD, IOPB_RESET); mce_iopb_write(2, SRCF_OPTION, 0); /* prefer polling */ @@ -707,13 +742,13 @@ vs_reset(struct vs_softc *sc, int bus) vs_bzero(sh_MCE, CQE_SIZE); mce_write(2, CQE_IOPB_ADDR, sh_MCE_IOPB); - mce_write(1, CQE_IOPB_LENGTH, IOPB_SHORT_SIZE); + mce_write(1, CQE_IOPB_LENGTH, 0); mce_write(1, CQE_WORK_QUEUE, 0); mce_write(2, CQE_QECR, M_QECR_GO); /* poll for the command to complete */ for (;;) { - do_vspoll(sc, NULL, 0, 0); + do_vspoll(sc, NULL, 0); /* ack & clear scsi error condition cause by reset */ if (CRSW & M_CRSW_ER) { CRB_CLR_DONE; @@ -729,30 +764,14 @@ vs_reset(struct vs_softc *sc, int bus) splx(s); } -/* - * Process an interrupt from the MVME328 - * We'll generally update: xs->{flags,resid,error,sense,status} and - * occasionally xs->retries. - */ -int -vs_checkintr(struct vs_softc *sc, struct scsi_xfer *xs, int *status) +void +vs_free(M328_CMD *m328_cmd) { - u_int32_t len; - int error; - - len = vs_read(4, sh_RET_IOPB + IOPB_LENGTH); - error = vs_read(2, sh_RET_IOPB + IOPB_STATUS); - *status = error >> 8; - - xs->resid = xs->datalen - len; - - if ((error & 0xff) == SCSI_SELECTION_TO) { - xs->error = XS_SELTIMEOUT; - xs->status = -1; - *status = -1; + if (m328_cmd->top_sg_list != NULL) { + vs_dealloc_scatter_gather(m328_cmd->top_sg_list); + m328_cmd->top_sg_list = (M328_SG)NULL; } - - return 1; + FREE(m328_cmd, M_DEVBUF); /* free the command tag */ } /* normal interrupt routine */ @@ -762,7 +781,6 @@ vs_nintr(void *vsc) struct vs_softc *sc = (struct vs_softc *)vsc; M328_CMD *m328_cmd; struct scsi_xfer *xs; - int status; int s; if ((CRSW & CONTROLLER_ERROR) == CONTROLLER_ERROR) @@ -780,15 +798,9 @@ vs_nintr(void *vsc) */ if (m328_cmd != NULL) { xs = m328_cmd->xs; - if (m328_cmd->top_sg_list != NULL) { - vs_dealloc_scatter_gather(m328_cmd->top_sg_list); - m328_cmd->top_sg_list = (M328_SG)NULL; - } - FREE(m328_cmd, M_DEVBUF); /* free the command tag */ + vs_free(m328_cmd); - if (vs_checkintr(sc, xs, &status)) { - vs_scsidone(sc, xs, status); - } + vs_scsidone(sc, xs); } /* ack the interrupt */ @@ -827,10 +839,7 @@ vs_eintr(void *vsc) return 1; } - if (xs == NULL) - printf("%s: ", sc->sc_dev.dv_xname); - else - sc_print_addr(xs->sc_link); + vs_print_addr(sc, xs); switch (ecode) { case CEVSB_ERR_TYPE: @@ -838,12 +847,6 @@ vs_eintr(void *vsc) break; case CEVSB_ERR_TO: printf("timeout\n"); - if (xs != NULL) { - xs->error = XS_SELTIMEOUT; - xs->status = -1; - xs->flags |= ITSDONE; - scsi_done(xs); - } break; case CEVSB_ERR_TR: printf("reconnect error\n"); @@ -865,6 +868,13 @@ vs_eintr(void *vsc) break; } + if (xs != NULL) { + xs->error = XS_SELTIMEOUT; + xs->status = -1; + xs->flags |= ITSDONE; + scsi_done(xs); + } + if (CRSW & M_CRSW_ER) CRB_CLR_ER; CRB_CLR_DONE; @@ -895,20 +905,18 @@ vs_clear_return_info(struct vs_softc *sc) int vs_queue_number(struct scsi_link *sl, struct vs_softc *sc) { - int bus; + int bus, target; bus = !!(sl->flags & SDEV_2NDBUS); + target = sl->target; - if (sl->target == sc->sc_id[bus]) + if (target == sc->sc_id[bus]) return 0; - if (bus == 0) - return sl->target == 0 ? sc->sc_id[bus] : sl->target; - else - if (sl->target > sc->sc_id[bus]) - return 8 + sl->target - 1; - else - return 8 + sl->target; + if (target > sc->sc_id[bus]) + target--; + + return (bus == 0 ? 1 : 8) + target; } /* diff --git a/sys/arch/mvme88k/dev/vsreg.h b/sys/arch/mvme88k/dev/vsreg.h index b5a5664882a..96f2f4e69cc 100644 --- a/sys/arch/mvme88k/dev/vsreg.h +++ b/sys/arch/mvme88k/dev/vsreg.h @@ -1,4 +1,4 @@ -/* $OpenBSD: vsreg.h,v 1.12 2004/07/20 20:32:02 miod Exp $ */ +/* $OpenBSD: vsreg.h,v 1.13 2004/09/06 06:25:28 miod Exp $ */ /* * Copyright (c) 2004, Miodrag Vallat. * Copyright (c) 1999 Steve Murphree, Jr. @@ -40,8 +40,8 @@ */ #define JAGUAR_MIN_Q_SIZ 2 -#define JAGUAR_MAX_Q_SIZ 2 #define JAGUAR_MAX_CTLR_CMDS 80 /* Interphase says so */ +#define JAGUAR_MAX_Q_SIZ (JAGUAR_MAX_CTLR_CMDS / NUM_WQ) /* * COUGAR specific device limits diff --git a/sys/arch/mvme88k/dev/vsvar.h b/sys/arch/mvme88k/dev/vsvar.h index 3cac7e42ce9..18a4659aa15 100644 --- a/sys/arch/mvme88k/dev/vsvar.h +++ b/sys/arch/mvme88k/dev/vsvar.h @@ -1,4 +1,4 @@ -/* $OpenBSD: vsvar.h,v 1.16 2004/07/30 19:02:06 miod Exp $ */ +/* $OpenBSD: vsvar.h,v 1.17 2004/09/06 06:25:28 miod Exp $ */ /* * Copyright (c) 2004, Miodrag Vallat. * Copyright (c) 1999 Steve Murphree, Jr. @@ -130,7 +130,7 @@ struct vs_softc { #define THAW_REG mcsb_read(2, MCSB_THAW) #define THAW(x) \ do { \ - mcsb_write(1, MCSB_THAW, (x) << 8); \ + mcsb_write(1, MCSB_THAW, (x)); \ mcsb_write(1, MCSB_THAW + 1, M_THAW_TWQE); \ } while (0) |