summaryrefslogtreecommitdiff
path: root/sys/arch/atari/dev/fd.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/arch/atari/dev/fd.c')
-rw-r--r--sys/arch/atari/dev/fd.c1202
1 files changed, 1202 insertions, 0 deletions
diff --git a/sys/arch/atari/dev/fd.c b/sys/arch/atari/dev/fd.c
new file mode 100644
index 00000000000..586e0b2f53a
--- /dev/null
+++ b/sys/arch/atari/dev/fd.c
@@ -0,0 +1,1202 @@
+/* $NetBSD: fd.c,v 1.10.2.1 1995/10/14 20:19:41 leo Exp $ */
+
+/*
+ * Copyright (c) 1995 Leo Weppelman.
+ * 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 Leo Weppelman.
+ * 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.
+ */
+
+/*
+ * This file contains a driver for the Floppy Disk Controller (FDC)
+ * on the Atari TT. It uses the WD 1772 chip, modified for steprates.
+ *
+ * The ST floppy disk controller shares the access to the DMA circuitry
+ * with other devices. For this reason the floppy disk controller makes
+ * use of some special DMA accessing code.
+ *
+ * Interrupts from the FDC are in fact DMA interrupts which get their
+ * first level handling in 'dma.c' . If the floppy driver is currently
+ * using DMA the interrupt is signalled to 'fdcint'.
+ *
+ * TODO:
+ * - Test it with 2 drives (I don't have them)
+ * - Test it with an HD-drive (Don't have that either)
+ * - Finish ioctl's
+ */
+
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/kernel.h>
+#include <sys/malloc.h>
+#include <sys/buf.h>
+#include <sys/device.h>
+#include <sys/ioctl.h>
+#include <sys/fcntl.h>
+#include <sys/conf.h>
+#include <sys/disklabel.h>
+#include <sys/disk.h>
+#include <sys/dkbad.h>
+#include <atari/atari/device.h>
+#include <machine/disklabel.h>
+#include <machine/iomap.h>
+#include <machine/mfp.h>
+#include <machine/dma.h>
+#include <machine/video.h>
+#include <atari/dev/fdreg.h>
+
+/*
+ * Be verbose for debugging
+ */
+/*#define FLP_DEBUG 1 */
+
+#define FDC_MAX_DMA_AD 0x1000000 /* No DMA possible beyond */
+
+/* Parameters for the disk drive. */
+#define SECTOR_SIZE 512 /* physical sector size in bytes */
+#define NR_DRIVES 2 /* maximum number of drives */
+#define NR_TYPES 3 /* number of diskette/drive combinations*/
+#define MAX_ERRORS 10 /* how often to try rd/wt before quitting*/
+#define STEP_DELAY 6000 /* 6ms (6000us) delay after stepping */
+
+
+#define INV_TRK 32000 /* Should fit in unsigned short */
+#define INV_PART NR_TYPES
+
+/*
+ * Driver states
+ */
+#define FLP_IDLE 0x00 /* floppy is idle */
+#define FLP_MON 0x01 /* idle with motor on */
+#define FLP_STAT 0x02 /* determine floppy status */
+#define FLP_XFER 0x04 /* read/write data from floppy */
+
+/*
+ * Timer delay's
+ */
+#define FLP_MONDELAY (3 * hz) /* motor-on delay */
+#define FLP_XFERDELAY (2 * hz) /* timeout on transfer */
+
+/*
+ * The density codes
+ */
+#define FLP_DD 0 /* Double density */
+#define FLP_HD 1 /* High density */
+
+
+#define b_block b_resid /* FIXME: this is not the place */
+
+/*
+ * Global data for all physical floppy devices
+ */
+static short selected = 0; /* drive/head currently selected*/
+static short motoron = 0; /* motor is spinning */
+static short nopens = 0; /* Number of opens executed */
+
+static short fd_state = FLP_IDLE; /* Current driver state */
+static int lock_stat= 0; /* dma locking status */
+static short fd_cmd = 0; /* command being executed */
+static char *fd_error= NULL; /* error from fd_xfer_ok() */
+
+/*
+ * Private per device data
+ */
+struct fd_softc {
+ struct dkdevice dkdev;
+ struct buf bufq; /* queue of buf's */
+ int unit; /* unit for atari controlling hw*/
+ int nheads; /* number of heads in use */
+ int nsectors; /* number of sectors/track */
+ int density; /* density code */
+ int nblocks; /* number of blocks on disk */
+ int curtrk; /* track head positioned on */
+ short flags; /* misc flags */
+ short part; /* Current open partition */
+ int sector; /* logical sector for I/O */
+ caddr_t io_data; /* KVA for data transfer */
+ int io_bytes; /* bytes left for I/O */
+ int io_dir; /* B_READ/B_WRITE */
+ int errcnt; /* current error count */
+ u_char *bounceb; /* Bounce buffer */
+
+};
+
+/*
+ * Flags in fd_softc:
+ */
+#define FLPF_NOTRESP 0x001 /* Unit not responding */
+#define FLPF_ISOPEN 0x002 /* Unit is open */
+#define FLPF_SPARE 0x004 /* Not used */
+#define FLPF_HAVELAB 0x008 /* We have a valid label */
+#define FLPF_BOUNCE 0x010 /* Now using the bounce buffer */
+#define FLPF_WRTPROT 0x020 /* Unit is write-protected */
+#define FLPF_EMPTY 0x040 /* Unit is empty */
+#define FLPF_INOPEN 0x080 /* Currently being opened */
+#define FLPF_GETSTAT 0x100 /* Getting unit status */
+
+struct fd_types {
+ int nheads; /* Heads in use */
+ int nsectors; /* sectors per track */
+ int nblocks; /* number of blocks */
+ int density; /* density code */
+} fdtypes[NR_TYPES] = {
+ { 1, 9, 720 , FLP_DD }, /* 360 Kb */
+ { 2, 9, 1440 , FLP_DD }, /* 720 Kb */
+ { 2, 18, 2880 , FLP_HD }, /* 1.44 Mb */
+};
+
+typedef void (*FPV)();
+
+/*
+ * Private drive functions....
+ */
+static void fdstart __P((struct fd_softc *));
+static void fddone __P((struct fd_softc *));
+static void fdstatus __P((struct fd_softc *));
+static void fd_xfer __P((struct fd_softc *));
+static void fdcint __P((struct fd_softc *));
+static int fd_xfer_ok __P((struct fd_softc *));
+static void fdmotoroff __P((struct fd_softc *));
+static void fdminphys __P((struct buf *));
+static void fdtestdrv __P((struct fd_softc *));
+static int fdgetdisklabel __P((struct fd_softc *, dev_t));
+static int fdselect __P((int, int, int));
+static void fddeselect __P((void));
+
+extern __inline__ u_char read_fdreg(u_short regno)
+{
+ DMA->dma_mode = regno;
+ return(DMA->dma_data);
+}
+
+extern __inline__ void write_fdreg(u_short regno, u_short val)
+{
+ DMA->dma_mode = regno;
+ DMA->dma_data = val;
+}
+
+extern __inline__ u_char read_dmastat(void)
+{
+ DMA->dma_mode = FDC_CS | DMA_SCREG;
+ return(DMA->dma_stat);
+}
+
+/*
+ * Autoconfig stuff....
+ */
+static int fdcmatch __P((struct device *, struct cfdata *, void *));
+static int fdcprint __P((void *, char *));
+static void fdcattach __P((struct device *, struct device *, void *));
+
+struct cfdriver fdccd = {
+ NULL, "fdc", (cfmatch_t)fdcmatch, fdcattach, DV_DULL,
+ sizeof(struct device), NULL, 0 };
+
+static int
+fdcmatch(pdp, cfp, auxp)
+struct device *pdp;
+struct cfdata *cfp;
+void *auxp;
+{
+ if(strcmp("fdc", auxp) || cfp->cf_unit != 0)
+ return(0);
+ return(1);
+}
+
+static void
+fdcattach(pdp, dp, auxp)
+struct device *pdp, *dp;
+void *auxp;
+{
+ struct fd_softc fdsoftc;
+ int i, nfound = 0;
+
+ printf("\n");
+ fddeselect();
+ for(i = 0; i < NR_DRIVES; i++) {
+
+ /*
+ * Test if unit is present
+ */
+ fdsoftc.unit = i;
+ fdsoftc.flags = 0;
+ st_dmagrab(fdcint, fdtestdrv, &fdsoftc, &lock_stat, 0);
+ st_dmafree(&fdsoftc, &lock_stat);
+
+ if(!(fdsoftc.flags & FLPF_NOTRESP)) {
+ nfound++;
+ config_found(dp, (void*)i, fdcprint);
+ }
+ }
+
+ if(nfound) {
+ /*
+ * enable disk related interrupts
+ */
+ MFP->mf_ierb |= IB_DINT;
+ MFP->mf_iprb &= ~IB_DINT;
+ MFP->mf_imrb |= IB_DINT;
+ }
+}
+
+static int
+fdcprint(auxp, pnp)
+void *auxp;
+char *pnp;
+{
+ return(UNCONF);
+}
+
+static int fdmatch __P((struct device *, struct cfdata *, void *));
+static void fdattach __P((struct device *, struct device *, void *));
+ void fdstrategy __P((struct buf *));
+struct dkdriver fddkdriver = { fdstrategy };
+
+struct cfdriver fdcd = {
+ NULL, "fd", (cfmatch_t)fdmatch, fdattach, DV_DISK,
+ sizeof(struct fd_softc), NULL, 0 };
+
+static int
+fdmatch(pdp, cfp, auxp)
+struct device *pdp;
+struct cfdata *cfp;
+void *auxp;
+{
+ int unit = (int)auxp;
+ return(1);
+}
+
+static void
+fdattach(pdp, dp, auxp)
+struct device *pdp, *dp;
+void *auxp;
+{
+ struct fd_softc *sc;
+
+ sc = (struct fd_softc *)dp;
+
+ printf("\n");
+
+ sc->dkdev.dk_driver = &fddkdriver;
+}
+
+fdioctl(dev, cmd, addr, flag, p)
+dev_t dev;
+u_long cmd;
+int flag;
+caddr_t addr;
+struct proc *p;
+{
+ struct fd_softc *sc;
+ void *data;
+
+ sc = getsoftc(fdcd, DISKUNIT(dev));
+
+ if((sc->flags & FLPF_HAVELAB) == 0)
+ return(EBADF);
+
+ switch(cmd) {
+ case DIOCSBAD:
+ return(EINVAL);
+ case DIOCGDINFO:
+ *(struct disklabel *)addr = sc->dkdev.dk_label;
+ return(0);
+ case DIOCGPART:
+ ((struct partinfo *)addr)->disklab =
+ &sc->dkdev.dk_label;
+ ((struct partinfo *)addr)->part =
+ &sc->dkdev.dk_label.d_partitions[DISKPART(dev)];
+ return(0);
+#ifdef notyet /* XXX LWP */
+ case DIOCSRETRIES:
+ case DIOCSSTEP:
+ case DIOCSDINFO:
+ case DIOCWDINFO:
+ case DIOCWLABEL:
+#endif /* notyet */
+ default:
+ return(ENOTTY);
+ }
+}
+
+/*
+ * Open the device. If this is the first open on both the floppy devices,
+ * intialize the controller.
+ * Note that partition info on the floppy device is used to distinguise
+ * between 780Kb and 360Kb floppy's.
+ * partition 0: 360Kb
+ * partition 1: 780Kb
+ */
+Fdopen(dev, flags, devtype, proc)
+dev_t dev;
+int flags, devtype;
+struct proc *proc;
+{
+ struct fd_softc *sc;
+ int sps;
+
+#ifdef FLP_DEBUG
+ printf("Fdopen dev=0x%x\n", dev);
+#endif
+
+ if(DISKPART(dev) >= NR_TYPES)
+ return(ENXIO);
+
+ if((sc = getsoftc(fdcd, DISKUNIT(dev))) == NULL)
+ return(ENXIO);
+
+ /*
+ * If no floppy currently open, reset the controller and select
+ * floppy type.
+ */
+ if(!nopens) {
+
+#ifdef FLP_DEBUG
+ printf("Fdopen device not yet open\n");
+#endif
+ nopens++;
+ write_fdreg(FDC_CS, IRUPT);
+ delay(40);
+ }
+
+ /*
+ * Sleep while other process is opening the device
+ */
+ sps = splbio();
+ while(sc->flags & FLPF_INOPEN)
+ tsleep((caddr_t)sc, PRIBIO, "Fdopen", 0);
+ splx(sps);
+
+ if(!(sc->flags & FLPF_ISOPEN)) {
+ /*
+ * Initialise some driver values.
+ */
+ int part = DISKPART(dev);
+ void *addr;
+
+ sc->bufq.b_actf = NULL;
+ sc->unit = DISKUNIT(dev);
+ sc->part = part;
+ sc->nheads = fdtypes[part].nheads;
+ sc->nsectors = fdtypes[part].nsectors;
+ sc->nblocks = fdtypes[part].nblocks;
+ sc->density = fdtypes[part].density;
+ sc->curtrk = INV_TRK;
+ sc->sector = 0;
+ sc->errcnt = 0;
+ sc->bounceb = (u_char*)alloc_stmem(SECTOR_SIZE, &addr);
+ if(sc->bounceb == NULL)
+ return(ENOMEM); /* XXX */
+
+ /*
+ * Go get write protect + loaded status
+ */
+ sc->flags |= FLPF_INOPEN|FLPF_GETSTAT;
+ sps = splbio();
+ st_dmagrab(fdcint, fdstatus, sc, &lock_stat, 0);
+ while(sc->flags & FLPF_GETSTAT)
+ tsleep((caddr_t)sc, PRIBIO, "Fdopen", 0);
+ splx(sps);
+ wakeup((caddr_t)sc);
+
+ if((sc->flags & FLPF_WRTPROT) && (flags & FWRITE)) {
+ sc->flags = 0;
+ return(EPERM);
+ }
+ if(sc->flags & FLPF_EMPTY) {
+ sc->flags = 0;
+ return(ENXIO);
+ }
+ sc->flags &= ~(FLPF_INOPEN|FLPF_GETSTAT);
+ sc->flags |= FLPF_ISOPEN;
+ }
+ else {
+ /*
+ * Multiply opens are granted when accessing the same type of
+ * floppy (eq. the same partition).
+ */
+ if(sc->part != DISKPART(dev))
+ return(ENXIO); /* XXX temporarely out of business */
+ }
+ fdgetdisklabel(sc, dev);
+#ifdef FLP_DEBUG
+ printf("Fdopen open succeeded on type %d\n", sc->part);
+#endif
+}
+
+fdclose(dev, flags, devtype, proc)
+dev_t dev;
+int flags, devtype;
+struct proc *proc;
+{
+ struct fd_softc *sc;
+
+ sc = getsoftc(fdcd, DISKUNIT(dev));
+ free_stmem(sc->bounceb);
+ sc->flags = 0;
+ nopens--;
+
+#ifdef FLP_DEBUG
+ printf("Closed floppy device -- nopens: %d\n", nopens);
+#endif
+ return(0);
+}
+
+void
+fdstrategy(bp)
+struct buf *bp;
+{
+ struct fd_softc *sc;
+ struct disklabel *lp;
+ int sps, nblocks;
+
+ sc = getsoftc(fdcd, DISKUNIT(bp->b_dev));
+
+#ifdef FLP_DEBUG
+ printf("fdstrategy: 0x%x\n", bp);
+#endif
+
+ /*
+ * check for valid partition and bounds
+ */
+ lp = &sc->dkdev.dk_label;
+ if ((sc->flags & FLPF_HAVELAB) == 0) {
+ bp->b_error = EIO;
+ goto bad;
+ }
+ if (bounds_check_with_label(bp, lp, 0) <= 0)
+ goto done;
+
+ if (bp->b_bcount == 0)
+ goto done;
+
+ /*
+ * queue the buf and kick the low level code
+ */
+ sps = splbio();
+ disksort(&sc->bufq, bp);
+ if (!lock_stat) {
+ if (fd_state & FLP_MON)
+ untimeout((FPV)fdmotoroff, (void*)sc);
+ fd_state = FLP_IDLE;
+ st_dmagrab(fdcint, fdstart, sc, &lock_stat, 0);
+ }
+ splx(sps);
+
+ return;
+bad:
+ bp->b_flags |= B_ERROR;
+done:
+ bp->b_resid = bp->b_bcount;
+ biodone(bp);
+}
+
+/*
+ * no dumps to floppy disks thank you.
+ */
+int
+fddump(dev_t dev)
+{
+ return(ENXIO);
+}
+
+/*
+ * no dumps to floppy disks thank you.
+ */
+int
+fdsize(dev)
+dev_t dev;
+{
+ return(-1);
+}
+
+int
+fdread(dev, uio)
+dev_t dev;
+struct uio *uio;
+{
+ return(physio(fdstrategy, NULL, dev, B_READ, fdminphys, uio));
+}
+
+int
+fdwrite(dev, uio)
+dev_t dev;
+struct uio *uio;
+{
+ return(physio(fdstrategy, NULL, dev, B_WRITE, fdminphys, uio));
+}
+
+/*
+ * Called through DMA-dispatcher, get status.
+ */
+static void
+fdstatus(sc)
+struct fd_softc *sc;
+{
+#ifdef FLP_DEBUG
+ printf("fdstatus\n");
+#endif
+ sc->errcnt = 0;
+ fd_state = FLP_STAT;
+ fd_xfer(sc);
+}
+
+/*
+ * Called through the dma-dispatcher. So we know we are the only ones
+ * messing with the floppy-controler.
+ * Initialize some fields in the fdsoftc for the state-machine and get
+ * it going.
+ */
+static void
+fdstart(sc)
+struct fd_softc *sc;
+{
+ struct buf *bp;
+
+ bp = sc->bufq.b_actf;
+ sc->sector = bp->b_blkno; /* Start sector for I/O */
+ sc->io_data = bp->b_data; /* KVA base for I/O */
+ sc->io_bytes = bp->b_bcount; /* Transfer size in bytes */
+ sc->io_dir = bp->b_flags & B_READ;/* Direction of transfer */
+ sc->errcnt = 0; /* No errors yet */
+ fd_state = FLP_XFER; /* Yes, we're going to transfer */
+
+ fd_xfer(sc);
+}
+
+/*
+ * The current transaction is finished (for good or bad). Let go of
+ * the the dma-resources. Call biodone() to finish the transaction.
+ * Find a new transaction to work on.
+ */
+static void
+fddone(sc)
+register struct fd_softc *sc;
+{
+ struct buf *bp, *dp;
+ struct fd_softc *sc1;
+ int i, sps;
+
+ /*
+ * Give others a chance to use the dma.
+ */
+ st_dmafree(sc, &lock_stat);
+
+
+ if(fd_state != FLP_STAT) {
+ /*
+ * Finish current transaction.
+ */
+ sps = splbio();
+ dp = &sc->bufq;
+ bp = dp->b_actf;
+ if(bp == NULL)
+ panic("fddone");
+ dp->b_actf = bp->b_actf;
+ splx(sps);
+
+#ifdef FLP_DEBUG
+ printf("fddone: unit: %d, buf: %x, resid: %d\n",sc->unit,bp,
+ sc->io_bytes);
+#endif
+ bp->b_resid = sc->io_bytes;
+ biodone(bp);
+ }
+ fd_state = FLP_MON;
+
+ if(lock_stat)
+ return; /* XXX Is this possible? */
+
+ /*
+ * Find a new transaction on round-robin basis.
+ */
+ for(i = sc->unit + 1; ;i++) {
+ if(i >= fdcd.cd_ndevs)
+ i = 0;
+ if((sc1 = fdcd.cd_devs[i]) == NULL)
+ continue;
+ if(sc1->bufq.b_actf)
+ break;
+ if(i == sc->unit) {
+ timeout((FPV)fdmotoroff, (void*)sc, FLP_MONDELAY);
+#ifdef FLP_DEBUG
+ printf("fddone: Nothing to do\n");
+#endif
+ return; /* No work */
+ }
+ }
+ fd_state = FLP_IDLE;
+#ifdef FLP_DEBUG
+ printf("fddone: Staring job on unit %d\n", sc1->unit);
+#endif
+ st_dmagrab(fdcint, fdstart, sc1, &lock_stat, 0);
+}
+
+static int
+fdselect(drive, head, dense)
+int drive, head, dense;
+{
+ int i, sps, spinning;
+#ifdef FLP_DEBUG
+ printf("fdselect: drive=%d, head=%d, dense=%d\n", drive, head, dense);
+#endif
+ i = ((drive == 1) ? PA_FLOP1 : PA_FLOP0) | head;
+ spinning = motoron;
+ motoron = 1;
+
+ switch(dense) {
+ case FLP_DD:
+ DMA->dma_drvmode = 0;
+ break;
+ case FLP_HD:
+ DMA->dma_drvmode = (FDC_HDSET|FDC_HDSIG);
+ break;
+ default:
+ panic("fdselect: unknown density code\n");
+ }
+ if(i != selected) {
+ sps = splhigh();
+
+ selected = i;
+ SOUND->sd_selr = YM_IOA;
+ SOUND->sd_wdat = (SOUND->sd_rdat & 0x78) | (i ^ 0x07);
+ splx(sps);
+ }
+ return(spinning);
+}
+
+static void
+fddeselect()
+{
+ int sps;
+
+ sps = splhigh();
+ SOUND->sd_selr = YM_IOA;
+ SOUND->sd_wdat = SOUND->sd_rdat | 0x07;
+ splx(sps);
+
+ motoron = selected = 0;
+ DMA->dma_drvmode = 0;
+}
+
+/****************************************************************************
+ * The following functions assume to be running as a result of a *
+ * disk-interrupt (e.q. spl = splbio). *
+ * They form the finit-state machine, the actual driver. *
+ * *
+ * fdstart()/ --> fd_xfer() -> activate hardware *
+ * fdopen() ^ *
+ * | *
+ * +-- not ready -<------------+ *
+ * | *
+ * fdmotoroff()/ --> fdcint() -> fd_xfer_ok() ---+ *
+ * h/w interrupt | *
+ * \|/ *
+ * finished ---> fdone() *
+ * *
+ ****************************************************************************/
+static void
+fd_xfer(sc)
+struct fd_softc *sc;
+{
+ register int head = 0;
+ register int track, sector, hbit;
+ int i;
+ u_long phys_addr;
+
+ switch(fd_state) {
+ case FLP_XFER:
+ /*
+ * Calculate head/track values
+ */
+ track = sc->sector / sc->nsectors;
+ head = track % sc->nheads;
+ track = track / sc->nheads;
+#ifdef FLP_DEBUG
+ printf("fd_xfer: sector:%d,head:%d,track:%d\n", sc->sector,head,
+ track);
+#endif
+ break;
+
+ case FLP_STAT:
+ /*
+ * FLP_STAT only wants to recalibrate
+ */
+ sc->curtrk = INV_TRK;
+ break;
+ default:
+ panic("fd_xfer: wrong state (0x%x)", fd_state);
+ }
+
+ /*
+ * Select the drive.
+ */
+ hbit = fdselect(sc->unit, head, sc->density) ? HBIT : 0;
+
+ if(sc->curtrk == INV_TRK) {
+ /*
+ * Recalibrate, since we lost track of head positioning.
+ * The floppy disk controller has no way of determining its
+ * absolute arm position (track). Instead, it steps the
+ * arm a track at a time and keeps track of where it
+ * thinks it is (in software). However, after a SEEK, the
+ * hardware reads information from the diskette telling
+ * where the arm actually is. If the arm is in the wrong place,
+ * a recalibration is done, which forces the arm to track 0.
+ * This way the controller can get back into sync with reality.
+ */
+ fd_cmd = RESTORE;
+ write_fdreg(FDC_CS, RESTORE|VBIT|hbit);
+ timeout((FPV)fdmotoroff, (void*)sc, FLP_XFERDELAY);
+
+#ifdef FLP_DEBUG
+ printf("fd_xfer:Recalibrating drive %d\n", sc->unit);
+#endif
+ return;
+ }
+
+ write_fdreg(FDC_TR, sc->curtrk);
+
+ /*
+ * Issue a SEEK command on the indicated drive unless the arm is
+ * already positioned on the correct track.
+ */
+ if(track != sc->curtrk) {
+ sc->curtrk = track; /* be optimistic */
+ write_fdreg(FDC_DR, track);
+ write_fdreg(FDC_CS, SEEK|RATE6|VBIT|hbit);
+ timeout((FPV)fdmotoroff, (void*)sc, FLP_XFERDELAY);
+ fd_cmd = SEEK;
+#ifdef FLP_DEBUG
+ printf("fd_xfer:Seek to track %d on drive %d\n",track,sc->unit);
+#endif
+ return;
+ }
+
+ /*
+ * The drive is now on the proper track. Read or write 1 block.
+ */
+ sector = sc->sector % sc->nsectors;
+ sector++; /* start numbering at 1 */
+
+ write_fdreg(FDC_SR, sector);
+
+ phys_addr = (u_long)kvtop(sc->io_data);
+ if(phys_addr >= FDC_MAX_DMA_AD) {
+ /*
+ * We _must_ bounce this address
+ */
+ phys_addr = (u_long)kvtop(sc->bounceb);
+ if(sc->io_dir == B_WRITE)
+ bcopy(sc->io_data, sc->bounceb, SECTOR_SIZE);
+ sc->flags |= FLPF_BOUNCE;
+ }
+ st_dmaaddr_set((caddr_t)phys_addr); /* DMA address setup */
+
+#ifdef FLP_DEBUG
+ printf("fd_xfer:Start io (io_addr:%x)\n", kvtop(sc->io_data));
+#endif
+
+ if(sc->io_dir == B_READ) {
+ /* Issue the command */
+ st_dmacomm(DMA_FDC | DMA_SCREG, 1);
+ write_fdreg(FDC_CS, F_READ|hbit);
+ fd_cmd = F_READ;
+ }
+ else {
+ /* Issue the command */
+ st_dmacomm(DMA_WRBIT | DMA_FDC | DMA_SCREG, 1);
+ write_fdreg(DMA_WRBIT | FDC_CS, F_WRITE|hbit|EBIT|PBIT);
+ fd_cmd = F_WRITE;
+ }
+ timeout((FPV)fdmotoroff, (void*)sc, FLP_XFERDELAY);
+}
+
+/* return values of fd_xfer_ok(): */
+#define X_OK 0
+#define X_AGAIN 1
+#define X_ERROR 2
+#define X_FAIL 3
+
+/*
+ * Hardware interrupt function.
+ */
+static void
+fdcint(sc)
+struct fd_softc *sc;
+{
+ struct buf *bp;
+
+#ifdef FLP_DEBUG
+ printf("fdcint: unit = %d\n", sc->unit);
+#endif
+
+ /*
+ * Cancel timeout (we made it, didn't we)
+ */
+ untimeout((FPV)fdmotoroff, (void*)sc);
+
+ switch(fd_xfer_ok(sc)) {
+ case X_ERROR :
+ if(++(sc->errcnt) < MAX_ERRORS) {
+ /*
+ * Command failed but still retries left.
+ */
+ break;
+ }
+ /* FALL THROUGH */
+ case X_FAIL :
+ /*
+ * Non recoverable error. Fall back to motor-on
+ * idle-state.
+ */
+ if(fd_error != NULL) {
+ printf("Floppy error: %s\n", fd_error);
+ fd_error = NULL;
+ }
+
+ if(fd_state == FLP_STAT) {
+ sc->flags |= FLPF_EMPTY;
+ sc->flags &= ~FLPF_GETSTAT;
+ wakeup((caddr_t)sc);
+ fddone(sc);
+ return;
+ }
+
+ bp = sc->bufq.b_actf;
+
+ bp->b_error = EIO;
+ bp->b_flags |= B_ERROR;
+ fd_state = FLP_MON;
+
+ break;
+ case X_AGAIN:
+ /*
+ * Start next part of state machine.
+ */
+ break;
+ case X_OK:
+ /*
+ * Command ok and finished. Reset error-counter.
+ * If there are no more bytes to transfer fall back
+ * to motor-on idle state.
+ */
+ sc->errcnt = 0;
+
+ if(fd_state == FLP_STAT) {
+ sc->flags &= ~FLPF_GETSTAT;
+ wakeup((caddr_t)sc);
+ fddone(sc);
+ return;
+ }
+
+ if((sc->flags & FLPF_BOUNCE) && (sc->io_dir == B_READ))
+ bcopy(sc->bounceb, sc->io_data, SECTOR_SIZE);
+ sc->flags &= ~FLPF_BOUNCE;
+
+ sc->sector++;
+ sc->io_data += SECTOR_SIZE;
+ sc->io_bytes -= SECTOR_SIZE;
+ if(sc->io_bytes <= 0)
+ fd_state = FLP_MON;
+ }
+ if(fd_state == FLP_MON)
+ fddone(sc);
+ else fd_xfer(sc);
+}
+
+/*
+ * Determine status of last command. Should only be called through
+ * 'fdcint()'.
+ * Returns:
+ * X_ERROR : Error on command; might succeed next time.
+ * X_FAIL : Error on command; will never succeed.
+ * X_AGAIN : Part of a command succeeded, call 'fd_xfer()' to complete.
+ * X_OK : Command succeeded and is complete.
+ *
+ * This function only affects sc->curtrk.
+ */
+static int
+fd_xfer_ok(sc)
+register struct fd_softc *sc;
+{
+ register int status;
+
+#ifdef FLP_DEBUG
+ printf("fd_xfer_ok: cmd: 0x%x, state: 0x%x\n", fd_cmd, fd_state);
+#endif
+ switch(fd_cmd) {
+ case IRUPT:
+ /*
+ * Timeout. Force a recalibrate before we try again.
+ */
+ status = read_fdreg(FDC_CS);
+
+ fd_error = "Timeout";
+ sc->curtrk = INV_TRK;
+ return(X_ERROR);
+ case F_READ:
+ /*
+ * Test for DMA error
+ */
+ status = read_dmastat();
+ if(!(status & DMAOK)) {
+ fd_error = "Dma error";
+ return(X_ERROR);
+ }
+ /*
+ * Get controller status and check for errors.
+ */
+ status = read_fdreg(FDC_CS);
+ if(status & (RNF | CRCERR | LD_T00)) {
+ fd_error = "Read error";
+ if(status & RNF)
+ sc->curtrk = INV_TRK;
+ return(X_ERROR);
+ }
+ break;
+ case F_WRITE:
+ /*
+ * Test for DMA error
+ */
+ status = read_dmastat();
+ if(!(status & DMAOK)) {
+ fd_error = "Dma error";
+ return(X_ERROR);
+ }
+ /*
+ * Get controller status and check for errors.
+ */
+ status = read_fdreg(FDC_CS);
+ if(status & WRI_PRO) {
+ fd_error = "Write protected";
+ return(X_FAIL);
+ }
+ if(status & (RNF | CRCERR | LD_T00)) {
+ fd_error = "Write error";
+ sc->curtrk = INV_TRK;
+ return(X_ERROR);
+ }
+ break;
+ case SEEK:
+ status = read_fdreg(FDC_CS);
+ if(status & (RNF | CRCERR)) {
+ fd_error = "Seek error";
+ sc->curtrk = INV_TRK;
+ return(X_ERROR);
+ }
+ return(X_AGAIN);
+ case RESTORE:
+ /*
+ * Determine if the recalibration succeeded.
+ */
+ status = read_fdreg(FDC_CS);
+ if(status & RNF) {
+ fd_error = "Recalibrate error";
+ /* reset controller */
+ write_fdreg(FDC_CS, IRUPT);
+ sc->curtrk = INV_TRK;
+ return(X_ERROR);
+ }
+ sc->curtrk = 0;
+ if(fd_state == FLP_STAT) {
+ if(status & WRI_PRO)
+ sc->flags |= FLPF_WRTPROT;
+ break;
+ }
+ return(X_AGAIN);
+ default:
+ fd_error = "Driver error: fd_xfer_ok : Unknown state";
+ return(X_FAIL);
+ }
+ return(X_OK);
+}
+
+/*
+ * All timeouts will call this function.
+ */
+static void
+fdmotoroff(sc)
+struct fd_softc *sc;
+{
+ int sps;
+
+ /*
+ * Get at harware interrupt level
+ */
+ sps = splbio();
+
+#if FLP_DEBUG
+ printf("fdmotoroff, state = 0x%x\n", fd_state);
+#endif
+
+ switch(fd_state) {
+ case FLP_STAT :
+ case FLP_XFER :
+ /*
+ * Timeout during a transfer; cancel transaction
+ * set command to 'IRUPT'.
+ * A drive-interrupt is simulated to trigger the state
+ * machine.
+ */
+ /*
+ * Cancel current transaction
+ */
+ fd_cmd = IRUPT;
+ write_fdreg(FDC_CS, IRUPT);
+ delay(20);
+ (void)read_fdreg(FDC_CS);
+ write_fdreg(FDC_CS, RESTORE);
+ break;
+
+ case FLP_MON :
+ /*
+ * Turn motor off.
+ */
+ if(selected)
+ fddeselect();
+ fd_state = FLP_IDLE;
+ break;
+ }
+ splx(sps);
+}
+
+/*
+ * min byte count to whats left of the track in question
+ */
+static void
+fdminphys(bp)
+struct buf *bp;
+{
+ struct fd_softc *sc;
+ int sec, toff, tsz;
+
+ if((sc = getsoftc(fdcd, DISKUNIT(bp->b_dev))) == NULL)
+ panic("fdminphys: couldn't get softc");
+
+ sec = bp->b_blkno % (sc->nsectors * sc->nheads);
+ toff = sec * SECTOR_SIZE;
+ tsz = sc->nsectors * sc->nheads * SECTOR_SIZE;
+
+#ifdef FLP_DEBUG
+ printf("fdminphys: before %d", bp->b_bcount);
+#endif
+
+ bp->b_bcount = min(bp->b_bcount, tsz - toff);
+
+#ifdef FLP_DEBUG
+ printf(" after %d\n", bp->b_bcount);
+#endif
+
+ minphys(bp);
+}
+
+/*
+ * Used to find out wich drives are actually connected. We do this by issueing
+ * is 'RESTORE' command and check if the 'track-0' bit is set. This also works
+ * if the drive is present but no floppy is inserted.
+ */
+static void
+fdtestdrv(fdsoftc)
+struct fd_softc *fdsoftc;
+{
+ int i, status;
+
+ /*
+ * Select the right unit and head.
+ */
+ fdselect(fdsoftc->unit, 0, FLP_DD);
+
+ write_fdreg(FDC_CS, RESTORE|HBIT);
+
+ /*
+ * Wait for about 2 seconds.
+ */
+ delay(2000000);
+
+ status = read_fdreg(FDC_CS);
+ if(status & (RNF|BUSY)) {
+ write_fdreg(FDC_CS, IRUPT); /* reset controller */
+ delay(40);
+ }
+
+ if(!(status & LD_T00))
+ fdsoftc->flags |= FLPF_NOTRESP;
+
+ fddeselect();
+}
+
+/*
+ * Build disk label. For now we only create a label from what we know
+ * from 'sc'.
+ */
+static int
+fdgetdisklabel(sc, dev)
+struct fd_softc *sc;
+dev_t dev;
+{
+ struct disklabel *lp, *dlp;
+ int part;
+
+ /*
+ * If we already got one, get out.
+ */
+ if(sc->flags & FLPF_HAVELAB)
+ return(0);
+
+#ifdef FLP_DEBUG
+ printf("fdgetdisklabel()\n");
+#endif
+
+ part = DISKPART(dev);
+ lp = &sc->dkdev.dk_label;
+ bzero(lp, sizeof(struct disklabel));
+
+ lp->d_secsize = SECTOR_SIZE;
+ lp->d_ntracks = sc->nheads;
+ lp->d_nsectors = sc->nsectors;
+ lp->d_secpercyl = lp->d_ntracks * lp->d_nsectors;
+ lp->d_ncylinders = sc->nblocks / lp->d_secpercyl;
+ lp->d_secperunit = sc->nblocks;
+
+ lp->d_type = DTYPE_FLOPPY;
+ lp->d_rpm = 300; /* good guess I suppose. */
+ lp->d_interleave = 1; /* FIXME: is this OK? */
+ lp->d_bbsize = 0;
+ lp->d_sbsize = 0;
+ lp->d_npartitions = part + 1;
+ lp->d_trkseek = STEP_DELAY;
+ lp->d_magic = DISKMAGIC;
+ lp->d_magic2 = DISKMAGIC;
+ lp->d_checksum = dkcksum(lp);
+ lp->d_partitions[part].p_size = lp->d_secperunit;
+ lp->d_partitions[part].p_fstype = FS_UNUSED;
+ lp->d_partitions[part].p_fsize = 1024;
+ lp->d_partitions[part].p_frag = 8;
+ sc->flags |= FLPF_HAVELAB;
+
+ return(0);
+}