diff options
Diffstat (limited to 'sys/arch/i386/isa/fd.c')
-rw-r--r-- | sys/arch/i386/isa/fd.c | 155 |
1 files changed, 95 insertions, 60 deletions
diff --git a/sys/arch/i386/isa/fd.c b/sys/arch/i386/isa/fd.c index 0120048c001..6508a7bdd39 100644 --- a/sys/arch/i386/isa/fd.c +++ b/sys/arch/i386/isa/fd.c @@ -1,5 +1,5 @@ -/* $OpenBSD: fd.c,v 1.14 1996/05/07 07:22:16 deraadt Exp $ */ -/* $NetBSD: fd.c,v 1.88 1996/05/03 19:14:53 christos Exp $ */ +/* $OpenBSD: fd.c,v 1.15 1996/05/25 22:17:48 deraadt Exp $ */ +/* $NetBSD: fd.c,v 1.90 1996/05/12 23:12:03 mycroft Exp $ */ /*- * Copyright (c) 1993, 1994, 1995 Charles Hannum. @@ -55,8 +55,9 @@ #include <sys/queue.h> #include <machine/cpu.h> -#include <machine/pio.h> +#include <machine/bus.h> #include <machine/conf.h> +#include <machine/intr.h> #include <dev/isa/isavar.h> #include <dev/isa/isadmavar.h> @@ -95,7 +96,9 @@ struct fdc_softc { struct isadev sc_id; void *sc_ih; - int sc_iobase; + bus_chipset_tag_t sc_bc; /* ISA chipset identifier */ + bus_io_handle_t sc_ioh; /* ISA io handle */ + int sc_drq; struct fd_softc *sc_fd[4]; /* pointers to children */ @@ -204,7 +207,7 @@ void fd_set_motor __P((struct fdc_softc *fdc, int reset)); void fd_motor_off __P((void *arg)); void fd_motor_on __P((void *arg)); int fdcresult __P((struct fdc_softc *fdc)); -int out_fdc __P((int iobase, u_char x)); +int out_fdc __P((bus_chipset_tag_t bc, bus_io_handle_t ioh, u_char x)); void fdcstart __P((struct fdc_softc *fdc)); void fdcstatus __P((struct device *dv, int n, char *s)); void fdctimeout __P((void *arg)); @@ -220,41 +223,58 @@ fdcprobe(parent, match, aux) void *match, *aux; { register struct isa_attach_args *ia = aux; - int iobase = ia->ia_iobase; + bus_chipset_tag_t bc; + bus_io_handle_t ioh; + int rv; + + bc = ia->ia_bc; + rv = 0; + + /* Map the i/o space. */ + if (bus_io_map(bc, ia->ia_iobase, FDC_NPORT, &ioh)) + return 0; /* reset */ - outb(iobase + fdout, 0); + bus_io_write_1(bc, ioh, fdout, 0); delay(100); - outb(iobase + fdout, FDO_FRST); + bus_io_write_1(bc, ioh, fdout, FDO_FRST); /* see if it can handle a command */ - if (out_fdc(iobase, NE7CMD_SPECIFY) < 0) - return 0; - out_fdc(iobase, 0xdf); - out_fdc(iobase, 2); + if (out_fdc(bc, ioh, NE7CMD_SPECIFY) < 0) + goto out; + out_fdc(bc, ioh, 0xdf); + out_fdc(bc, ioh, 2); #ifdef NEWCONFIG - if (iobase == IOBASEUNK || ia->ia_drq == DRQUNK) + if (ia->ia_iobase == IOBASEUNK || ia->ia_drq == DRQUNK) return 0; if (ia->ia_irq == IRQUNK) { ia->ia_irq = isa_discoverintr(fdcforceintr, aux); if (ia->ia_irq == IRQNONE) - return 0; + goto out; /* reset it again */ - outb(iobase + fdout, 0); + bus_io_write_1(bc, ioh, fdout, 0); delay(100); - outb(iobase + fdout, FDO_FRST); + bus_io_write_1(bc, ioh, fdout, FDO_FRST); } #endif + rv = 1; ia->ia_iosize = FDC_NPORT; ia->ia_msize = 0; - return 1; + + out: + bus_io_unmap(bc, ioh, FDC_NPORT); + return rv; } #ifdef NEWCONFIG +/* + * XXX This is broken, and needs fixing. In general, the interface needs + * XXX to change. + */ void fdcforceintr(aux) void *aux; @@ -264,9 +284,9 @@ fdcforceintr(aux) /* the motor is off; this should generate an error with or without a disk drive present */ - out_fdc(iobase, NE7CMD_SEEK); - out_fdc(iobase, 0); - out_fdc(iobase, 0); + out_fdc(bc, ioh, NE7CMD_SEEK); + out_fdc(bc, ioh, 0); + out_fdc(bc, ioh, 0); } #endif @@ -303,11 +323,21 @@ fdcattach(parent, self, aux) void *aux; { struct fdc_softc *fdc = (void *)self; + bus_chipset_tag_t bc; + bus_io_handle_t ioh; struct isa_attach_args *ia = aux; struct fdc_attach_args fa; int type; - fdc->sc_iobase = ia->ia_iobase; + bc = ia->ia_bc; + + /* Re-map the I/O space. */ + if (bus_io_map(bc, ia->ia_iobase, FDC_NPORT, &ioh)) + panic("fdcattach: couldn't map I/O ports"); + + fdc->sc_bc = bc; + fdc->sc_ioh = ioh; + fdc->sc_drq = ia->ia_drq; fdc->sc_state = DEVIDLE; TAILQ_INIT(&fdc->sc_drives); @@ -350,7 +380,8 @@ fdprobe(parent, match, aux) struct cfdata *cf = match; struct fdc_attach_args *fa = aux; int drive = fa->fa_drive; - int iobase = fdc->sc_iobase; + bus_chipset_tag_t bc = fdc->sc_bc; + bus_io_handle_t ioh = fdc->sc_ioh; int n; if (cf->cf_loc[0] != -1 && cf->cf_loc[0] != drive) @@ -364,14 +395,14 @@ fdprobe(parent, match, aux) return 0; /* select drive and turn on motor */ - outb(iobase + fdout, drive | FDO_FRST | FDO_MOEN(drive)); + bus_io_write_1(bc, ioh, fdout, drive | FDO_FRST | FDO_MOEN(drive)); /* wait for motor to spin up */ delay(250000); - out_fdc(iobase, NE7CMD_RECAL); - out_fdc(iobase, drive); + out_fdc(bc, ioh, NE7CMD_RECAL); + out_fdc(bc, ioh, drive); /* wait for recalibrate */ delay(2000000); - out_fdc(iobase, NE7CMD_SENSEI); + out_fdc(bc, ioh, NE7CMD_SENSEI); n = fdcresult(fdc); #ifdef FD_DEBUG { @@ -385,7 +416,7 @@ fdprobe(parent, match, aux) if (n != 2 || (fdc->sc_status[0] & 0xf8) != 0x20) return 0; /* turn off motor */ - outb(iobase + fdout, FDO_FRST); + bus_io_write_1(bc, ioh, fdout, FDO_FRST); return 1; } @@ -634,7 +665,7 @@ fd_set_motor(fdc, reset) for (n = 0; n < 4; n++) if ((fd = fdc->sc_fd[n]) && (fd->sc_flags & FD_MOTOR)) status |= FDO_MOEN(n); - outb(fdc->sc_iobase + fdout, status); + bus_io_write_1(fdc->sc_bc, fdc->sc_ioh, fdout, status); } void @@ -669,13 +700,15 @@ int fdcresult(fdc) struct fdc_softc *fdc; { - int iobase = fdc->sc_iobase; + bus_chipset_tag_t bc = fdc->sc_bc; + bus_io_handle_t ioh = fdc->sc_ioh; u_char i; int j = 100000, n = 0; for (; j; j--) { - i = inb(iobase + fdsts) & (NE7_DIO | NE7_RQM | NE7_CB); + i = bus_io_read_1(bc, ioh, fdsts) & + (NE7_DIO | NE7_RQM | NE7_CB); if (i == NE7_RQM) return n; if (i == (NE7_DIO | NE7_RQM | NE7_CB)) { @@ -683,7 +716,7 @@ fdcresult(fdc) log(LOG_ERR, "fdcresult: overrun\n"); return -1; } - fdc->sc_status[n++] = inb(iobase + fddata); + fdc->sc_status[n++] = bus_io_read_1(bc, ioh, fddata); } } log(LOG_ERR, "fdcresult: timeout\n"); @@ -691,19 +724,20 @@ fdcresult(fdc) } int -out_fdc(iobase, x) - int iobase; +out_fdc(bc, ioh, x) + bus_chipset_tag_t bc; + bus_io_handle_t ioh; u_char x; { int i = 100000; - while ((inb(iobase + fdsts) & NE7_DIO) && i-- > 0); + while ((bus_io_read_1(bc, ioh, fdsts) & NE7_DIO) && i-- > 0); if (i <= 0) return -1; - while ((inb(iobase + fdsts) & NE7_RQM) == 0 && i-- > 0); + while ((bus_io_read_1(bc, ioh, fdsts) & NE7_RQM) == 0 && i-- > 0); if (i <= 0) return -1; - outb(iobase + fddata, x); + bus_io_write_1(bc, ioh, fddata, x); return 0; } @@ -777,7 +811,7 @@ fdcstatus(dv, n, s) struct fdc_softc *fdc = (void *)dv->dv_parent; if (n == 0) { - out_fdc(fdc->sc_iobase, NE7CMD_SENSEI); + out_fdc(fdc->sc_bc, fdc->sc_ioh, NE7CMD_SENSEI); (void) fdcresult(fdc); n = 2; } @@ -849,7 +883,8 @@ fdcintr(arg) #define cyl fdc->sc_status[1] struct fd_softc *fd; struct buf *bp; - int iobase = fdc->sc_iobase; + bus_chipset_tag_t bc = fdc->sc_bc; + bus_io_handle_t ioh = fdc->sc_ioh; int read, head, sec, i, nblks; struct fd_type *type; @@ -904,13 +939,13 @@ loop: if (fd->sc_cylin == bp->b_cylin) goto doio; - out_fdc(iobase, NE7CMD_SPECIFY);/* specify command */ - out_fdc(iobase, fd->sc_type->steprate); - out_fdc(iobase, 6); /* XXX head load time == 6ms */ + out_fdc(bc, ioh, NE7CMD_SPECIFY);/* specify command */ + out_fdc(bc, ioh, fd->sc_type->steprate); + out_fdc(bc, ioh, 6); /* XXX head load time == 6ms */ - out_fdc(iobase, NE7CMD_SEEK); /* seek function */ - out_fdc(iobase, fd->sc_drive); /* drive number */ - out_fdc(iobase, bp->b_cylin * fd->sc_type->step); + out_fdc(bc, ioh, NE7CMD_SEEK); /* seek function */ + out_fdc(bc, ioh, fd->sc_drive); /* drive number */ + out_fdc(bc, ioh, bp->b_cylin * fd->sc_type->step); fd->sc_cylin = -1; fdc->sc_state = SEEKWAIT; @@ -950,24 +985,24 @@ loop: isa_dmastart(read, bp->b_data + fd->sc_skip, fd->sc_nbytes, fdc->sc_drq); #endif - outb(iobase + fdctl, type->rate); + bus_io_write_1(bc, ioh, fdctl, type->rate); #ifdef FD_DEBUG printf("fdcintr: %s drive %d track %d head %d sec %d nblks %d\n", read ? "read" : "write", fd->sc_drive, fd->sc_cylin, head, sec, nblks); #endif if (read) - out_fdc(iobase, NE7CMD_READ); /* READ */ + out_fdc(bc, ioh, NE7CMD_READ); /* READ */ else - out_fdc(iobase, NE7CMD_WRITE); /* WRITE */ - out_fdc(iobase, (head << 2) | fd->sc_drive); - out_fdc(iobase, fd->sc_cylin); /* track */ - out_fdc(iobase, head); - out_fdc(iobase, sec + 1); /* sector +1 */ - out_fdc(iobase, type->secsize); /* sector size */ - out_fdc(iobase, type->sectrac); /* sectors/track */ - out_fdc(iobase, type->gap1); /* gap1 size */ - out_fdc(iobase, type->datalen); /* data length */ + out_fdc(bc, ioh, NE7CMD_WRITE); /* WRITE */ + out_fdc(bc, ioh, (head << 2) | fd->sc_drive); + out_fdc(bc, ioh, fd->sc_cylin); /* track */ + out_fdc(bc, ioh, head); + out_fdc(bc, ioh, sec + 1); /* sector +1 */ + out_fdc(bc, ioh, type->secsize); /* sector size */ + out_fdc(bc, ioh, type->sectrac); /* sectors/track */ + out_fdc(bc, ioh, type->gap1); /* gap1 size */ + out_fdc(bc, ioh, type->datalen); /* data length */ fdc->sc_state = IOCOMPLETE; disk_busy(&fd->sc_dk); @@ -987,7 +1022,7 @@ loop: disk_unbusy(&fd->sc_dk, 0); /* no data on seek */ /* Make sure seek really happened. */ - out_fdc(iobase, NE7CMD_SENSEI); + out_fdc(bc, ioh, NE7CMD_SENSEI); if (fdcresult(fdc) != 2 || (st0 & 0xf8) != 0x20 || cyl != bp->b_cylin * fd->sc_type->step) { #ifdef FD_DEBUG @@ -1067,14 +1102,14 @@ loop: untimeout(fdctimeout, fdc); /* clear the controller output buffer */ for (i = 0; i < 4; i++) { - out_fdc(iobase, NE7CMD_SENSEI); + out_fdc(bc, ioh, NE7CMD_SENSEI); (void) fdcresult(fdc); } /* fall through */ case DORECAL: - out_fdc(iobase, NE7CMD_RECAL); /* recalibrate function */ - out_fdc(iobase, fd->sc_drive); + out_fdc(bc, ioh, NE7CMD_RECAL); /* recalibrate function */ + out_fdc(bc, ioh, fd->sc_drive); fdc->sc_state = RECALWAIT; timeout(fdctimeout, fdc, 5 * hz); return 1; /* will return later */ @@ -1087,7 +1122,7 @@ loop: return 1; /* will return later */ case RECALCOMPLETE: - out_fdc(iobase, NE7CMD_SENSEI); + out_fdc(bc, ioh, NE7CMD_SENSEI); if (fdcresult(fdc) != 2 || (st0 & 0xf8) != 0x20 || cyl != 0) { #ifdef FD_DEBUG fdcstatus(&fd->sc_dev, 2, "recalibrate failed"); |