diff options
Diffstat (limited to 'sys/arch/vax/vsa/ncr.c')
-rw-r--r-- | sys/arch/vax/vsa/ncr.c | 1283 |
1 files changed, 1283 insertions, 0 deletions
diff --git a/sys/arch/vax/vsa/ncr.c b/sys/arch/vax/vsa/ncr.c new file mode 100644 index 00000000000..69e74118c1d --- /dev/null +++ b/sys/arch/vax/vsa/ncr.c @@ -0,0 +1,1283 @@ +/* $NetBSD: ncr.c,v 1.5 1996/10/13 03:36:14 christos Exp $ */ + +/* #define DEBUG /* */ +/* #define TRACE /* */ +/* #define POLL_MODE /* */ +#define USE_VMAPBUF + +/* + * Copyright (c) 1995 David Jones, Gordon W. Ross + * Copyright (c) 1994 Adam Glass + * 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. The name of the authors may not be used to endorse or promote products + * derived from this software without specific prior written permission. + * 4. All advertising materials mentioning features or use of this software + * must display the following acknowledgement: + * This product includes software developed by + * Adam Glass, David Jones, and Gordon Ross + * + * THIS SOFTWARE IS PROVIDED BY THE AUTHORS ``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 AUTHORS 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. + */ + +/* + * This file contains only the machine-dependent parts of the + * Sun3 SCSI driver. (Autoconfig stuff and DMA functions.) + * The machine-independent parts are in ncr5380sbc.c + * + * Supported hardware includes: + * Sun SCSI-3 on OBIO (Sun3/50,Sun3/60) + * Sun SCSI-3 on VME (Sun3/160,Sun3/260) + * + * Could be made to support the Sun3/E if someone wanted to. + * + * Note: Both supported variants of the Sun SCSI-3 adapter have + * some really unusual "features" for this driver to deal with, + * generally related to the DMA engine. The OBIO variant will + * ignore any attempt to write the FIFO count register while the + * SCSI bus is in DATA_IN or DATA_OUT phase. This is dealt with + * by setting the FIFO count early in COMMAND or MSG_IN phase. + * + * The VME variant has a bit to enable or disable the DMA engine, + * but that bit also gates the interrupt line from the NCR5380! + * Therefore, in order to get any interrupt from the 5380, (i.e. + * for reselect) one must clear the DMA engine transfer count and + * then enable DMA. This has the further complication that you + * CAN NOT touch the NCR5380 while the DMA enable bit is set, so + * we have to turn DMA back off before we even look at the 5380. + * + * What wonderfully whacky hardware this is! + * + * Credits, history: + * + * David Jones wrote the initial version of this module, which + * included support for the VME adapter only. (no reselection). + * + * Gordon Ross added support for the OBIO adapter, and re-worked + * both the VME and OBIO code to support disconnect/reselect. + * (Required figuring out the hardware "features" noted above.) + * + * The autoconfiguration boilerplate came from Adam Glass. + * + * VS2000: + */ + +#include <sys/param.h> +#include <sys/systm.h> +#include <sys/kernel.h> +#include <sys/conf.h> +#include <sys/file.h> +#include <sys/stat.h> +#include <sys/ioctl.h> +#include <sys/buf.h> +#include <sys/proc.h> +#include <sys/user.h> +#include <sys/map.h> +#include <sys/device.h> +#include <sys/dkstat.h> +#include <sys/disklabel.h> +#include <sys/disk.h> +#include <sys/syslog.h> + +/* #include <sys/errno.h> */ + +#include <scsi/scsi_all.h> +#include <scsi/scsi_debug.h> +#include <scsi/scsiconf.h> + +#include <machine/uvax.h> +#include <machine/ka410.h> +#include <machine/ka43.h> +#include <machine/vsbus.h> /* struct confargs */ + +#include <dev/ic/ncr5380reg.h> +#include <dev/ic/ncr5380var.h> + +#define trace(x) +#define debug(x) + +#ifndef NCR5380_CSRBITS +#define NCR5380_CSRBITS \ + "\020\010DEND\007DREQ\006PERR\005IREQ\004MTCH\003DCON\002ATN\001ACK" +#endif + +#ifndef NCR5380_BUSCSRBITS +#define NCR5380_BUSCSRBITS \ + "\020\010RST\007BSY\006REQ\005MSG\004C/D\003I/O\002SEL\001DBP" +#endif + +#include "ncr.h" + +#ifdef DDB +#define integrate +#else +#define integrate static +#endif + +/* + * Transfers smaller than this are done using PIO + * (on assumption they're not worth DMA overhead) + */ +#define MIN_DMA_LEN 128 + +/* + * Transfers lager than 65535 bytes need to be split-up. + * (Some of the FIFO logic has only 16 bits counters.) + * Make the size an integer multiple of the page size + * to avoid buf/cluster remap problems. (paranoid?) + * + * bertram: VS2000 has an DMA-area which is 16KB, thus + * have a maximum DMA-size of 16KB... + */ +#ifdef DMA_SHARED +#define MAX_DMA_LEN 0x2000 /* (8 * 1024) */ +#define DMA_ADDR_HBYTE 0x20 +#define DMA_ADDR_LBYTE 0x00 +#else +#define MAX_DMA_LEN 0x4000 /* (16 * 1024) */ +#define DMA_ADDR_HBYTE 0x00 +#define DMA_ADDR_LBYTE 0x00 +#endif + +#ifdef DEBUG +int si_debug = 3; +static int si_link_flags = 0 /* | SDEV_DB2 */ ; +#endif + +/* + * This structure is used to keep track of mappedpwd DMA requests. + * Note: combined the UDC command block with this structure, so + * the array of these has to be in DVMA space. + */ +struct si_dma_handle { + int dh_flags; +#define SIDH_BUSY 1 /* This DH is in use */ +#define SIDH_OUT 2 /* DMA does data out (write) */ +#define SIDH_PHYS 4 +#define SIDH_DONE 8 + u_char * dh_addr; /* KVA of start of buffer */ + int dh_maplen; /* Length of KVA mapping. */ + u_char * dh_dvma; /* VA of buffer in DVMA space */ + int dh_xlen; +}; + +/* + * The first structure member has to be the ncr5380_softc + * so we can just cast to go back and fourth between them. + */ +struct si_softc { + struct ncr5380_softc ncr_sc; + volatile struct si_regs *sc_regs; /* do we really need this? */ + + struct si_dma_handle *sc_dma; + struct confargs *sc_cfargs; + + int sc_xflags; /* ka410/ka43: resid, sizeof(areg) */ + + char *sc_dbase; + int sc_dsize; + + volatile char *sc_dareg; + volatile short *sc_dcreg; + volatile char *sc_ddreg; + volatile int sc_dflags; + +#define VSDMA_LOCKED 0x80 /* */ +#define VSDMA_WANTED 0x40 /* */ +#define VSDMA_IWANTED 0x20 +#define VSDMA_BLOCKED 0x10 +#define VSDMA_DMABUSY 0x08 /* DMA in progress */ +#define VSDMA_REGBUSY 0x04 /* accessing registers */ +#define VSDMA_WRBUF 0x02 /* writing to bounce-buffer */ +#define VSDMA_RDBUF 0x01 /* reading from bounce-buffer */ + +#define VSDMA_STATUS 0xF0 +#define VSDMA_LCKTYPE 0x0F + +#ifdef POLL_MODE + volatile u_char *intreq; + volatile u_char *intclr; + volatile u_char *intmsk; + volatile int intbit; +#endif +}; + +extern int cold; /* enable polling while cold-flag set */ + +/* Options. Interesting values are: 1,3,7 */ +int si_options = 3; /* bertram: 3 or 7 ??? */ +#define SI_ENABLE_DMA 1 /* Use DMA (maybe polled) */ +#define SI_DMA_INTR 2 /* DMA completion interrupts */ +#define SI_DO_RESELECT 4 /* Allow disconnect/reselect */ + +#define DMA_DIR_IN 1 +#define DMA_DIR_OUT 0 + +/* How long to wait for DMA before declaring an error. */ +int si_dma_intr_timo = 500; /* ticks (sec. X 100) */ + +integrate char si_name[] = "ncr"; +integrate int si_match(); +integrate void si_attach(); +integrate int si_intr __P((void *)); + +integrate void si_minphys __P((struct buf *bp)); +integrate void si_reset_adapter __P((struct ncr5380_softc *sc)); + +void si_dma_alloc __P((struct ncr5380_softc *)); +void si_dma_free __P((struct ncr5380_softc *)); +void si_dma_poll __P((struct ncr5380_softc *)); + +void si_intr_on __P((struct ncr5380_softc *)); +void si_intr_off __P((struct ncr5380_softc *)); + +int si_dmaLockBus __P((struct ncr5380_softc *, int)); +int si_dmaToggleLock __P((struct ncr5380_softc *, int, int)); +int si_dmaReleaseBus __P((struct ncr5380_softc *, int)); + +void si_dma_setup __P((struct ncr5380_softc *)); +void si_dma_start __P((struct ncr5380_softc *)); +void si_dma_eop __P((struct ncr5380_softc *)); +void si_dma_stop __P((struct ncr5380_softc *)); + +static struct scsi_adapter si_ops = { + ncr5380_scsi_cmd, /* scsi_cmd() */ + si_minphys, /* scsi_minphys() */ + NULL, /* open_target_lu() */ + NULL, /* close_target_lu() */ +}; + +/* This is copied from julian's bt driver */ +/* "so we have a default dev struct for our link struct." */ +static struct scsi_device si_dev = { + NULL, /* Use default error handler. */ + NULL, /* Use default start handler. */ + NULL, /* Use default async handler. */ + NULL, /* Use default "done" routine. */ +}; + + +struct cfdriver ncr_cd = { + NULL, si_name, DV_DULL +}; +struct cfattach ncr_ca = { + sizeof(struct si_softc), si_match, si_attach, +}; + +void +dk_establish(p,q) + struct disk *p; + struct device *q; +{ +#if 0 + printf ("faking dk_establish()...\n"); +#endif +} + + +integrate int +si_match(parent, match, aux) + struct device *parent; + void *match, *aux; +{ + struct cfdata *cf = match; + struct confargs *ca = aux; + + trace(("ncr_match(0x%x, %d, %s)\n", parent, cf->cf_unit, ca->ca_name)); + + if (strcmp(ca->ca_name, "ncr") && + strcmp(ca->ca_name, "ncr5380") && + strcmp(ca->ca_name, "NCR5380")) + return (0); + + /* + * we just define it being there ... + */ + return (1); +} + +integrate void +si_set_portid(pid,port) + int pid; + int port; +{ + struct { + u_long :2; + u_long id0:3; + u_long id1:3; + u_long :26; + } *p; + +#ifdef DEBUG + int *ip; + ip = (void*)uvax_phys2virt(KA410_SCSIPORT); + p = (void*)uvax_phys2virt(KA410_SCSIPORT); + printf("scsi-id: (%x/%d) %d / %d\n", *ip, *ip, p->id0, p->id1); +#endif + + p = (void*)uvax_phys2virt(KA410_SCSIPORT); + switch (port) { + case 0: + p->id0 = pid; + printf(": scsi-id %d\n", p->id0); + break; + case 1: + p->id1 = pid; + printf(": scsi-id %d\n", p->id1); + break; + default: + printf("invalid port-number %d\n", port); + } +} + +integrate void +si_attach(parent, self, aux) + struct device *parent, *self; + void *aux; +{ + struct si_softc *sc = (struct si_softc *) self; + struct ncr5380_softc *ncr_sc = (struct ncr5380_softc *)sc; + volatile struct si_regs *regs; + struct confargs *ca = aux; + int i; + int *ip = aux;; + + trace (("ncr_attach(0x%x, 0x%x, %s)\n", parent, self, ca->ca_name)); + + /* + * + */ +#ifdef POLL_MODE + sc->intreq = (void*)uvax_phys2virt(KA410_INTREQ); + sc->intmsk = (void*)uvax_phys2virt(KA410_INTMSK); + sc->intclr = (void*)uvax_phys2virt(KA410_INTCLR); + sc->intbit = ca->ca_intbit; +#endif + + sc->sc_cfargs = ca; /* needed for interrupt-setup */ + + regs = (void*)uvax_phys2virt(ca->ca_ioaddr); + + sc->sc_dareg = (void*)uvax_phys2virt(ca->ca_dareg); + sc->sc_dcreg = (void*)uvax_phys2virt(ca->ca_dcreg); + sc->sc_ddreg = (void*)uvax_phys2virt(ca->ca_ddreg); + sc->sc_dbase = (void*)uvax_phys2virt(ca->ca_dbase); + sc->sc_dsize = ca->ca_dsize; + sc->sc_dflags = 4; /* XXX */ + sc->sc_xflags = ca->ca_dflag; /* should/will be renamed */ + /* + * Fill in the prototype scsi_link. + */ + ncr_sc->sc_link.channel = SCSI_CHANNEL_ONLY_ONE; + ncr_sc->sc_link.adapter_softc = sc; + ncr_sc->sc_link.adapter_target = ca->ca_idval; + ncr_sc->sc_link.adapter = &si_ops; + ncr_sc->sc_link.device = &si_dev; + + si_set_portid(ca->ca_idval, ncr_sc->sc_dev.dv_unit); + + /* + * Initialize fields used by the MI code + */ + ncr_sc->sci_r0 = (void*)®s->sci.sci_r0; + ncr_sc->sci_r1 = (void*)®s->sci.sci_r1; + ncr_sc->sci_r2 = (void*)®s->sci.sci_r2; + ncr_sc->sci_r3 = (void*)®s->sci.sci_r3; + ncr_sc->sci_r4 = (void*)®s->sci.sci_r4; + ncr_sc->sci_r5 = (void*)®s->sci.sci_r5; + ncr_sc->sci_r6 = (void*)®s->sci.sci_r6; + ncr_sc->sci_r7 = (void*)®s->sci.sci_r7; + + /* + * MD function pointers used by the MI code. + */ + ncr_sc->sc_pio_out = ncr5380_pio_out; + ncr_sc->sc_pio_in = ncr5380_pio_in; + ncr_sc->sc_dma_alloc = si_dma_alloc; + ncr_sc->sc_dma_free = si_dma_free; + ncr_sc->sc_dma_poll = si_dma_poll; /* si_dma_poll not used! */ + ncr_sc->sc_intr_on = si_intr_on; /* vsbus_unlockDMA; */ + ncr_sc->sc_intr_off = si_intr_off; /* vsbus_lockDMA; */ + + ncr_sc->sc_dma_setup = NULL; /* si_dma_setup not used! */ + ncr_sc->sc_dma_start = si_dma_start; + ncr_sc->sc_dma_eop = NULL; + ncr_sc->sc_dma_stop = si_dma_stop; + + ncr_sc->sc_flags = 0; + if (si_options & SI_DO_RESELECT) + ncr_sc->sc_flags |= NCR5380_PERMIT_RESELECT; + if ((si_options & SI_DMA_INTR) == 0) + ncr_sc->sc_flags |= NCR5380_FORCE_POLLING; + ncr_sc->sc_min_dma_len = MIN_DMA_LEN; + + /* + * Initialize fields used only here in the MD code. + */ + i = SCI_OPENINGS * sizeof(struct si_dma_handle); + sc->sc_dma = (struct si_dma_handle *) malloc(i); + if (sc->sc_dma == NULL) + panic("si: dvma_malloc failed\n"); + for (i = 0; i < SCI_OPENINGS; i++) + sc->sc_dma[i].dh_flags = 0; + + sc->sc_regs = regs; + +#ifdef DEBUG + if (si_debug) + printf("si: Set TheSoftC=%x TheRegs=%x\n", sc, regs); + ncr_sc->sc_link.flags |= si_link_flags; +#endif + + /* + * Initialize si board itself. + */ + si_reset_adapter(ncr_sc); + ncr5380_init(ncr_sc); + ncr5380_reset_scsibus(ncr_sc); + config_found(self, &(ncr_sc->sc_link), scsiprint); + + /* + * Now ready for interrupts. + */ + vsbus_intr_register(sc->sc_cfargs, si_intr, (void *)sc); + vsbus_intr_enable(sc->sc_cfargs); +} + +integrate void +si_minphys(struct buf *bp) +{ + debug(("minphys: blkno=%d, bcount=%d, data=0x%x, flags=%x\n", + bp->b_blkno, bp->b_bcount, bp->b_data, bp->b_flags)); + + if (bp->b_bcount > MAX_DMA_LEN) { +#ifdef DEBUG + if (si_debug) { + printf("si_minphys len = 0x%x.\n", bp->b_bcount); + Debugger(); + } +#endif + bp->b_bcount = MAX_DMA_LEN; + } + return (minphys(bp)); +} + + +#define CSR_WANT (SI_CSR_SBC_IP | SI_CSR_DMA_IP | \ + SI_CSR_DMA_CONFLICT | SI_CSR_DMA_BUS_ERR ) + +static int si_intrCount = 0; +static int lastCSR = 0; + +integrate int +si_intr(arg) + void *arg; +{ + struct ncr5380_softc *ncr_sc = arg; + struct si_softc *sc = arg; + int count, claimed; + + count = ++si_intrCount; + trace(("%s: si-intr(%d).....\n", ncr_sc->sc_dev.dv_xname, count)); + +#ifdef DEBUG + /* + * Each DMA interrupt is followed by one spurious(?) interrupt. + * if (ncr_sc->sc_state & NCR_WORKING == 0) we know, that the + * interrupt was not claimed by the higher-level routine, so that + * it might be save to ignore these... + */ + if ((ncr_sc->sc_state & NCR_DOINGDMA) == 0) { + printf("spurious(%d): %x, %d, status=%b\n", count, + sc->sc_dflags, ncr_sc->sc_ncmds, + *ncr_sc->sci_csr, NCR5380_CSRBITS); + } +#endif + /* + * If there was a DMA operation in progress, now it's no longer + * active, since whatever caused the interrupt also interrupted + * the DMA operation. Thus accessing the registers now doesn't + * harm anything which is not yet broken... + */ + debug(("si_intr(status: %x, dma-count: %d)\n", + *ncr_sc->sci_csr, *sc->sc_dcreg)); + + /* + * First check for DMA errors / incomplete transfers + * If operation was read/data-in, the copy data from buffer + */ + if (ncr_sc->sc_state & NCR_DOINGDMA) { + struct sci_req *sr = ncr_sc->sc_current; + struct si_dma_handle *dh = sr->sr_dma_hand; + int resid, ntrans; + + resid = *sc->sc_dcreg; + if (resid == 1 && sc->sc_xflags) { + debug(("correcting resid...\n")); + resid = 0; + } + ntrans = dh->dh_xlen + resid; + if (resid == 0) { + if ((dh->dh_flags & SIDH_OUT) == 0) { + si_dmaToggleLock(ncr_sc, + VSDMA_DMABUSY, VSDMA_RDBUF); + bcopy(sc->sc_dbase, dh->dh_dvma, ntrans); + si_dmaToggleLock(ncr_sc, + VSDMA_RDBUF, VSDMA_DMABUSY); + dh->dh_flags |= SIDH_DONE; + } + } + else { +#ifdef DEBUG + int csr = *ncr_sc->sci_csr; + printf("DMA incomplete (%d/%d) status = %b\n", + ntrans, resid, csr, NCR5380_CSRBITS); + if(csr != lastCSR) { + int k = (csr & ~lastCSR) | (~csr & lastCSR); + debug(("Changed status bits: %b\n", + k, NCR5380_CSRBITS)); + lastCSR = csr & 0xFF; + } +#endif + printf("DMA incomplete: ntrans=%d/%d, lock=%x\n", + ntrans, dh->dh_xlen, sc->sc_dflags); + ncr_sc->sc_state |= NCR_ABORTING; + } + + if ((sc->sc_dflags & VSDMA_BLOCKED) == 0) { + printf("not blocked during DMA.\n"); + } + sc->sc_dflags &= ~VSDMA_BLOCKED; + si_dmaReleaseBus(ncr_sc, VSDMA_DMABUSY); + } + if ((sc->sc_dflags & VSDMA_BLOCKED) != 0) { + printf("blocked while not doing DMA.\n"); + sc->sc_dflags &= ~VSDMA_BLOCKED; + } + + /* + * Now, whatever it was, let the ncr5380sbc routine handle it... + */ + claimed = ncr5380_intr(ncr_sc); +#ifdef DEBUG + if (!claimed) { + printf("si_intr: spurious from SBC\n"); + if (si_debug & 4) { + Debugger(); /* XXX */ + } + } +#endif + trace(("%s: si-intr(%d) done, claimed=%d\n", + ncr_sc->sc_dev.dv_xname, count, claimed)); + return (claimed); +} + + +integrate void +si_reset_adapter(struct ncr5380_softc *ncr_sc) +{ + struct si_softc *sc = (struct si_softc *)ncr_sc; + volatile struct si_regs *si = sc->sc_regs; + +#ifdef DEBUG + if (si_debug) { + printf("si_reset_adapter\n"); + } +#endif + SCI_CLR_INTR(ncr_sc); +} + + +/***************************************************************** + * Common functions for DMA + ****************************************************************/ + +/* + * Allocate a DMA handle and put it in sc->sc_dma. Prepare + * for DMA transfer. On the Sun3, this means mapping the buffer + * into DVMA space. dvma_mapin() flushes the cache for us. + */ +void +si_dma_alloc(ncr_sc) + struct ncr5380_softc *ncr_sc; +{ + struct si_softc *sc = (struct si_softc *)ncr_sc; + struct sci_req *sr = ncr_sc->sc_current; + struct scsi_xfer *xs = sr->sr_xs; + struct buf *bp = sr->sr_xs->bp; + struct si_dma_handle *dh; + int i, xlen; + u_long addr; + + trace (("si_dma_alloc()\n")); + +#ifdef DIAGNOSTIC + if (sr->sr_dma_hand != NULL) + panic("si_dma_alloc: already have DMA handle"); +#endif + + addr = (u_long) ncr_sc->sc_dataptr; + debug(("addr=%x, dataptr=%x\n", addr, ncr_sc->sc_dataptr)); + xlen = ncr_sc->sc_datalen; + + /* Make sure our caller checked sc_min_dma_len. */ + if (xlen < MIN_DMA_LEN) + panic("si_dma_alloc: xlen=0x%x\n", xlen); + + /* + * Never attempt single transfers of more than 63k, because + * our count register may be only 16 bits (an OBIO adapter). + * This should never happen since already bounded by minphys(). + * XXX - Should just segment these... + */ + if (xlen > MAX_DMA_LEN) { + printf("si_dma_alloc: excessive xlen=0x%x\n", xlen); + Debugger(); + ncr_sc->sc_datalen = xlen = MAX_DMA_LEN; + } + + /* Find free DMA handle. Guaranteed to find one since we have + as many DMA handles as the driver has processes. */ + for (i = 0; i < SCI_OPENINGS; i++) { + if ((sc->sc_dma[i].dh_flags & SIDH_BUSY) == 0) + goto found; + } + panic("si: no free DMA handles."); +found: + + dh = &sc->sc_dma[i]; + dh->dh_flags = SIDH_BUSY; + dh->dh_addr = (u_char*) addr; + dh->dh_maplen = xlen; + dh->dh_xlen = xlen; + dh->dh_dvma = 0; + + /* Copy the "write" flag for convenience. */ + if (xs->flags & SCSI_DATA_OUT) + dh->dh_flags |= SIDH_OUT; + +#if 1 + /* + * If the buffer has the flag B_PHYS, the the address specified + * in the buffer is a user-space address and we need to remap + * this address into kernel space so that using this buffer + * within the interrupt routine will work. + * If it's already a kernel space address, we need to make sure + * that all pages are in-core. the mapin() routine takes care + * of that. + */ + if (bp && (bp->b_flags & B_PHYS)) + dh->dh_flags |= SIDH_PHYS; +#endif + + if (!bp) { + printf("ncr.c: struct buf *bp is null-pointer.\n"); + dh->dh_flags = 0; + return; + } + if (bp->b_bcount < 0 || bp->b_bcount > MAX_DMA_LEN) { + printf("ncr.c: invalid bcount %d (0x%x)\n", + bp->b_bcount, bp->b_bcount); + dh->dh_flags = 0; + return; + } + dh->dh_dvma = bp->b_data; +#if 0 + /* + * mapping of user-space addresses is no longer neccessary, now + * that the vmapbuf/vunmapbuf routines exist. Now the higher-level + * driver already cares for the mapping! + */ + if (bp->b_flags & B_PHYS) { + xdebug(("not mapping in... %x/%x %x\n", bp->b_saveaddr, + bp->b_data, bp->b_bcount)); +#ifdef USE_VMAPBUF + dh->dh_addr = bp->b_data; + dh->dh_maplen = bp->b_bcount; + vmapbuf(bp, bp->b_bcount); + dh->dh_dvma = bp->b_data; +#else + dh->dh_dvma = (u_char*)vsdma_mapin(bp); +#endif + xdebug(("addr %x, maplen %d, dvma %x, bcount %d, dir %s\n", + dh->dh_addr, dh->dh_maplen, dh->dh_dvma, bp->b_bcount, + (dh->dh_flags & SIDH_OUT ? "OUT" : "IN"))); + } +#endif + /* success */ + sr->sr_dma_hand = dh; + + return; +} + + +void +si_dma_free(ncr_sc) + struct ncr5380_softc *ncr_sc; +{ + struct si_softc *sc = (struct si_softc *)ncr_sc; + struct sci_req *sr = ncr_sc->sc_current; + struct scsi_xfer *xs = sr->sr_xs; + struct buf *bp = sr->sr_xs->bp; + struct si_dma_handle *dh = sr->sr_dma_hand; + + trace (("si_dma_free()\n")); + +#ifdef DIAGNOSTIC + if (dh == NULL) + panic("si_dma_free: no DMA handle"); +#endif + + if (ncr_sc->sc_state & NCR_DOINGDMA) + panic("si_dma_free: free while in progress"); + + if (dh->dh_flags & SIDH_BUSY) { +#if 0 + debug(("bp->b_flags=0x%x\n", bp->b_flags)); + if (bp->b_flags & B_PHYS) { +#ifdef USE_VMAPBUF + printf("not unmapping(%x/%x %x/%x %d/%d)...\n", + dh->dh_addr, dh->dh_dvma, + bp->b_saveaddr, bp->b_data, + bp->b_bcount, dh->dh_maplen); + /* vunmapbuf(bp, dh->dh_maplen); */ + printf("done.\n"); +#endif + dh->dh_dvma = 0; + } +#endif + dh->dh_flags = 0; + } + sr->sr_dma_hand = NULL; +} + + +/* + * REGBUSY and DMABUSY won't collide since the higher-level driver + * issues intr_on/intr_off before/after doing DMA. The only problem + * is to handle RDBUF/WRBUF wrt REGBUSY/DMABUSY + * + * There might be race-conditions, but for now we don't care for them... + */ +int +si_dmaLockBus(ncr_sc, lt) + struct ncr5380_softc *ncr_sc; + int lt; /* Lock-Type */ +{ + struct si_softc *sc = (void*)ncr_sc; + int timeout = 200; /* wait .2 seconds max. */ + + trace(("si_dmaLockBus(%x), cold: %d, current: %x\n", + lt, cold, sc->sc_dflags)); + +#ifdef POLL_MODE + if (cold) + return (0); +#endif + + if ((ncr_sc->sc_current != NULL) && (lt == VSDMA_REGBUSY)) { + printf("trying to use regs while sc_current is set.\n"); + printf("lt=%x, fl=%x, cur=%x\n", + lt, sc->sc_dflags, ncr_sc->sc_current); + } + if ((ncr_sc->sc_current == NULL) && (lt != VSDMA_REGBUSY)) { + printf("trying to use/prepare DMA without current.\n"); + printf("lt=%x, fl=%x, cur=%x\n", + lt, sc->sc_dflags, ncr_sc->sc_current); + } + + if ((sc->sc_dflags & VSDMA_LOCKED) == 0) { + struct si_softc *sc = (struct si_softc *)ncr_sc; + sc->sc_dflags |= VSDMA_WANTED; + vsbus_lockDMA(sc->sc_cfargs); + sc->sc_dflags = VSDMA_LOCKED | lt; + return (0); + } + +#if 1 + while ((sc->sc_dflags & VSDMA_LCKTYPE) != lt) { + debug(("busy wait(1)...\n")); + if (--timeout == 0) { + printf("timeout in busy-wait(%x %x)\n", + lt, sc->sc_dflags); + sc->sc_dflags &= ~VSDMA_LCKTYPE; + break; + } + delay(1000); + } + debug(("busy wait(1) done.\n")); + sc->sc_dflags |= lt; + +#else + if ((sc->sc_dflags & VSDMA_LCKTYPE) != lt) { + switch (lt) { + + case VSDMA_RDBUF: + /* sc->sc_dflags |= VSDMA_IWANTED; */ + debug(("busy wait(1)...\n")); + while (sc->sc_dflags & + (VSDMA_WRBUF | VSDMA_DMABUSY)) { + if (--timeout == 0) { + printf("timeout in busy-wait(1)\n"); + sc->sc_dflags &= ~VSDMA_WRBUF; + sc->sc_dflags &= ~VSDMA_DMABUSY; + } + delay(1000); + } + /* sc->sc_dflags &= ~VSDMA_IWANTED; */ + debug(("busy wait(1) done.\n")); + sc->sc_dflags |= lt; + break; + + case VSDMA_WRBUF: + /* sc->sc_dflags |= VSDMA_IWANTED; */ + debug(("busy wait(2)...\n")); + while (sc->sc_dflags & + (VSDMA_RDBUF | VSDMA_DMABUSY)) { + if (--timeout == 0) { + printf("timeout in busy-wait(2)\n"); + sc->sc_dflags &= ~VSDMA_RDBUF; + sc->sc_dflags &= ~VSDMA_DMABUSY; + } + delay(1000); + } + /* sc->sc_dflags &= ~VSDMA_IWANTED; */ + debug(("busy wait(2) done.\n")); + sc->sc_dflags |= lt; + break; + + case VSDMA_DMABUSY: + /* sc->sc_dflags |= VSDMA_IWANTED; */ + debug(("busy wait(3)...\n")); + while (sc->sc_dflags & + (VSDMA_RDBUF | VSDMA_WRBUF)) { + if (--timeout == 0) { + printf("timeout in busy-wait(3)\n"); + sc->sc_dflags &= ~VSDMA_RDBUF; + sc->sc_dflags &= ~VSDMA_WRBUF; + } + delay(1000); + } + /* sc->sc_dflags &= ~VSDMA_IWANTED; */ + debug(("busy wait(3) done.\n")); + sc->sc_dflags |= lt; + break; + + case VSDMA_REGBUSY: + /* sc->sc_dflags |= VSDMA_IWANTED; */ + debug(("busy wait(4)...\n")); + while (sc->sc_dflags & + (VSDMA_RDBUF | VSDMA_WRBUF | VSDMA_DMABUSY)) { + if (--timeout == 0) { + printf("timeout in busy-wait(4)\n"); + sc->sc_dflags &= ~VSDMA_RDBUF; + sc->sc_dflags &= ~VSDMA_WRBUF; + sc->sc_dflags &= ~VSDMA_DMABUSY; + } + delay(1000); + } + /* sc->sc_dflags &= ~VSDMA_IWANTED; */ + debug(("busy wait(4) done.\n")); + sc->sc_dflags |= lt; + break; + + default: + printf("illegal lockType %x in si_dmaLockBus()\n"); + } + } + else + printf("already locked. (%x/%x)\n", lt, sc->sc_dflags); +#endif + if (sc->sc_dflags & lt) /* successfully locked for this type */ + return (0); + + printf("spurious %x in si_dmaLockBus(%x)\n", lt, sc->sc_dflags); +} + +/* + * the lock of this type is no longer needed. If all (internal) locks are + * released, release the DMA bus. + */ +int +si_dmaReleaseBus(ncr_sc, lt) + struct ncr5380_softc *ncr_sc; + int lt; /* Lock-Type */ +{ + struct si_softc *sc = (void*)ncr_sc; + + trace(("si_dmaReleaseBus(%x), cold: %d, current: %x\n", + lt, cold, sc->sc_dflags)); + +#ifdef POLL_MODE + if (cold) + return (0); +#endif + + if ((sc->sc_dflags & VSDMA_LCKTYPE) == lt) { + sc->sc_dflags &= ~lt; + } + else + printf("trying to release %x while flags = %x\n", lt, + sc->sc_dflags); + + if (sc->sc_dflags == VSDMA_LOCKED) { /* no longer needed */ + struct si_softc *sc = (struct si_softc *)ncr_sc; + vsbus_unlockDMA(sc->sc_cfargs); + sc->sc_dflags = 0; + return (0); + } +} + +/* + * Just toggle the type of lock without releasing the lock... + * This is usually needed before/after bcopy() to/from DMA-buffer + */ +int +si_dmaToggleLock(ncr_sc, lt1, lt2) + struct ncr5380_softc *ncr_sc; + int lt1, lt2; /* Lock-Type */ +{ + struct si_softc *sc = (void*)ncr_sc; + +#ifdef POLL_MODE + if (cold) + return (0); +#endif + + if (((sc->sc_dflags & lt1) != 0) && + ((sc->sc_dflags & lt2) == 0)) { + sc->sc_dflags |= lt2; + sc->sc_dflags &= ~lt1; + return (0); + } + printf("cannot toggle locking from %x to %x (current = %x)\n", + lt1, lt2, sc->sc_dflags); +} + +/* + * This is called when the bus is going idle, + * so we want to enable the SBC interrupts. + * That is controlled by the DMA enable! + * Who would have guessed! + * What a NASTY trick! + */ +void +si_intr_on(ncr_sc) + struct ncr5380_softc *ncr_sc; +{ + si_dmaReleaseBus(ncr_sc, VSDMA_REGBUSY); +} + +/* + * This is called when the bus is idle and we are + * about to start playing with the SBC chip. + * + * VS2000 note: we have four kinds of access which are mutually exclusive: + * - access to the NCR5380 registers + * - access to the HDC9224 registers + * - access to the DMA area + * - doing DMA + */ +void +si_intr_off(ncr_sc) + struct ncr5380_softc *ncr_sc; +{ + si_dmaLockBus(ncr_sc, VSDMA_REGBUSY); +} + +/***************************************************************** + * VME functions for DMA + ****************************************************************/ + + +/* + * This function is called during the COMMAND or MSG_IN phase + * that preceeds a DATA_IN or DATA_OUT phase, in case we need + * to setup the DMA engine before the bus enters a DATA phase. + * + * XXX: The VME adapter appears to suppress SBC interrupts + * when the FIFO is not empty or the FIFO count is non-zero! + * + * On the VME version we just clear the DMA count and address + * here (to make sure it stays idle) and do the real setup + * later, in dma_start. + */ +void +si_dma_setup(ncr_sc) + struct ncr5380_softc *ncr_sc; +{ + trace (("si_dma_setup(ncr_sc) !!!\n")); + + /* + * VS2000: nothing to do ... + */ +} + + +void +si_dma_start(ncr_sc) + struct ncr5380_softc *ncr_sc; +{ + struct si_softc *sc = (struct si_softc *)ncr_sc; + struct sci_req *sr = ncr_sc->sc_current; + struct si_dma_handle *dh = sr->sr_dma_hand; + volatile struct si_regs *si = sc->sc_regs; + long data_pa; + int xlen; + + trace(("si_dma_start(%x)\n", sr->sr_dma_hand)); + + /* + * we always transfer from/to base of DMA-area, + * thus the DMA-address is always the same, only size + * and direction matter/differ on VS2000 + */ + + debug(("ncr_sc->sc_datalen = %d\n", ncr_sc->sc_datalen)); + xlen = ncr_sc->sc_datalen; + dh->dh_xlen = xlen; + + /* + * VS2000 has a fixed 16KB-area where DMA is restricted to. + * All DMA-addresses are relative to this base: KA410_DMA_BASE + * Thus we need to copy the data into this area when writing, + * or copy from this area when reading. (kind of bounce-buffer) + */ + + /* Set direction (send/recv) */ + if (dh->dh_flags & SIDH_OUT) { + /* + * We know that we are called while intr_off (regs locked) + * thus we toggle the lock from REGBUSY to WRBUF + * also we set the BLOCKIT flag, so that the locking of + * the DMA bus won't be released to the HDC9224... + */ + debug(("preparing msg-out (bcopy)\n")); + si_dmaToggleLock(ncr_sc, VSDMA_REGBUSY, VSDMA_WRBUF); + bcopy(dh->dh_dvma, sc->sc_dbase, xlen); + si_dmaToggleLock(ncr_sc, VSDMA_WRBUF, VSDMA_REGBUSY); + *sc->sc_ddreg = DMA_DIR_OUT; + } + else { + debug(("preparing data-in (bzero)\n")); + /* bzero(sc->sc_dbase, xlen); */ + *sc->sc_ddreg = DMA_DIR_IN; + } + sc->sc_dflags |= VSDMA_BLOCKED; + + *sc->sc_dareg = DMA_ADDR_HBYTE; /* high byte (6 bits) */ + *sc->sc_dareg = DMA_ADDR_LBYTE; /* low byte */ + *sc->sc_dcreg = 0 - xlen; /* bertram XXX */ + +#ifdef DEBUG + if (si_debug & 2) { + printf("si_dma_start: dh=0x%x, pa=0x%x, xlen=%d, creg=0x%x\n", + dh, data_pa, xlen, *sc->sc_dcreg); + } +#endif + +#ifdef POLL_MODE + debug(("dma_start: cold=%d\n", cold)); + if (cold) { + *sc->intmsk &= ~sc->intbit; + *sc->intclr = sc->intbit; + } + else + *sc->intmsk |= sc->intbit; +#endif + /* + * Acknowledge the phase change. (After DMA setup!) + * Put the SBIC into DMA mode, and start the transfer. + */ + si_dmaToggleLock(ncr_sc, VSDMA_REGBUSY, VSDMA_DMABUSY); + if (dh->dh_flags & SIDH_OUT) { + *ncr_sc->sci_tcmd = PHASE_DATA_OUT; + SCI_CLR_INTR(ncr_sc); + *ncr_sc->sci_icmd = SCI_ICMD_DATA; + *ncr_sc->sci_mode |= (SCI_MODE_DMA | SCI_MODE_DMA_IE); + *ncr_sc->sci_dma_send = 0; /* start it */ + } else { + *ncr_sc->sci_tcmd = PHASE_DATA_IN; + SCI_CLR_INTR(ncr_sc); + *ncr_sc->sci_icmd = 0; + *ncr_sc->sci_mode |= (SCI_MODE_DMA | SCI_MODE_DMA_IE); + *ncr_sc->sci_irecv = 0; /* start it */ + } + ncr_sc->sc_state |= NCR_DOINGDMA; + /* + * having a delay (eg. printf) here, seems to solve the problem. + * Isn't that strange ???? + * Maybe the higher-level driver accesses one of the registers of + * the controller while DMA is in progress. Having a long enough + * delay here might prevent/delay this access until DMA bus is + * free again... + * + * The instruction ++++ printf("DMA started.\n"); ++++ + * is long/slow enough, to make the SSCI driver work. Thus we + * try to find a delay() long/slow enough to do the same. The + * argument to this delay is relative to the transfer-count. + */ + delay(3*xlen/4); /* XXX solve this problem!!! XXX */ + +#ifdef DEBUG + if (si_debug & 2) { + printf("si_dma_start: started, flags=0x%x\n", + ncr_sc->sc_state); + } +#endif +} + + +void +si_vme_dma_eop(ncr_sc) + struct ncr5380_softc *ncr_sc; +{ + trace (("si_vme_dma_eop() !!!\n")); + /* Not needed - DMA was stopped prior to examining sci_csr */ +} + +/* + * si_dma_stop() has now become almost a nop-routine, since DMA-buffer + * has already been read within si_intr(), so there's nothing left to do. + */ +void +si_dma_stop(ncr_sc) + struct ncr5380_softc *ncr_sc; +{ + struct si_softc *sc = (struct si_softc *)ncr_sc; + struct sci_req *sr = ncr_sc->sc_current; + struct si_dma_handle *dh = sr->sr_dma_hand; + volatile struct si_regs *si = sc->sc_regs; + int resid, ntrans; + + if ((ncr_sc->sc_state & NCR_DOINGDMA) == 0) { +#ifdef DEBUG + printf("si_dma_stop: dma not running\n"); +#endif + return; + } + ncr_sc->sc_state &= ~NCR_DOINGDMA; + + /* Note that timeout may have set the error flag. */ + if (ncr_sc->sc_state & NCR_ABORTING) { + printf("si_dma_stop: timeout?\n"); + goto out; + } + + /* + * Now try to figure out how much actually transferred + */ + si_dmaLockBus(ncr_sc, VSDMA_DMABUSY); + si_dmaToggleLock(ncr_sc, VSDMA_DMABUSY, VSDMA_REGBUSY); + resid = *sc->sc_dcreg; + /* + * XXX: don't correct at two places !!! + */ + if (resid == 1 && sc->sc_xflags) { + resid = 0; + } + ntrans = dh->dh_xlen + resid; + if (resid != 0) + printf("resid=%d, xlen=%d, ntrans=%d\n", + resid, dh->dh_xlen, ntrans); + +#ifdef DEBUG + if (si_debug & 2) { + printf("si_dma_stop: resid=0x%x ntrans=0x%x\n", + resid, ntrans); + } +#endif + + if (ntrans < MIN_DMA_LEN) { + printf("si: fifo count: 0x%x\n", resid); + ncr_sc->sc_state |= NCR_ABORTING; + goto out; + } + if (ntrans > ncr_sc->sc_datalen) + panic("si_dma_stop: excess transfer"); + + /* + * On VS2000 in case of a READ-operation, we must now copy + * the buffer-contents to the destination-address! + */ + if ((dh->dh_flags & SIDH_OUT) == 0 && + (dh->dh_flags & SIDH_DONE) == 0) { + printf("DMA buffer not yet copied.\n"); + si_dmaToggleLock(ncr_sc, VSDMA_REGBUSY, VSDMA_RDBUF); + bcopy(sc->sc_dbase, dh->dh_dvma, ntrans); + si_dmaToggleLock(ncr_sc, VSDMA_RDBUF, VSDMA_REGBUSY); + } + si_dmaReleaseBus(ncr_sc, VSDMA_REGBUSY); + + /* Adjust data pointer */ + ncr_sc->sc_dataptr += ntrans; + ncr_sc->sc_datalen -= ntrans; + +out: + si_dmaLockBus(ncr_sc, VSDMA_DMABUSY); + + /* Put SBIC back in PIO mode. */ + *ncr_sc->sci_mode &= ~(SCI_MODE_DMA | SCI_MODE_DMA_IE); + *ncr_sc->sci_icmd = 0; + + si_dmaReleaseBus(ncr_sc, VSDMA_DMABUSY); +} + +/* + * Poll (spin-wait) for DMA completion. + * Called right after xx_dma_start(), and + * xx_dma_stop() will be called next. + */ +void +si_dma_poll(ncr_sc) + struct ncr5380_softc *ncr_sc; +{ + struct si_softc *sc = (struct si_softc *)ncr_sc; + struct sci_req *sr = ncr_sc->sc_current; + struct si_dma_handle *dh = sr->sr_dma_hand; + int i, timeout; + + if (! cold) + printf("spurious call of DMA-poll ???"); + +#ifdef POLL_MODE + + delay(10000); + trace(("si_dma_poll(%x)\n", *sc->sc_dcreg)); + + /* + * interrupt-request has been cleared by dma_start, thus + * we do nothing else but wait for the intreq to reappear... + */ + + timeout = 5000; + for (i=0; i<timeout; i++) { + if (*sc->intreq & sc->intbit) + break; + delay(100); + } + if ((*sc->intreq & sc->intbit) == 0) { + printf("si: DMA timeout (while polling)\n"); + /* Indicate timeout as MI code would. */ + sr->sr_flags |= SR_OVERDUE; + } +#endif + return; +} |