148819Swilliam /*-
2*63368Sbostic * Copyright (c) 1990, 1993
3*63368Sbostic * The Regents of the University of California. All rights reserved.
448819Swilliam *
548819Swilliam * This code is derived from software contributed to Berkeley by
648819Swilliam * Don Ahn.
748819Swilliam *
848819Swilliam * %sccs.include.redist.c%
948819Swilliam *
10*63368Sbostic * @(#)fd.c 8.1 (Berkeley) 06/11/93
1148819Swilliam */
1248819Swilliam
1348819Swilliam /****************************************************************************/
1448819Swilliam /* standalone fd driver */
1548819Swilliam /****************************************************************************/
1656514Sbostic #include <sys/param.h>
1756514Sbostic #include <sys/dkbad.h>
1848819Swilliam
1956514Sbostic #include <i386/isa/disk.h>
2056514Sbostic #include <i386/isa/fdreg.h>
2156514Sbostic #include <i386/isa/isa.h>
2256514Sbostic #include <stand/saio.h>
2356514Sbostic
2448819Swilliam #define NUMRETRY 10
2548819Swilliam /*#define FDDEBUG*/
2648819Swilliam
2748819Swilliam #define NFD 2
2848819Swilliam #define FDBLK 512
2948819Swilliam #define NUMTYPES 4
3048819Swilliam
3148819Swilliam struct fd_type {
3248819Swilliam int sectrac; /* sectors per track */
3348819Swilliam int secsize; /* size code for sectors */
3448819Swilliam int datalen; /* data len when secsize = 0 */
3548819Swilliam int gap; /* gap len between sectors */
3648819Swilliam int tracks; /* total num of tracks */
3748819Swilliam int size; /* size of disk in sectors */
3848819Swilliam int steptrac; /* steps per cylinder */
3948819Swilliam int trans; /* transfer speed code */
4048819Swilliam };
4148819Swilliam
4248819Swilliam struct fd_type fd_types[NUMTYPES] = {
4348819Swilliam { 18,2,0xFF,0x1B,80,2880,1,0 }, /* 1.44 meg HD 3.5in floppy */
4448819Swilliam { 15,2,0xFF,0x1B,80,2400,1,0 }, /* 1.2 meg HD floppy */
4548819Swilliam { 9,2,0xFF,0x23,40,720,2,1 }, /* 360k floppy in 1.2meg drive */
4648819Swilliam { 9,2,0xFF,0x2A,40,720,1,1 }, /* 360k floppy in DD drive */
4748819Swilliam };
4848819Swilliam
4948819Swilliam
5048819Swilliam /* state needed for current transfer */
5148819Swilliam static int fd_type;
5248819Swilliam static int fd_motor;
5348819Swilliam static int fd_retry;
5448819Swilliam static int fd_drive;
5548819Swilliam static int fd_status[7];
5648819Swilliam
5748819Swilliam static int fdc = IO_FD1; /* floppy disk base */
5848819Swilliam
5948819Swilliam /* Make sure DMA buffer doesn't cross 64k boundary */
6048819Swilliam char bounce[FDBLK];
6148819Swilliam
6248819Swilliam
6348819Swilliam /****************************************************************************/
6448819Swilliam /* fdstrategy */
6548819Swilliam /****************************************************************************/
6648819Swilliam int
fdstrategy(io,func)6748819Swilliam fdstrategy(io,func)
6848819Swilliam register struct iob *io;
6948819Swilliam int func;
7048819Swilliam {
7148819Swilliam char *address;
7248819Swilliam long nblocks,blknum;
7348819Swilliam int unit, iosize;
7448819Swilliam
7548819Swilliam #ifdef FDDEBUG
7648819Swilliam printf("fdstrat ");
7748819Swilliam #endif
7848819Swilliam unit = io->i_unit;
7948819Swilliam fd_type = io->i_part;
8048819Swilliam
8148819Swilliam /*
8248819Swilliam * Set up block calculations.
8348819Swilliam */
8448819Swilliam iosize = io->i_cc / FDBLK;
8548819Swilliam blknum = (unsigned long) io->i_bn * DEV_BSIZE / FDBLK;
8648819Swilliam nblocks = fd_types[fd_type].size;
8748819Swilliam if ((blknum + iosize > nblocks) || blknum < 0) {
8848819Swilliam #ifndef SMALL
8948819Swilliam printf("bn = %d; sectors = %d; type = %d; fssize = %d ",
9048819Swilliam blknum, iosize, fd_type, nblocks);
9148819Swilliam printf("fdstrategy - I/O out of filesystem boundaries\n");
9248819Swilliam #endif
9348819Swilliam return(-1);
9448819Swilliam }
9548819Swilliam
9648819Swilliam address = io->i_ma;
9748819Swilliam while (iosize > 0) {
9848819Swilliam if (fdio(func, unit, blknum, address))
9948819Swilliam return(-1);
10048819Swilliam iosize--;
10148819Swilliam blknum++;
10248819Swilliam address += FDBLK;
10348819Swilliam }
10448819Swilliam return(io->i_cc);
10548819Swilliam }
10648819Swilliam
10748819Swilliam int
fdio(func,unit,blknum,address)10848819Swilliam fdio(func, unit, blknum, address)
10948819Swilliam int func,unit,blknum;
11048819Swilliam char *address;
11148819Swilliam {
11248819Swilliam int i,j,cyl,sectrac,sec,head,numretry;
11348819Swilliam struct fd_type *ft;
11448819Swilliam
11548819Swilliam ft = &fd_types[fd_type];
11648819Swilliam #ifdef FDDEBUG
11748819Swilliam printf("fdio ");
11848819Swilliam #endif
11948819Swilliam
12048819Swilliam sectrac = ft->sectrac;
12148819Swilliam cyl = blknum / (2 * sectrac);
12248819Swilliam numretry = NUMRETRY;
12348819Swilliam
12449078Sbostic if (func == F_WRITE) bcopy(address,bounce,FDBLK);
12548819Swilliam
12648819Swilliam retry:
12748819Swilliam out_fdc(15); /* Seek function */
12848819Swilliam out_fdc(unit); /* Drive number */
12948819Swilliam out_fdc(cyl);
13048819Swilliam
13148819Swilliam waitio();
13248819Swilliam
13348819Swilliam out_fdc(0x8);
13448819Swilliam i = in_fdc(); j = in_fdc();
13548819Swilliam if (!(i&0x20) || (cyl != j)) {
13648819Swilliam numretry--;
13748819Swilliam if (numretry) goto retry;
13848819Swilliam #ifndef SMALL
13948819Swilliam printf("Seek error %d, req = %d, at = %d\n",i,cyl,j);
14048819Swilliam printf("unit %d, type %d, sectrac %d, blknum %d\n",
14148819Swilliam unit,fd_type,sectrac,blknum);
14248819Swilliam #endif
14348819Swilliam return -1;
14448819Swilliam }
14548819Swilliam
14648819Swilliam /* set up transfer */
14749078Sbostic fd_dma(func == F_READ, bounce, FDBLK);
14848819Swilliam sec = blknum % (sectrac * 2);
14948819Swilliam head = sec / sectrac;
15048819Swilliam sec = sec % sectrac + 1;
15148819Swilliam #ifdef FDDEBUG
15248819Swilliam printf("sec %d hd %d cyl %d ", sec, head, cyl);
15348819Swilliam #endif
15448819Swilliam
15549078Sbostic if (func == F_READ) out_fdc(0xE6);/* READ */
15648819Swilliam else out_fdc(0xC5); /* WRITE */
15748819Swilliam out_fdc(head << 2 | fd_drive); /* head & unit */
15848819Swilliam out_fdc(cyl); /* track */
15948819Swilliam out_fdc(head);
16048819Swilliam out_fdc(sec); /* sector XXX +1? */
16148819Swilliam out_fdc(ft->secsize); /* sector size */
16248819Swilliam out_fdc(sectrac); /* sectors/track */
16348819Swilliam out_fdc(ft->gap); /* gap size */
16448819Swilliam out_fdc(ft->datalen); /* data length */
16548819Swilliam
16648819Swilliam waitio();
16748819Swilliam
16848819Swilliam for(i=0;i<7;i++) {
16948819Swilliam fd_status[i] = in_fdc();
17048819Swilliam }
17148819Swilliam if (fd_status[0]&0xF8) {
17248819Swilliam numretry--;
17348819Swilliam if (numretry) goto retry;
17448819Swilliam #ifndef SMALL
17549879Sbostic printf("FD err %lx %lx %lx %lx %lx %lx %lx\n",
17648819Swilliam fd_status[0], fd_status[1], fd_status[2], fd_status[3],
17748819Swilliam fd_status[4], fd_status[5], fd_status[6] );
17848819Swilliam #endif
17948819Swilliam return -1;
18048819Swilliam }
18149078Sbostic if (func == F_READ) bcopy(bounce,address,FDBLK);
18248819Swilliam return 0;
18348819Swilliam }
18448819Swilliam
18548819Swilliam /****************************************************************************/
18648819Swilliam /* fdc in/out */
18748819Swilliam /****************************************************************************/
18848819Swilliam int
in_fdc()18948819Swilliam in_fdc()
19048819Swilliam {
19148819Swilliam int i;
19248819Swilliam while ((i = inb(fdc+fdsts) & 192) != 192) if (i == 128) return -1;
19348819Swilliam return inb(0x3f5);
19448819Swilliam }
19548819Swilliam
dump_stat()19648819Swilliam dump_stat()
19748819Swilliam {
19848819Swilliam int i;
19948819Swilliam for(i=0;i<7;i++) {
20048819Swilliam fd_status[i] = in_fdc();
20148819Swilliam if (fd_status[i] < 0) break;
20248819Swilliam }
20348819Swilliam #ifdef FDDEBUGx
20449879Sbostic printf("FD bad status :%lx %lx %lx %lx %lx %lx %lx\n",
20548819Swilliam fd_status[0], fd_status[1], fd_status[2], fd_status[3],
20648819Swilliam fd_status[4], fd_status[5], fd_status[6] );
20748819Swilliam #endif
20848819Swilliam }
20948819Swilliam
set_intr()21048819Swilliam set_intr()
21148819Swilliam {
21248819Swilliam /* initialize 8259's */
21348819Swilliam outb(0x20,0x11);
21448819Swilliam outb(0x21,32);
21548819Swilliam outb(0x21,4);
21648819Swilliam outb(0x21,1);
21748819Swilliam outb(0x21,0x0f); /* turn on int 6 */
21848819Swilliam
21948819Swilliam /*
22048819Swilliam outb(0xa0,0x11);
22148819Swilliam outb(0xa1,40);
22248819Swilliam outb(0xa1,2);
22348819Swilliam outb(0xa1,1);
22448819Swilliam outb(0xa1,0xff); */
22548819Swilliam
22648819Swilliam }
22748819Swilliam
22848819Swilliam
22948819Swilliam
waitio()23048819Swilliam waitio()
23148819Swilliam {
23248819Swilliam char c;
23348819Swilliam int n;
23448819Swilliam
23548819Swilliam do
23648819Swilliam outb(0x20,0xc); /* read polled interrupt */
23748819Swilliam while ((c=inb(0x20))&0x7f != 6); /* wait for int */
23848819Swilliam outb(0x20,0x20);
23948819Swilliam }
24048819Swilliam
out_fdc(x)24148819Swilliam out_fdc(x)
24248819Swilliam int x;
24348819Swilliam {
24448819Swilliam int r;
24548819Swilliam do {
24648819Swilliam r = (inb(fdc+fdsts) & 192);
24748819Swilliam if (r==128) break;
24848819Swilliam if (r==192) {
24948819Swilliam dump_stat(); /* error: direction. eat up output */
25048819Swilliam }
25148819Swilliam } while (1);
25248819Swilliam outb(0x3f5,x&0xFF);
25348819Swilliam }
25448819Swilliam
25548819Swilliam
25648819Swilliam /****************************************************************************/
25748819Swilliam /* fdopen/fdclose */
25848819Swilliam /****************************************************************************/
fdopen(io)25948819Swilliam fdopen(io)
26048819Swilliam register struct iob *io;
26148819Swilliam {
26248819Swilliam int unit, type, i;
26348819Swilliam struct fd_type *ft;
26448819Swilliam
26548819Swilliam unit = io->i_unit;
26648819Swilliam type = io->i_part;
26748819Swilliam io->i_boff = 0; /* no disklabels -- tar/dump wont work */
26848819Swilliam #ifdef FDDEBUG
26948819Swilliam printf("fdopen %d %d ", unit, type);
27048819Swilliam #endif
27148819Swilliam ft = &fd_types[type];
27248819Swilliam fd_drive = unit;
27348819Swilliam
27448819Swilliam set_intr(); /* init intr cont */
27548819Swilliam
27648819Swilliam /* Try a reset, keep motor on */
27748819Swilliam outb(0x3f2,0);
27848819Swilliam for(i=0; i < 100000; i++);
27948819Swilliam outb(0x3f2,unit | (unit ? 32 : 16) );
28048819Swilliam for(i=0; i < 100000; i++);
28148819Swilliam outb(0x3f2,unit | 0xC | (unit ? 32 : 16) );
28248819Swilliam outb(0x3f7,ft->trans);
28348819Swilliam fd_motor = 1;
28448819Swilliam
28548819Swilliam waitio();
28648819Swilliam
28748819Swilliam out_fdc(3); /* specify command */
28848819Swilliam out_fdc(0xDF);
28948819Swilliam out_fdc(2);
29048819Swilliam
29148819Swilliam out_fdc(7); /* Recalibrate Function */
29248819Swilliam out_fdc(unit);
29348819Swilliam
29448819Swilliam waitio();
29548819Swilliam return(0);
29648819Swilliam }
29748819Swilliam
29848819Swilliam
29948819Swilliam /****************************************************************************/
30048819Swilliam /* fd_dma */
30148819Swilliam /* set up DMA read/write operation and virtual address addr for nbytes */
30248819Swilliam /****************************************************************************/
fd_dma(read,addr,nbytes)30348819Swilliam fd_dma(read,addr,nbytes)
30448819Swilliam int read;
30548819Swilliam unsigned long addr;
30648819Swilliam int nbytes;
30748819Swilliam {
30848819Swilliam /* Set read/write bytes */
30948819Swilliam if (read) {
31048819Swilliam outb(0xC,0x46); outb(0xB,0x46);
31148819Swilliam } else {
31248819Swilliam outb(0xC,0x4A); outb(0xB,0x4A);
31348819Swilliam }
31448819Swilliam /* Send start address */
31548819Swilliam outb(0x4,addr & 0xFF);
31648819Swilliam outb(0x4,(addr>>8) & 0xFF);
31748819Swilliam outb(0x81,(addr>>16) & 0xFF);
31848819Swilliam /* Send count */
31948819Swilliam nbytes--;
32048819Swilliam outb(0x5,nbytes & 0xFF);
32148819Swilliam outb(0x5,(nbytes>>8) & 0xFF);
32248819Swilliam /* set channel 2 */
32348819Swilliam outb(0x0A,2);
32448819Swilliam }
32548819Swilliam
326