Annotation of coherent/b/kernel/io.386/fl.c, revision 1.1

1.1     ! root        1: /* (-lgl
        !             2:  *     COHERENT Device Driver Kit version 2.0.1.b
        !             3:  *     Copyright (c) 1982, 1991, 1992 by Mark Williams Company.
        !             4:  *     All rights reserved. May not be copied without permission.
        !             5:  -lgl) */
        !             6: /*
        !             7:  * This is a driver for the IBM AT and "above"
        !             8:  * floppy drives, using interrupts and DMA on
        !             9:  * the NEC 756 floppy chip. Ugh.
        !            10:  * Handles single, double and quad
        !            11:  * density drives, 8, 9, 15 or 18 sectors per track.
        !            12:  * It wil NOT run on an XT;  it requires the CMOS setup memory.
        !            13:  *
        !            14:  * Minor device assignments: xxuuhkkk
        !            15:  *     uu - unit = 0/1/2/3
        !            16:  *     kkk - kind, struct fdata infra.
        !            17:  *     h - alternating head rather than side by side
        !            18:  *
        !            19:  */
        !            20: 
        !            21: #include       <sys/coherent.h>
        !            22: #ifdef _I386
        !            23: #include       <sys/reg.h>
        !            24: #else
        !            25: #include       <sys/i8086.h>
        !            26: #endif
        !            27: #include       <sys/buf.h>
        !            28: #include       <sys/con.h>
        !            29: #include       <sys/stat.h>
        !            30: #include       <errno.h>
        !            31: #include       <sys/uproc.h>
        !            32: #include       <sys/proc.h>
        !            33: #include       <sys/fdioctl.h>
        !            34:        /*
        !            35:         * Additional internal commands not available
        !            36:         * to outside users:
        !            37:         */
        !            38: #include       <sys/sched.h>
        !            39: #include       <sys/dmac.h>
        !            40: #include       <sys/devices.h>
        !            41: #include       <sys/seg.h>
        !            42: #include       <sys/inode.h>
        !            43: 
        !            44: #define                BIT(n)          (1 << (n))
        !            45: 
        !            46: #define MAXDRVS  2     /* Maximum number of drives that we will support */
        !            47: #define MAXSCTRS 21    /* Maximum acceptable sectors per track.  It is  */
        !            48:                        /* possible to put as many as 10 sectors/track   */
        !            49:                        /* on low density diskettes, 18 sectors/track on */
        !            50:                        /* 5 1/4" high-density diskettes, and 21 sectors */
        !            51:                        /* per track on 3 1/2" high density diskettes.   */
        !            52:                        /* the maximum on "2.88" meg diskettes is not    */
        !            53:                        /* known as of this writing, but may be as high  */
        !            54:                        /* as 42.                                        */
        !            55: #define MAXTYPE  4     /* Maximum "type" value as set in the CMOS RAM.  */
        !            56:                        /* The current value of this is 4, the type for  */
        !            57:                        /* 1.44 meg diskettes.                           */
        !            58: 
        !            59: #ifdef FL_XTRA
        !            60: /* This is conditioned out for now due to end-of-volume problems. */
        !            61: #define FL_CYL_2STEP   42
        !            62: #define FL_CYL_HDLO    82
        !            63: #define FL_CYL_HDHI    83
        !            64: #else
        !            65: #define FL_CYL_2STEP   40
        !            66: #define FL_CYL_HDLO    80
        !            67: #define FL_CYL_HDHI    80
        !            68: #endif
        !            69: 
        !            70: /*
        !            71:  * Patchable parameters.
        !            72:  */
        !            73: 
        !            74: int    fl_srt = 0xD;   /* Floppy seek step rate, in unit -2 millisec */
        !            75:                        /* NOT DIRECTLY ENCODED */
        !            76:                        /* All drives HAVE to work at 5 msec/step, so */
        !            77:                        /* they HAVE to work at 6 msec/step!  (The use of */
        !            78:                        /* 8 msec/step in the old PCs was just IBM being */
        !            79:                        /* excessively conservative. MS/PC-DOS, starting with */
        !            80:                        /* version 2.0, changes it to 6 msec/step.  There */
        !            81:                        /* are, in fact, lots of drives that will step at */
        !            82:                        /* 4 msec/step, but there's no sense in pushing it.) */
        !            83: int    fl_hlt = 1;     /* Floppy head load time, in unit 4 millisec */
        !            84: int    fl_hut = 0xF;   /* Floppy head unload time, in unit 32 millisec */
        !            85: int    fl_disp = 0;    /* If nonzero, print drive parameters on screen */
        !            86: 
        !            87: /*
        !            88:  * Patchable variables for compatibility with IBM products:
        !            89:  *
        !            90:  * FL_DSK_CH_PROB - some machines always have the disk changed line turned on.
        !            91:  * Currently some PS/1's (Consultant, Professional - possibly most of them)
        !            92:  * have this problem, so the default value of zero assumes normal disk change
        !            93:  * line operation.
        !            94:  *
        !            95:  * FL_AUTO_PARM - Only try to autosense floppy parameters if this variable
        !            96:  * is nonzero.  The PS/2-L40 floppy controller apparently has trouble changing
        !            97:  * from low density to high density.  Missing address marks when reading
        !            98:  * a HD floppy are the symptom if FL_AUTO_PARM is set when it shouldn't be.
        !            99:  */
        !           100: int    FL_DSK_CH_PROB = 0;
        !           101: int    FL_AUTO_PARM = 0;
        !           102: 
        !           103: static int jopen;
        !           104: 
        !           105: /*
        !           106: Here's the problem. We need to be able to tell if the disk has been changed.
        !           107: There is an i/o port we can read, but on some systems, we can get constant
        !           108: false positives. This causes massive floppy slowdown as the disk is constantly
        !           109: recalibrating. The solution is not completely satisfactory, but it's the best
        !           110: one I could come up with. Basically, what I said was since we can't tell when
        !           111: the disk has changed, we will act as if it has changed every time we do an
        !           112: open or a reset. The code
        !           113: 
        !           114:                if (FL_DSK_CH_PROB)
        !           115:                        jopen = 2;
        !           116: 
        !           117: indicates the need to pretend that the disk has changed. It is set to 2 since
        !           118: there are two parts to the change procedure. Additional code dependent on
        !           119: the value of FL_DSK_CH_PROB says that if we have not just down an open, then
        !           120: we should skip the recal.  Otherwise, decrement the counter, and do the
        !           121: recal. - mlk */
        !           122: 
        !           123: int    flload();
        !           124: int    flunload();
        !           125: void   flopen();
        !           126: int    flclose();
        !           127: int    flblock();
        !           128: int    flread();
        !           129: int    flwrite();
        !           130: int    flioctl();
        !           131: int    fldelay();
        !           132: int    flintr();
        !           133: int    fltimeout();
        !           134: int    nulldev();
        !           135: 
        !           136: void   fldrvstatus();
        !           137: CON    flcon   = {
        !           138:        DFBLK | DFCHR,                  /* Flags */
        !           139:        FL_MAJOR,                       /* Major index */
        !           140:        flopen,                         /* Open */
        !           141:        flclose,                        /* Close */
        !           142:        flblock,                        /* Block */
        !           143:        flread,                         /* Read */
        !           144:        flwrite,                        /* Write */
        !           145:        flioctl,                        /* Ioctl */
        !           146:        nulldev,                        /* Powerfail */
        !           147:        fltimeout,                      /* Timeout */
        !           148:        flload,                         /* Load */
        !           149:        flunload                        /* Unload */
        !           150: };
        !           151: 
        !           152: #define        MTIMER  2                       /* Motor timeout */
        !           153: #define FDCDOR 0x3F2                   /* Digital output */
        !           154: #define        FDCDAT  0x3F5                   /* Data register */
        !           155: #define        FDCMSR  0x3F4                   /* Main status register */
        !           156: #define        FDCRATE 0x3F7                   /* Transfer rate (500,300,250 Kbps) */
        !           157: #define FDCCHGL 0x3F7                  /* Port where we read the disk */
        !           158:                                        /* changed line */
        !           159: #define DSKCHGD 0x80                   /* Diskette changed line bit. */
        !           160: 
        !           161: #define        DORDS   0x03                    /* Drive select bits */
        !           162: #define        DORNMR  0x04                    /* Not master reset */
        !           163: #define        DORIEN  0x08                    /* Interrupt, DMA enable */
        !           164: #define        DORMS   0xF0                    /* Motor enables */
        !           165: 
        !           166: #define        MSRDB   0x0F                    /* Drive busy */
        !           167: #define        MSRCB   0x10                    /* Control busy */
        !           168: #define        MSRNDMA 0x20                    /* Not DMA */
        !           169: #define        MSRDIO  0x40                    /* Data direction */
        !           170: #define        MSRRQM  0x80                    /* Request for master */
        !           171: 
        !           172: /*
        !           173:  * Status Register 0 - Bit Definitions.
        !           174:  */
        !           175: #define        ST0_US0 0x01                    /* Unit Select 0 */
        !           176: #define        ST0_US1 0x02                    /* Unit Select 1 */
        !           177: #define        ST0_HD  0x04                    /* Head Address */
        !           178: #define        ST0_NR  0x08                    /* Not Ready */
        !           179: #define        ST0_EC  0x10                    /* Equipment Check */
        !           180: #define        ST0_SE  0x20                    /* Seek End */
        !           181: #define        ST0_IC  0xC0                    /* Interrupt code */
        !           182: #define        ST0_NT  0x00                    /* Normal Termination */
        !           183: 
        !           184: /*
        !           185:  * Status Register 1 - Bit Definitions.
        !           186:  */
        !           187: #define        ST1_MA  0x01                    /* Missing Address Mark */
        !           188: #define        ST1_NW  0x02                    /* Not writeable */
        !           189: #define        ST1_ND  0x04                    /* No Data */
        !           190:        /*      0x08 */                 /* Not used - always 0 */
        !           191: #define        ST1_OR  0x10                    /* Overrun */
        !           192: #define        ST1_DE  0x20                    /* Data Error */
        !           193:        /*      0x40 */                 /* Not used - always 0 */
        !           194: #define        ST1_EN  0x80                    /* End of Cylinder */
        !           195: 
        !           196: /*
        !           197:  * Status Register 2 - Bit Definitions.
        !           198:  */
        !           199: #define        ST2_MD  0x01                    /* Missing Address Mark in Data Field */
        !           200: #define        ST2_BC  0x02                    /* Bad Cylinder */
        !           201: #define        ST2_SN  0x04                    /* Scan Not Satisfied */
        !           202: #define        ST2_SH  0x08                    /* Scan Equal Hit */
        !           203: #define        ST2_WC  0x10                    /* Wrong Cylinder */
        !           204: #define        ST2_DD  0x20                    /* Data Error in Data Field */
        !           205: #define        ST2_CM  0x40                    /* Control Mark */
        !           206:        /*      0x80 */                 /* Not used - always 0 */
        !           207: 
        !           208: /*
        !           209:  * Status Register 3 - Bit Definitions.
        !           210:  */
        !           211: #define        ST3_US0 0x01                    /* Unit Select 0 */
        !           212: #define        ST3_US1 0x02                    /* Unit Select 1 */
        !           213: #define        ST3_HD  0x04                    /* Head Address */
        !           214: #define        ST3_TS  0x08                    /* Two Sides */
        !           215: #define        ST3_T0  0x10                    /* Track 0 */
        !           216: #define        ST3_RDY 0x20                    /* Ready */
        !           217: #define ST3_WP 0x40                    /* Write Protected */
        !           218: #define        ST3_FT  0x80                    /* Fault */
        !           219: 
        !           220: /*
        !           221:  * Controller Commands.
        !           222:  */
        !           223: #define        CMDSPEC 0x03                    /* Specify */
        !           224: #define        CMDRCAL 0x07                    /* Recal */
        !           225: #define        CMDSEEK 0x0F                    /* Seek */
        !           226: #define        CMDRDAT 0x66                    /* Read data */
        !           227: #define        CMDWDAT 0x45                    /* Write data */
        !           228: #define CMDSINT 0x08                   /* Sense interrupt status */
        !           229: #define CMDSDRV 0x04                   /* Sense drive status */
        !           230: #define CMDRDID 0x4A                   /* Read ID */
        !           231: #define        CMDFMT  0x4D                    /* Format track */
        !           232: 
        !           233: /*
        !           234:  * Driver States.
        !           235:  */
        !           236: #define        SIDLE   0                       /* Idle */
        !           237: #define        SSEEK   1                       /* Need seek */
        !           238: #define        SRDWR   2                       /* Need read/write command */
        !           239: #define        SENDIO  3                       /* Need end I/O processing */
        !           240: #define        SDELAY  4                       /* Delay before next disk operation */
        !           241: #define        SHDLY   5                       /* Head settling delay before r/w */
        !           242: #define SLOCK  6                       /* Got DMA controller lock */
        !           243: #define SRECAL1 7                      /* First recalibrate attempt */
        !           244: #define SRECAL2 8                      /* Try seeking to track 2 */
        !           245: #define SRECAL3 9                      /* Second recalibrate, if necessary */
        !           246: #define SGOTO2 10                      /* After seek to cylinder 2 */
        !           247: #define SRDID  11                      /* Get sector ID from FDC */
        !           248: #define SSIDTST 12                     /* Testing # of sides */
        !           249: 
        !           250: #define funit(x)       (minor(x) >> 4)   /* Unit/drive number */
        !           251: #define fkind(x)       (minor(x) & 0x7)  /* Kind of format */
        !           252: #define fhbyh(x)       (minor(x) & 0x8)  /* 0=Side by side, 1=Head by head */
        !           253: 
        !           254: static                 /* Parameters for each kind of format */
        !           255: struct FDATA   {
        !           256:        int     fd_size;        /* Blocks per diskette */
        !           257:        int     fd_nhds;        /* Heads per drive */
        !           258:        int     fd_trks;        /* Tracks per side */
        !           259:        int     fd_offs;        /* Sector base */
        !           260:        int     fd_nspt;        /* Sectors per track */
        !           261:        char    fd_GPL[4];      /* Controller gap param (indexed by rate) */
        !           262:        char    fd_N;           /* Controller size param */
        !           263:        char    fd_FGPL;        /* Format gap length */
        !           264: } fdata[] = {
        !           265: /* 8 sectors per track, surface by surface seek. */
        !           266:        {  320,1,40,0, 8, { 0x00,0x20,0x20 }, 2,0x58 }, /* Single sided */
        !           267:        {  640,2,40,0, 8, { 0x00,0x20,0x20 }, 2,0x58 }, /* Double sided */
        !           268:        { 1280,2,80,0, 8, { 0x00,0x20,0x20 }, 2,0x58 }, /* Quad density */
        !           269: /* 9 sectors per track, surface by surface seek. */
        !           270:        {  360,1,40,0, 9, { 0x00,0x20,0x20 }, 2,0x50 }, /* Single sided */
        !           271:        {  720,2,40,0, 9, { 0x00,0x20,0x20 }, 2,0x50 }, /* Double sided */
        !           272:        { 1440,2,80,0, 9, { 0x00,0x20,0x20 }, 2,0x50 }, /* Quad density */
        !           273: /* 15 sectors per track, surface by surface seek. */
        !           274:        { 2400,2,80,0,15, { 0x1B,0x00,0x00 }, 2,0x54 }, /* High capacity */
        !           275: /* 18 sectors per track, surface by surface seek. */
        !           276:        { 2880,2,80,0,18, { 0x1B,0x00,0x00 }, 2,0x6C }  /* 1.44 3.5" */
        !           277: };
        !           278: 
        !           279: static                 /* Parameters for each device type */
        !           280: struct FRATES  {
        !           281:        char    fl_hi_kind;             /* "fdata" initial try for hi dens. */
        !           282:        char    fl_hi_rate;             /* -1 here for lo-dens.             */
        !           283:        char    fl_lo_kind;             /* Lo-dens "fdata" entry to try 1st.*/
        !           284:        char    fl_lo_rate;             /* Proper lo-density rate for type. */
        !           285:        char    dflt_kind;              /* Default parameters.              */
        !           286:        char    dflt_rate;              /* Default data rate.               */
        !           287: } frates[] =   {
        !           288:        { 4,-1, 4,-1, 4, 2 },           /* Type 0 = no drive */
        !           289:        { 4,-1, 4, 2, 4, 2 },           /* Type 1 = 360K  */
        !           290:        { 6, 0, 4, 1, 6, 0 },           /* Type 2 = 1.2M  */
        !           291:        { 4,-1, 5, 2, 5, 2 },           /* Type 3 = 720K  */
        !           292:        { 7, 0, 5, 2, 7, 0 }            /* Type 4 = 1.44M */
        !           293: };
        !           294: 
        !           295: static
        !           296: struct FL      {
        !           297:        BUF     *fl_actf;               /* Queue, forward */
        !           298:        BUF     *fl_actl;               /* Queue, backward */
        !           299:        paddr_t fl_addr;                /* Address */
        !           300:        int     fl_nsec;                /* # of sectors */
        !           301:        int     fl_secn;                /* Current sector */
        !           302:        struct  FDATA fl_fd[MAXDRVS];   /* Disk kind data */
        !           303:        int     fl_fcyl;                /* Floppy cylinder # */
        !           304:        int     fl_2step[MAXDRVS];      /* =1 for double-stepping */
        !           305:        char    fl_incal[MAXDRVS];      /* Disk in cal flags and current cyl */
        !           306:        char    fl_dsk_chngd[MAXDRVS];  /* Diskette changed flags */
        !           307:        char    fl_ndsk;                /* # of drives */
        !           308:        char    fl_unit;                /* Current unit # */
        !           309:        char    fl_selected_unit;       /* Last unit selected */
        !           310:        char    fl_mask;                /* Handy unit mask */
        !           311:        char    fl_hbyh;                /* 0/1 = Side by side/Head by head */
        !           312:        char    fl_nerr;                /* Error count */
        !           313:        int     fl_ncmdstat;            /* Number of cmd status bytes recvd */
        !           314:        char    fl_cmdstat[8];          /* Command Status buffer */
        !           315:        int     fl_nintstat;            /* Number of intr status bytes recvd */
        !           316:        char    fl_intstat[4];          /* Interrupt Status buffer */
        !           317:        int     fl_ndrvstat;            /* Number of drv status bytes read */
        !           318:        char    fl_drvstat[2];          /* Drive Status buffer */
        !           319:        int     fl_fsec;                /* Floppy sector # */
        !           320:        int     fl_head;                /* Floppy head */
        !           321:        char    fl_state;               /* Processing state */
        !           322:        char    fl_mstatus;             /* Motor status */
        !           323:        char    fl_time[MAXDRVS];       /* Motor timeout */
        !           324:        char    fl_rate[MAXDRVS];       /* Data rate: 500,300,250,?? kbps */
        !           325:        char    fl_type[MAXDRVS];       /* Type of drive: 2 = HiCap */
        !           326:        char    fl_rate_set;            /* Currently set data rate */
        !           327:        int     fl_wflag;               /* Write operation  */
        !           328:        int     fl_recov;               /* Recovery initiated */
        !           329:        int     fl_opct[MAXDRVS];       /* open count for each unit */
        !           330:        int     fl_we[MAXDRVS];         /* write enable for each unit */
        !           331: }      fl;
        !           332: 
        !           333: /*
        !           334:  *     We need some areas in global RAM to use as BUF structures
        !           335:  *     and as data areas for special functions such as formatting,
        !           336:  *     reading the drive status and reading sector IDs.  There is
        !           337:  *     one set for each drive.  When the blocks for a drive are in
        !           338:  *     use, the "drv_locked" char is set to non-zero, and is
        !           339:  *     cleared otherwise.  While this scheme doesn't provide
        !           340:  *     complete reentrancy, it does allow both drives to be used
        !           341:  *     "at once" by separate tasks.
        !           342:  */
        !           343: static char    drv_locked[MAXDRVS];    /* One for each possible drive */
        !           344: static char    sw3[MAXDRVS];
        !           345: static  BUF     flbuf[MAXDRVS];
        !           346: /*
        !           347:  *     These next items are related to the floppy disk system
        !           348:  *     in general, so we only need one of each.
        !           349:  */
        !           350: static TIM     fltim;
        !           351: static TIM     fldmalck;       /* DMA lock deferred function structure. */
        !           352: static char    scratch_buffer[BSIZE];
        !           353: static char    fl_clrng_cd;
        !           354: static char    fl_intlv_ct,    /* Counts sectors to find interleave.    */
        !           355:                fl_get_intlv,   /* =1 to start search for interleave.    */
        !           356:                fl_lk4_id,      /* Sector ID to look for for interleave. */
        !           357:                fl_alt_kind,    /* Alternate disk parameter index and    */
        !           358:                fl_alt_rate,    /* data rate to use if not first value.  */
        !           359:                fl_1st_ID,      /* Detects when all track sectors scanned*/
        !           360:                fl_hi_ID;       /* Highest sector ID read so far.        */
        !           361: 
        !           362: /*-------- DEBUG START ---------*/
        !           363: #define FL_DEBUG 1
        !           364: #if FL_DEBUG
        !           365: #define PUSH3(a,b,c)   push3(a,b,c)
        !           366: #define POP3()         pop3()
        !           367: 
        !           368: #define XX     75
        !           369: 
        !           370: static int A[XX], B[XX], C[XX];
        !           371: static int xxp = 0;    /* next avail index */
        !           372: 
        !           373: static push3(a,b,c)
        !           374: int a,b,c;
        !           375: {
        !           376:        if (xxp < XX) {
        !           377:                A[xxp] = a;
        !           378:                B[xxp] = b;
        !           379:                C[xxp] = c;
        !           380:                xxp++;
        !           381:        }
        !           382: }
        !           383: 
        !           384: static pop3()
        !           385: {
        !           386:        int i;
        !           387: 
        !           388:        printf("pop3: ");
        !           389:        for (i = 0; i < xxp; i++) {
        !           390:                printf("c5=%d 1st=%d hi=%d ", A[i], B[i], C[i]);
        !           391:        }
        !           392:        xxp = 0;
        !           393:        printf(":done ");
        !           394: }
        !           395: #else
        !           396: #define PUSH3(a,b,c)
        !           397: #define POP3()
        !           398: #endif
        !           399: /*-------- DEBUG END ---------*/
        !           400: 
        !           401: /*
        !           402:  * The load routine asks the
        !           403:  * switches how many drives are present
        !           404:  * in the machine, and sets up the field
        !           405:  * in the floppy database. It also grabs
        !           406:  * the level 6 interrupt vector.
        !           407:  */
        !           408: static
        !           409: flload()
        !           410: {
        !           411:        register int    eflag;
        !           412:        register int    s, t;
        !           413: 
        !           414:        fl_clrng_cd = 0;
        !           415: 
        !           416:        /*
        !           417:         * Ensure DMA channel 2 is turned off.
        !           418:         * The Computerland ROM does not disable DMA channel after autoboot
        !           419:         * from hard disk.  The Western Digital controller board appears to
        !           420:         * send a dma burst when the floppy controller chip is reset.
        !           421:         */
        !           422:        dmaoff(2);
        !           423: 
        !           424:        /*
        !           425:         * Read floppy equipment byte from CMOS ram
        !           426:         *      drive 0 is in high nibble, drive 1 is in low nibble.
        !           427:         */
        !           428:        outb(0x70, 0x10);
        !           429:        /* delay */
        !           430:        eflag = inb(0x71);
        !           431: 
        !           432:        /*
        !           433:         * Reinitialize patchable parameters for IBM AT.
        !           434:         */
        !           435:        fl_srt = 0xD;   /* Floppy seek step rate, in unit -2 ms */
        !           436:                        /* NOT DIRECTLY ENCODED */
        !           437:        fl_hlt = 25;    /* Floppy head load time, in unit 4 ms */
        !           438: 
        !           439:        /*
        !           440:         * Define AT drive information.
        !           441:         */
        !           442:        fl.fl_type[0] = eflag >> 4;
        !           443:        fl.fl_type[1] = eflag & 15;
        !           444:        fl.fl_ndsk = 0;
        !           445:        for (s = 0; s < MAXDRVS; s++) {
        !           446:                drv_locked[s] = 0;
        !           447:                fl.fl_dsk_chngd[s] = 1;
        !           448:                fl.fl_incal[s] = -1;
        !           449:                t = fl.fl_type[s];
        !           450:                if (t > MAXTYPE)                  /* --in case we get, like, */
        !           451:                        fl.fl_type[s] = t = MAXTYPE;    /* a 2.88 meg drive. */
        !           452:                fl.fl_rate[s] = frates[t].dflt_rate;
        !           453:                fl.fl_fd[s] = fdata[frates[t].dflt_kind];
        !           454:                if (t) fl.fl_ndsk = s + 1;        /* Type 0 = no drive. */
        !           455: 
        !           456:                if (FL_DSK_CH_PROB)
        !           457:                        jopen = 2;
        !           458:        }
        !           459: 
        !           460:        fl.fl_rate_set = 0;
        !           461: 
        !           462:        /*
        !           463:         * Initialize the floppy disk controller (if we
        !           464:         * have any floppy drives).
        !           465:         */
        !           466:        if (fl.fl_ndsk) {
        !           467: 
        !           468:                s = sphi();
        !           469: 
        !           470:                outb(FDCDOR, 0);
        !           471:                setivec(6, &flintr);
        !           472:                /* "Not Reset FDC" must remain low for at least 14 clocks */
        !           473:                outb(FDCDOR, 0);
        !           474:                fl.fl_state = SIDLE;
        !           475:                outb(FDCDOR, DORNMR | DORIEN);
        !           476: 
        !           477:                fl.fl_mstatus = 0;              /* No motors on */
        !           478:                fl.fl_selected_unit = -1;       /* No unit selected */
        !           479: 
        !           480:                flspecify();
        !           481:                outb(FDCRATE, fl.fl_rate_set);
        !           482: 
        !           483:                spl(s);
        !           484:        }
        !           485: }
        !           486: 
        !           487: /*
        !           488:  * Release resources.
        !           489:  */
        !           490: flunload()
        !           491: {
        !           492:        /*
        !           493:         * Cancel timed function.
        !           494:         */
        !           495:        timeout(&fltim, 0, NULL, NULL);
        !           496: 
        !           497:        /*
        !           498:         * Cancel periodic (1 second) invocation.
        !           499:         */
        !           500:        drvl[FL_MAJOR].d_time = 0;
        !           501: 
        !           502:        /*
        !           503:         * Turn motors off.
        !           504:         */
        !           505:        outb(FDCDOR, DORNMR);           /* Leave interrupts disabled. */
        !           506: 
        !           507:        /*
        !           508:         * Clear interrupt vector.
        !           509:         */
        !           510:        if (fl.fl_ndsk)
        !           511:                clrivec(6);
        !           512: }
        !           513: 
        !           514: /*
        !           515:  * The open routine screens out
        !           516:  * opens of illegal minor devices and
        !           517:  * performs the NEC specify command if
        !           518:  * this is the very first floppy disk
        !           519:  * open call.
        !           520:  */
        !           521: static void
        !           522: flopen(dev, mode)
        !           523: dev_t  dev;
        !           524: int    mode;
        !           525: {
        !           526:        register int unit_number = funit(dev);
        !           527:        register int s;
        !           528: 
        !           529:        /*
        !           530:         * Validate existence and data rate (Gap length != 0).
        !           531:         */
        !           532:        if ((unit_number >= fl.fl_ndsk)
        !           533:          || (fl.fl_type[unit_number] == 0)
        !           534:          || (fdata[fkind(dev)].fd_GPL[flrate(dev)] == 0)) {
        !           535:                u.u_error = ENXIO;
        !           536:                goto badFlopen;         /* status. */
        !           537:        }
        !           538: 
        !           539:        if (FL_DSK_CH_PROB)
        !           540:                jopen = 2;
        !           541: 
        !           542:        /*
        !           543:         * May need to write - see if diskette is write proteced.
        !           544:         * We do this with a "Sense Drive Status" command.  Since
        !           545:         * this requires the use of the FDC, we have to schedule it
        !           546:         * like data transfer I/O or FORMAT even though it doesn't
        !           547:         * use the DMA.
        !           548:         */
        !           549:        if (fl.fl_opct[unit_number] == 0) {     /* first open for this floppy */
        !           550:                if (drv_locked[unit_number]) {  /* Work areas avail? */
        !           551:                        u.u_error = EBUSY;              /* No. */
        !           552:                        goto badFlopen;         /* status. */
        !           553:                } else {
        !           554:                        drv_locked[unit_number] = 1;    /* Grab work areas. */
        !           555:                        flbuf[unit_number].b_dev = dev;
        !           556:                        flbuf[unit_number].b_req = BFLSTAT;
        !           557:                        sw3[unit_number] = 0;
        !           558:                                                        /* Get drive status. */
        !           559:                        s = flQhang(&flbuf[unit_number]);
        !           560: 
        !           561:                        for (;;) {
        !           562:                                s = sphi();
        !           563:                                if (fl.fl_state == SIDLE)
        !           564:                                        flfsm();
        !           565:                                spl(s);
        !           566:                                if (sw3[unit_number])
        !           567:                                        break;
        !           568:                                if (fl.fl_state != SIDLE)
        !           569: #ifdef _I386
        !           570:                                        x_sleep(&fl.fl_state,
        !           571:                                          pridisk, slpriSigCatch, "flopen");
        !           572: #else
        !           573:                                        v_sleep(&fl.fl_state,
        !           574:                                          CVBLKIO, IVBLKIO, SVBLKIO,
        !           575:                                          "flopen");
        !           576: #endif
        !           577:                                if (SELF->p_ssig && nondsig()) {  /* signal? */
        !           578:                                        u.u_error = EINTR;
        !           579:                                        goto badFlopen;
        !           580:                                }
        !           581:                        }
        !           582: 
        !           583:                         if (flbuf[unit_number].b_resid != 0) {
        !           584:                                u.u_error = EDATTN;     /* Couldn't get drive */
        !           585:                                goto badFlopen;         /* status. */
        !           586:                        }
        !           587: 
        !           588:                        /* The payoff - set write enable status. */
        !           589:                        fl.fl_we[unit_number] =
        !           590:                          ((sw3[unit_number] & ST3_WP)==0);
        !           591:                        drv_locked[unit_number] = 0;    /* Release work areas */
        !           592:                }
        !           593: 
        !           594:                /*
        !           595:                 * If the drive is low density (no change line) we should
        !           596:                 * flag the need to verify the disk format and density.
        !           597:                 * High density drives (which are also dual density) have
        !           598:                 * change lines that we can check each time we want to read
        !           599:                 * the drive.
        !           600:                 */
        !           601:                if (frates[fl.fl_type[unit_number]].fl_hi_rate == -1) {
        !           602:                        fl.fl_incal[unit_number] = -1;
        !           603:                        fl.fl_dsk_chngd[unit_number] = 1;
        !           604: 
        !           605:                        if (FL_DSK_CH_PROB)
        !           606:                                jopen = 2;
        !           607:                }
        !           608:        }       /* end of first open stuff */
        !           609: 
        !           610:        /* If opening for write, volume must be write enabled. */
        !           611:        if ((mode & IPW) && !fl.fl_we[unit_number]) {
        !           612:                printf("fd%d: <Write Protected>\n", fl.fl_unit);
        !           613:                u.u_error = EROFS;      /* Diskette write */
        !           614:                goto badFlopen;         /* protected. */
        !           615:        }
        !           616: 
        !           617:        fl.fl_opct[unit_number]++;
        !           618: badFlopen:
        !           619:        return;
        !           620: }
        !           621: 
        !           622: /*
        !           623:  * flclose()
        !           624:  */
        !           625: static
        !           626: flclose(dev, mode)
        !           627: dev_t  dev;
        !           628: int    mode;
        !           629: {
        !           630:        register int unit_number = funit(dev);
        !           631: 
        !           632:        fl.fl_opct[unit_number]--;
        !           633: }
        !           634: 
        !           635: /*
        !           636:  * The read routine just calls
        !           637:  * off to the common raw I/O processing
        !           638:  * code, using a static buffer header in
        !           639:  * the driver.
        !           640:  */
        !           641: 
        !           642: static
        !           643: flread(dev, iop)
        !           644: 
        !           645: dev_t  dev;
        !           646: IO     *iop;
        !           647: 
        !           648: {
        !           649:        dmareq(&flbuf[funit(dev)], iop, dev, BREAD);
        !           650: }
        !           651: 
        !           652: /*
        !           653:  * The write routine is just like the
        !           654:  * read routine, except that the function code
        !           655:  * is write instead of read.
        !           656:  */
        !           657: 
        !           658: static
        !           659: flwrite(dev, iop)
        !           660: 
        !           661: dev_t  dev;
        !           662: IO     *iop;
        !           663: 
        !           664: {
        !           665:        dmareq(&flbuf[funit(dev)], iop, dev, BWRITE);
        !           666: }
        !           667: 
        !           668: /*
        !           669:  * The ioctl routine simply queues a format request
        !           670:  * using the flbuf for the specified drive.
        !           671:  * The only valid command is to format a track.
        !           672:  * The parameter block contains the header records supplied to the controller.
        !           673:  */
        !           674: 
        !           675: static
        !           676: flioctl(dev, com, par)
        !           677: dev_t  dev;
        !           678: int    com;
        !           679: char   *par;
        !           680: {
        !           681:        register unsigned s;
        !           682:        register struct fdata *fdp;
        !           683:        unsigned hd, cyl;
        !           684: 
        !           685:        if (com != FDFORMAT) {
        !           686:                u.u_error = EINVAL;
        !           687:                return;
        !           688:        }
        !           689: 
        !           690:        fdp = &fdata[fkind(dev)];               /* Locate formatting    */
        !           691:        cyl = getubd(par);                      /* parameters.          */
        !           692:        hd  = getubd(par+1);
        !           693: 
        !           694:        if (hd > 1 || cyl >= fdp->fd_trks) {
        !           695:                u.u_error = EINVAL;
        !           696:                return;
        !           697:        }
        !           698: 
        !           699:        /*
        !           700:         * The following may need some explanation.
        !           701:         * dmareq will:
        !           702:         *      claim the buffer,
        !           703:         *      bounds check the parameter buffer,
        !           704:         *      lock the parameter buffer in memory,
        !           705:         *      convert io_seek to b_bno,
        !           706:         *      dispatch the request,
        !           707:         *      wait for completion,
        !           708:         *      and unlock the parameter buffer.
        !           709:         * The b_bno is reconverted to hd, cyl in flfsm.
        !           710:         */
        !           711: 
        !           712:        s = fhbyh(dev) ? (cyl * fdp->fd_nhds + hd) : (hd * fdp->fd_trks + cyl);
        !           713:        s *= fdp->fd_nspt;
        !           714:        u.u_io.io_seek = ((long)s) * BSIZE;
        !           715: #ifdef _I386
        !           716:        u.u_io.io.vbase = par;
        !           717: #else
        !           718:        u.u_io.io_base = par;
        !           719: #endif
        !           720:        u.u_io.io_ioc = fdp->fd_nspt * 4;
        !           721:        dmareq(&flbuf[funit(dev)], &u.u_io, dev, BFLFMT);
        !           722:        return 0;
        !           723: }
        !           724: 
        !           725: /*
        !           726:  * Start up block I/O on a
        !           727:  * buffer. Check that the block number
        !           728:  * is not out of range, given the style of
        !           729:  * the disk. Put the buffer header into the
        !           730:  * device queue. Start up the disk if the
        !           731:  * device is idle.
        !           732:  */
        !           733: static
        !           734: flblock(bp)
        !           735: register BUF   *bp;
        !           736: {
        !           737:        register int    s;
        !           738:        register unsigned bno;
        !           739: 
        !           740:        /* Nasty implicit dependency on BSIZE == 2**9 */
        !           741:        bno = bp->b_bno + (bp->b_count >> 9) - 1;
        !           742: 
        !           743: { /*DEBUG*/
        !           744:        int first = bp->b_bno, last = bno;
        !           745:        int fdatasz = fdata[fkind(bp->b_dev)].fd_size;
        !           746:        int fl_fdsz = fl.fl_fd[funit(bp->b_dev)].fd_size;
        !           747: /*DEBUG*/
        !           748: 
        !           749:        if ((bp->b_req == BFLFMT)
        !           750:        &&   ((unsigned)bp->b_bno >= fdata[fkind(bp->b_dev)].fd_size)) {
        !           751:                bp->b_flag |= BFERR;
        !           752:                bdone(bp);
        !           753:                return;
        !           754:        }
        !           755: 
        !           756:        if (bp->b_req != BFLFMT) {
        !           757:                if ((unsigned)bp->b_bno >=
        !           758:                  fl.fl_fd[funit(bp->b_dev)].fd_size)  {
        !           759:                        bp->b_flag |= BFERR;
        !           760:                        bdone(bp);
        !           761:                        return;
        !           762:                }
        !           763:                if (bno >= fl.fl_fd[funit(bp->b_dev)].fd_size) {
        !           764:                        if (bp->b_flag & BFRAW) {
        !           765:                                bp->b_flag |= BFERR;
        !           766:                        }
        !           767:                        bp->b_resid = bp->b_count;
        !           768:                        bdone(bp);              /* return w/ b_resid != 0 */
        !           769:                        return;
        !           770:                }
        !           771:                if ((bp->b_count & 0x1FF) != 0) {
        !           772:                        bp->b_flag |= BFERR;
        !           773:                        bdone(bp);
        !           774:                        return;
        !           775:                }
        !           776:        }
        !           777: } /*DEBUG*/
        !           778: 
        !           779:        flQhang(bp);                    /* Put the block in the queue. */
        !           780: 
        !           781:        s = sphi();
        !           782:        if (fl.fl_state == SIDLE)       /* --if necessary, to */
        !           783:                flfsm();                /* get things moving. */
        !           784:        spl(s);
        !           785: }
        !           786: 
        !           787: /*
        !           788:  * This routine hangs a BUF in the processing queue
        !           789:  */
        !           790: 
        !           791: static
        !           792: flQhang(bp)
        !           793: register BUF *bp;
        !           794: {
        !           795:        register int s = sphi();     /* No interrupts during chaining, please */
        !           796: 
        !           797:        bp->b_actf = NULL;
        !           798: 
        !           799:        if (fl.fl_actf == NULL)
        !           800:                fl.fl_actf = bp;
        !           801:        else
        !           802:                fl.fl_actl->b_actf = bp;
        !           803: 
        !           804:        fl.fl_actl = bp;
        !           805: 
        !           806:        spl(s);
        !           807: }
        !           808: 
        !           809: /*
        !           810:  * This finite state machine is
        !           811:  * responsible for all sequencing on the disk.
        !           812:  * It builds the commands, does the seeks, spins up
        !           813:  * the drive motor for 1 second on the first call,
        !           814:  * and so on.
        !           815:  * Note that the format command is rather obscurely shoehorned into this.
        !           816:  */
        !           817: static
        !           818: flfsm()
        !           819: {
        !           820:        register BUF    *bp;
        !           821:        register int    flcmd;
        !           822:        register int    i;
        !           823:        int             dods;   /* for PS/1, do disk swap */
        !           824: 
        !           825: again:
        !           826:        bp = fl.fl_actf;
        !           827: 
        !           828:        switch (fl.fl_state) {
        !           829: 
        !           830:        case SIDLE:
        !           831: T_HAL(0x40000, printf("SIDLE "));
        !           832:                drvl[FL_MAJOR].d_time = 1;
        !           833: 
        !           834:                if (bp == NULL)
        !           835:                        break;
        !           836: 
        !           837:                fl.fl_unit = funit(bp->b_dev);
        !           838:                fl.fl_mask = 0x10 << fl.fl_unit;
        !           839: 
        !           840: #if 0
        !           841: printf("drv%d: cmd=%d (%s), position=%d, count=%d\n",
        !           842: fl.fl_unit,
        !           843: bp->b_req,
        !           844:   (bp->b_req == BREAD)      ? "BREAD"
        !           845: : (bp->b_req == BWRITE)      ? "BWRITE"
        !           846: : (bp->b_req == BFLSTAT)     ? "BFLSTAT"
        !           847: : (bp->b_req == BFLFMT)      ? "BFLFMT"       : "?????",
        !           848: bp->b_bno,
        !           849: bp->b_count);
        !           850: #endif
        !           851:                /*
        !           852:                 * We do an entire check for drive status here
        !           853:                 */
        !           854:                if (bp->b_req == BFLSTAT) {
        !           855:                        fl.fl_drvstat[0] = 0;
        !           856:                        fldrvstatus();
        !           857:                        sw3[fl.fl_unit] = fl.fl_drvstat[0] | 3;
        !           858:                        bp->b_resid = (fl.fl_ndrvstat == 1) ? 0 : 1;
        !           859:                        fl.fl_actf  = bp->b_actf;
        !           860:                        fl.fl_state = SIDLE;
        !           861:                        goto again;
        !           862:                }
        !           863: 
        !           864:                fl.fl_hbyh = fhbyh(bp->b_dev);
        !           865: 
        !           866:                fl.fl_addr = bp->b_paddr;
        !           867:                fl.fl_secn = bp->b_bno;
        !           868:                fl.fl_time[fl.fl_unit] = 0;
        !           869: 
        !           870:                if ((fl.fl_nsec = bp->b_count>>9) == 0)
        !           871:                        fl.fl_nsec = 1;
        !           872: 
        !           873:                fl.fl_nerr = 0;
        !           874: 
        !           875:                /*
        !           876:                 * Motor is turned off - turn it on, wait 1 second
        !           877:                 * (for write operations only)
        !           878:                 */
        !           879:                if (((fl.fl_mstatus & fl.fl_mask) == 0)
        !           880:                || (fl.fl_unit != fl.fl_selected_unit)) {
        !           881:                        fldrvselect();
        !           882:                        if ((bp->b_req == BWRITE)
        !           883:                        || (bp->b_req == BFLFMT)) {
        !           884:                                timeout(&fltim, HZ, fldelay, SSEEK);
        !           885:                                fl.fl_state = SDELAY;
        !           886:                                break;
        !           887:                        }
        !           888:                }
        !           889: 
        !           890:                /* no break */
        !           891: 
        !           892:        case SSEEK:
        !           893: T_HAL(0x40000, printf("SSEEK "));
        !           894:                fldrvselect();                  /* Keep drive turned on */
        !           895: 
        !           896:                /*
        !           897:                 * Test dual-density drive's disk changed line.  We must
        !           898:                 * test now before we (possibly) recalibrate the drive
        !           899:                 * which would lose us the disk changed indication.
        !           900:                 */
        !           901: 
        !           902:                if ((frates[fl.fl_type[fl.fl_unit]].fl_hi_rate != -1)
        !           903:                &&   (inb(FDCCHGL) & DSKCHGD)
        !           904:                &&   (fl_clrng_cd == 0)) {
        !           905:                        /* See note at def of FL_DSK_CH_PROB above */
        !           906:                        if (FL_DSK_CH_PROB) {
        !           907:                                if (jopen) {
        !           908:                                        jopen--;
        !           909:                                        fl.fl_dsk_chngd[fl.fl_unit] = 1;
        !           910:                                        fl.fl_incal[fl.fl_unit] = -1;
        !           911:                                }
        !           912:                        } else {
        !           913:                                fl.fl_dsk_chngd[fl.fl_unit] = 1;
        !           914:                                fl.fl_incal[fl.fl_unit] = -1;
        !           915:                        }
        !           916:                }
        !           917: 
        !           918:                fl_clrng_cd = 0;
        !           919: 
        !           920:                /*
        !           921:                 * If we have a format command on cylinder zero, head
        !           922:                 * zero, we must recalibrate the drive first, and set
        !           923:                 * up the transfer speed and FDC stuff.  We ignore
        !           924:                 * a disk changed condition since the current format
        !           925:                 * (it may, remember, be unformatted!) is of no
        !           926:                 * consequence.
        !           927:                 */
        !           928:                if (bp->b_req == BFLFMT) {
        !           929:                        fl.fl_dsk_chngd[fl.fl_unit] = 0;
        !           930:                        fl.fl_fd[fl.fl_unit] = fdata[fkind(bp->b_dev)];
        !           931:                        fl.fl_rate[fl.fl_unit] =
        !           932:                        fl.fl_rate_set           = flrate(bp->b_dev);
        !           933:                        if ((fl.fl_fd[fl.fl_unit].fd_trks < 45)
        !           934:                          && (fl.fl_type[fl.fl_unit] != 1))
        !           935:                                fl.fl_2step[fl.fl_unit] = 1;
        !           936:                        outb(FDCRATE, fl.fl_rate_set);
        !           937:                        if (fl.fl_secn == 0)
        !           938:                                fl.fl_incal[fl.fl_unit] = -1;
        !           939:                }
        !           940:                /*
        !           941:                 * Drive is not calibrated - seek to track 0.
        !           942:                 */
        !           943:                if (fl.fl_incal[fl.fl_unit] == -1) {
        !           944:                        flput(CMDRCAL);
        !           945:                        flput(fl.fl_unit);
        !           946:                        fl.fl_state = SRECAL1;
        !           947:                        break;
        !           948:                } else goto Recalibrated;
        !           949: 
        !           950:        case SRECAL1:
        !           951: T_HAL(0x40000, printf("SRECAL1 "));
        !           952:                /*
        !           953:                 * If the recalibrate had to step more than 77 cylinders
        !           954:                 * it will fail.  We must check for this condition and
        !           955:                 * try once more.  With some controllers we will also get
        !           956:                 * an error if the head STARTS over cylinder 0.  In either
        !           957:                 * event we will force a seek to track 2, then recalibrate
        !           958:                 * again.  If this fails, we can't recalibrate the drive.
        !           959:                 */
        !           960:                if ((fl.fl_nintstat != 2)
        !           961:                  || ((fl.fl_intstat[0] & (ST0_IC | ST0_SE)) != ST0_SE)) {
        !           962:                        flput(CMDSEEK);
        !           963:                        flput(fl.fl_unit);
        !           964:                        flput(2);
        !           965:                        fl.fl_state = SRECAL2;
        !           966:                        break;
        !           967:                } else goto RecalibrateOK;
        !           968:        case SRECAL2:
        !           969: T_HAL(0x40000, printf("SRECAL2 "));
        !           970:                flput(CMDRCAL);
        !           971:                flput(fl.fl_unit);
        !           972:                fl.fl_state = SRECAL3;
        !           973:                break;
        !           974:        case SRECAL3:
        !           975: T_HAL(0x40000, printf("SRECAL3 "));
        !           976:                if ((fl.fl_nintstat != 2)
        !           977:                  || ((fl.fl_intstat[0] & (ST0_IC | ST0_SE)) != ST0_SE)) {
        !           978: RecalFailed:
        !           979:                        printf("fd%d: <Can't Recalibrate>\n", fl.fl_unit);
        !           980:                        clrQ(bp->b_dev);
        !           981:                        goto again;
        !           982:                }
        !           983: RecalibrateOK:
        !           984:                flput(CMDSEEK);                 /* We now get off of cyl 0  */
        !           985:                flput(fl.fl_unit);              /* to try to clear the disk */
        !           986:                flput(2);                       /* changed line, which acts */
        !           987:                fl.fl_state = SGOTO2;           /* differently on different */
        !           988:                break;                          /* controllers.  <sigh>  We */
        !           989:                                                /* use cyl 2 since all for- */
        !           990:        case SGOTO2:                            /* matted disks will have a */
        !           991: T_HAL(0x40000, printf("SGOTO2 "));
        !           992:                if ((fl.fl_nintstat != 2)       /* track here.              */
        !           993:                  || ((fl.fl_intstat[0] & (ST0_IC | ST0_SE)) != ST0_SE))
        !           994:                        goto RecalFailed;
        !           995: 
        !           996:                fl.fl_incal[fl.fl_unit] = 2;    /* Heads now on cylinder 2. */
        !           997: 
        !           998: Recalibrated:
        !           999:                /*
        !          1000:                 * Now, if we don't have to check the interleave factor,
        !          1001:                 * we can continue with the seek!
        !          1002:                 */
        !          1003:                if (fl.fl_dsk_chngd[fl.fl_unit] == 0) goto RateKnown;
        !          1004: 
        !          1005:                /*
        !          1006:                 * <sigh>.  Okay, first we'll try the requested density.
        !          1007:                 */
        !          1008: 
        !          1009:                                                /* First we'll make sure   */
        !          1010:                                                /* we're sitting on cyl 2. */
        !          1011:                if (fl.fl_incal[fl.fl_unit] != 2) goto RecalibrateOK;
        !          1012: 
        !          1013:                /*
        !          1014:                 * We start by trying the requested density:
        !          1015:                 */
        !          1016: 
        !          1017:                                                        /* Get requested rate.*/
        !          1018:                i = fl.fl_rate[fl.fl_unit] = flrate(bp->b_dev);
        !          1019:                                                        /* This next mess gets*/
        !          1020:                                                        /* the disk parameters*/
        !          1021:                                                        /* and the alternate  */
        !          1022:                                                        /* values.            */
        !          1023:                if (i == frates[fl.fl_type[fl.fl_unit]].fl_hi_rate){
        !          1024: 
        !          1025:                        fl.fl_fd[fl.fl_unit] =
        !          1026:                         fdata[frates[fl.fl_type[fl.fl_unit]].fl_hi_kind];
        !          1027:                         if (FL_AUTO_PARM) {
        !          1028:                                fl_alt_kind =
        !          1029:                                  frates[fl.fl_type[fl.fl_unit]].fl_lo_kind;
        !          1030:                                fl_alt_rate =
        !          1031:                                  frates[fl.fl_type[fl.fl_unit]].fl_lo_rate;
        !          1032:                        }
        !          1033:                        else {
        !          1034:                                fl_alt_kind =
        !          1035:                                  frates[fl.fl_type[fl.fl_unit]].fl_hi_kind;
        !          1036:                                fl_alt_rate =
        !          1037:                                    frates[fl.fl_type[fl.fl_unit]].fl_hi_rate;
        !          1038:                        }
        !          1039:                } else {
        !          1040:                        fl.fl_fd[fl.fl_unit] =
        !          1041:                         fdata[frates[fl.fl_type[fl.fl_unit]].fl_lo_kind];
        !          1042:                         if (FL_AUTO_PARM) {
        !          1043:                                fl_alt_kind =
        !          1044:                                  frates[fl.fl_type[fl.fl_unit]].fl_hi_kind;
        !          1045:                                fl_alt_rate =
        !          1046:                                    frates[fl.fl_type[fl.fl_unit]].fl_hi_rate;
        !          1047:                        } else {
        !          1048:                                fl_alt_kind =
        !          1049:                                  frates[fl.fl_type[fl.fl_unit]].fl_lo_kind;
        !          1050:                                fl_alt_rate =
        !          1051:                                  frates[fl.fl_type[fl.fl_unit]].fl_lo_rate;
        !          1052:                        }
        !          1053:                }
        !          1054: 
        !          1055:                fl.fl_state  = SRDID;           /* Set up to read sector IDs. */
        !          1056:                fl_get_intlv = 1;
        !          1057:                fl_intlv_ct  =
        !          1058:                fl_lk4_id    =
        !          1059:                fl_1st_ID    =
        !          1060:                fl_hi_ID     = 0;
        !          1061: 
        !          1062:                /*
        !          1063:                 * Now we try the rate to see if we can read sector IDs
        !          1064:                 */
        !          1065: TryRate:
        !          1066:                if (fl.fl_rate_set != i)
        !          1067:                        outb(FDCRATE, fl.fl_rate_set = i);
        !          1068: GetNextID:
        !          1069:                flput(CMDRDID);
        !          1070:                flput(fl.fl_unit);              /* Always read side 0. */
        !          1071: 
        !          1072:                break;                          /* Wait for ID to arrive. */
        !          1073: 
        !          1074:        case SRDID:
        !          1075: T_HAL(0x40000, printf("SRDID "));
        !          1076: 
        !          1077:                if ((fl.fl_ncmdstat < 7)        /* Did we get an ID? */
        !          1078:                  || ((fl.fl_cmdstat[0] & ST0_IC) != ST0_NT) ) {
        !          1079:                        if (fl_alt_rate == -1) { /* No, is there an alternate?*/
        !          1080:                                flstatus();      /* No, we can't go on.       */
        !          1081:                                clrQ(bp->b_dev);
        !          1082: PUSH3(-1,0,1);
        !          1083:                                goto again;
        !          1084:                        } else {
        !          1085:                                fl.fl_fd[fl.fl_unit] = fdata[fl_alt_kind];
        !          1086:                                i = fl.fl_rate[fl.fl_unit] = fl_alt_rate;
        !          1087:                                fl_alt_rate = -1; /* Flag tried alternate.    */
        !          1088: PUSH3(-1,0,2);
        !          1089:                                goto TryRate;     /* Try alternate density.   */
        !          1090:                        }
        !          1091:                }
        !          1092: 
        !          1093:                /*
        !          1094:                 * Test interleave
        !          1095:                 */
        !          1096: 
        !          1097:                if (fl_get_intlv)               /* Looking for interleave? */
        !          1098:                        if (fl_lk4_id) {        /* Yes; started yet?       */
        !          1099:                                if (fl_lk4_id == fl.fl_cmdstat[5]) /* Yes. */
        !          1100:                                        fl_get_intlv = 0; /* We have a hit.*/
        !          1101:                                else                      /* No hit yet,   */
        !          1102:                                        fl_intlv_ct++;    /* count sector. */
        !          1103:                        } else if (fl.fl_cmdstat[5] < 5) {/* Can we start yet?*/
        !          1104:                                fl_intlv_ct = 1;             /* Yes; count, */
        !          1105:                                fl_lk4_id = fl.fl_cmdstat[5] + 1;/* set ID  */
        !          1106:                        }                                        /* to find.*/
        !          1107: 
        !          1108:                /*
        !          1109:                 * Look for highest ID on track
        !          1110:                 */
        !          1111: 
        !          1112: PUSH3(fl.fl_cmdstat[5], fl_1st_ID, fl_hi_ID);
        !          1113:                if (fl.fl_cmdstat[5] != fl_1st_ID) {
        !          1114:                        if (fl_1st_ID == 0)
        !          1115:                                fl_1st_ID = fl.fl_cmdstat[5];
        !          1116:                        if (fl.fl_cmdstat[5] > fl_hi_ID)
        !          1117:                                fl_hi_ID = fl.fl_cmdstat[5];
        !          1118:                        goto GetNextID;
        !          1119:                }
        !          1120: 
        !          1121:                /*
        !          1122:                 * Be sure we have the interleave
        !          1123:                 */
        !          1124: 
        !          1125:                if (fl_get_intlv) goto GetNextID;
        !          1126: 
        !          1127:                /*
        !          1128:                 * So now we know the density and sectors/track
        !          1129:                 */
        !          1130: 
        !          1131:                fl.fl_dsk_chngd[fl.fl_unit] = 0;
        !          1132:                fl.fl_fd[fl.fl_unit].fd_nspt = fl_hi_ID;
        !          1133: 
        !          1134:                /*
        !          1135:                 * There is a problem with the approach used here -
        !          1136:                 * it assumes that once scan of a track starts, all
        !          1137:                 * sectors appear in physical order without any misses.
        !          1138:                 * Unfortunately, this is not always the case, especially
        !          1139:                 * with 1.44 M 3-1/2" drives.
        !          1140:                 *
        !          1141:                 * A workaround which fixes incorrect nspt reading appears
        !          1142:                 * below.
        !          1143:                 */
        !          1144:                if (fl.fl_fd[fl.fl_unit].fd_nspt > 15
        !          1145:                  && fl.fl_fd[fl.fl_unit].fd_nspt < 18)
        !          1146:                        fl.fl_fd[fl.fl_unit].fd_nspt = 18;
        !          1147: 
        !          1148:                /*
        !          1149:                 * We're (supposedly) sitting on track 2.  We'll
        !          1150:                 * look at the last sector ID we've read.  If it's 1,
        !          1151:                 * we need to do double-stepping.
        !          1152:                 */
        !          1153: 
        !          1154:                if (fl.fl_2step[fl.fl_unit] = (fl.fl_cmdstat[3] == 1)) {
        !          1155:                        fl.fl_fd[fl.fl_unit].fd_trks = FL_CYL_2STEP;
        !          1156:                        fl.fl_incal[fl.fl_unit] = 1;
        !          1157:                } else                                  /* Most 1.2M drives */
        !          1158:                        fl.fl_fd[fl.fl_unit].fd_trks = /* have 83 cyls!   */
        !          1159:                          (fl.fl_type[fl.fl_unit] == 2)
        !          1160:                          ? FL_CYL_HDHI : FL_CYL_HDLO;
        !          1161: 
        !          1162:                /*
        !          1163:                 * We next test for one or two sides:
        !          1164:                 */
        !          1165: 
        !          1166:                if (fl.fl_rate[fl.fl_unit] == 0) {          /* If diskette is */
        !          1167:                        fl.fl_fd[fl.fl_unit].fd_nhds = 2; /* high-density it*/
        !          1168: PUSH3(-1,0,4);
        !          1169:                        goto DiskEstablished;               /* will have two  */
        !          1170:                }                                           /* sides.         */
        !          1171: 
        !          1172:                /*
        !          1173:                 * If the diskette is requested by caller as low density
        !          1174:                 * we use the specified number of sides------
        !          1175:                 */
        !          1176: 
        !          1177:                if (fdata[fkind(bp->b_dev)].fd_nspt < 12) {
        !          1178:                        fl.fl_fd[fl.fl_unit].fd_nhds =
        !          1179:                          fdata[fkind(bp->b_dev)].fd_nhds;
        !          1180: PUSH3(-1,0,5);
        !          1181:                        goto DiskEstablished;
        !          1182:                }
        !          1183: 
        !          1184:                /*
        !          1185:                 * --otherwise we check to see:
        !          1186:                 */
        !          1187: 
        !          1188:                flput(CMDRDID);                 /* Just try to read ANY sector*/
        !          1189:                flput(fl.fl_unit | 0x04);       /* ID from side two.          */
        !          1190:                fl.fl_state = SSIDTST;
        !          1191: PUSH3(-1,0,6);
        !          1192:                break;
        !          1193: 
        !          1194:        case SSIDTST:                           /* If we succeeded, we have */
        !          1195: T_HAL(0x40000, printf("SSIDTST "));
        !          1196:                                                /* 2 sides, else we have 1. */
        !          1197:                fl.fl_fd[fl.fl_unit].fd_nhds = ((fl.fl_ncmdstat < 7)
        !          1198:                          || ((fl.fl_cmdstat[0] & ST0_IC) != ST0_NT) ) ? 1 : 2;
        !          1199: 
        !          1200:                /*
        !          1201:                 * So now we now know all about the diskette!
        !          1202:                 */
        !          1203: DiskEstablished:
        !          1204:                fl.fl_fd[fl.fl_unit].fd_size = fl.fl_fd[fl.fl_unit].fd_nhds
        !          1205:                  * fl.fl_fd[fl.fl_unit].fd_trks
        !          1206:                  * fl.fl_fd[fl.fl_unit].fd_nspt;
        !          1207: 
        !          1208:                if (fl_disp) {
        !          1209:                        printf("fl%d: rate=%d, sctrs/trk=%d, hds=%d, cyls=%d,"
        !          1210:                          " size=%d, intlv=%d, stp=%d\n",
        !          1211:                          fl.fl_unit,
        !          1212:                          fl.fl_rate[fl.fl_unit],
        !          1213:                          fl.fl_fd[fl.fl_unit].fd_nspt,
        !          1214:                          fl.fl_fd[fl.fl_unit].fd_nhds,
        !          1215:                          fl.fl_fd[fl.fl_unit].fd_trks,
        !          1216:                          fl.fl_fd[fl.fl_unit].fd_size,
        !          1217:                          fl_intlv_ct,
        !          1218:                          fl.fl_2step[fl.fl_unit]+1);
        !          1219:                        POP3();
        !          1220:                }
        !          1221: 
        !          1222:                /*
        !          1223:                 * Finally, if the diskette drive has a change line
        !          1224:                 * we'll force it off by reading a sector on cylinder 2.
        !          1225:                 * Usually the testing for rate, the seek to cylinder 2
        !          1226:                 * and the soon-to-follow read or write would clear this,
        !          1227:                 * but there are some controllers that don't always clear
        !          1228:                 * it.  The machine I've been testing on clears it sometimes
        !          1229:                 * and not other times.  I hope there aren't machines even
        !          1230:                 * flakier than this!
        !          1231:                 */
        !          1232: 
        !          1233:                if ((frates[fl.fl_type[fl.fl_unit]].fl_hi_rate != -1)
        !          1234:                  && (inb(FDCCHGL) & DSKCHGD)) {
        !          1235:                        dods = 1;
        !          1236: 
        !          1237:                        /* See note at def of FL_DSK_CH_PROB above */
        !          1238:                        if (FL_DSK_CH_PROB) {
        !          1239:                                if (jopen) {
        !          1240:                                        jopen--;
        !          1241:                                } else
        !          1242:                                        dods = 0;
        !          1243:                        }
        !          1244:                        if (dods) {
        !          1245:                                fl.fl_fcyl = fl.fl_incal[fl.fl_unit];
        !          1246:                                fl.fl_head = 0;
        !          1247:                                fl.fl_fsec = 1;
        !          1248: #ifdef _I386
        !          1249:                                fl.fl_addr = MAPIO(allocp.sr_segp->s_vmem,
        !          1250:                                ((int)scratch_buffer - (int)allocp.sr_base));
        !          1251: #else
        !          1252:                                fl.fl_addr = vtop(scratch_buffer, sds);
        !          1253: #endif
        !          1254:                                fl_clrng_cd = 1;
        !          1255:                                goto Sought;
        !          1256:                        }       
        !          1257:                }
        !          1258: 
        !          1259: RateKnown:
        !          1260:                /*
        !          1261:                 * Set data rate if changed.
        !          1262:                 */
        !          1263:                if (fl.fl_rate_set != (i = fl.fl_rate[fl.fl_unit]))
        !          1264:                        outb(FDCRATE, fl.fl_rate_set = i);
        !          1265: 
        !          1266:                /*
        !          1267:                 * Next we must convert the ordinal block number to
        !          1268:                 * cylinder/head/sector form.
        !          1269:                 */
        !          1270:                fl.fl_fsec = (fl.fl_secn % fl.fl_fd[fl.fl_unit].fd_nspt) + 1;
        !          1271: 
        !          1272:                /*
        !          1273:                 * Seek cylinder by cylinder (XENIX/DOS compatible).
        !          1274:                 */
        !          1275:                if (fl.fl_hbyh) {
        !          1276:                        fl.fl_head = fl.fl_secn / fl.fl_fd[fl.fl_unit].fd_nspt;
        !          1277:                        fl.fl_fcyl = fl.fl_head / fl.fl_fd[fl.fl_unit].fd_nhds;
        !          1278:                        fl.fl_head = fl.fl_head % fl.fl_fd[fl.fl_unit].fd_nhds;
        !          1279:                }
        !          1280:                
        !          1281:                /*
        !          1282:                 * Seek surface by surface.
        !          1283:                 */
        !          1284:                else {
        !          1285:                        fl.fl_fcyl = fl.fl_secn / fl.fl_fd[fl.fl_unit].fd_nspt;
        !          1286:                        fl.fl_head = fl.fl_fcyl / fl.fl_fd[fl.fl_unit].fd_trks;
        !          1287:                        fl.fl_fcyl = fl.fl_fcyl % fl.fl_fd[fl.fl_unit].fd_trks;
        !          1288:                }
        !          1289:                                        /* Don't seek unless we have to. */
        !          1290:                if (fl.fl_fcyl == fl.fl_incal[fl.fl_unit])
        !          1291:                        goto Sought;            /* Past tense of seek. */
        !          1292: 
        !          1293:                fl.fl_incal[fl.fl_unit] = fl.fl_fcyl; /* Save new cylinder. */
        !          1294: 
        !          1295:                flput(CMDSEEK);
        !          1296:                flput((fl.fl_head<<2) | fl.fl_unit);
        !          1297:                                                /* If disk is around 40 tracks*/
        !          1298:                                                /* and drive is not 40 track- */
        !          1299:                if ((fl.fl_fd[fl.fl_unit].fd_trks < 45)
        !          1300:                  && (fl.fl_type[fl.fl_unit] != 1))
        !          1301:                        flput(fl.fl_fcyl << 1);         /* -use double step.  */
        !          1302:                else
        !          1303:                        flput(fl.fl_fcyl);              /* Single step.       */
        !          1304: 
        !          1305:                fl.fl_state = SHDLY;
        !          1306:                break;
        !          1307: 
        !          1308:        case SHDLY:
        !          1309: T_HAL(0x40000, printf("SHDLY "));
        !          1310:                /*
        !          1311:                 * Delay for minimum 15 milliseconds after seek before w/fmt.
        !          1312:                 * 2 clock ticks would give 10-20 millisecond (100 Hz clock).
        !          1313:                 * 3 clock ticks gives      20-30 millisecond (100 Hz clock).
        !          1314:                 */
        !          1315:                if (bp->b_req != BREAD) {
        !          1316:                        timeout(&fltim, 3, fldelay, SRDWR);
        !          1317:                        fl.fl_state = SDELAY;
        !          1318:                        break;
        !          1319:                }
        !          1320:                /* no break */
        !          1321: 
        !          1322:        case SRDWR:
        !          1323: T_HAL(0x40000, printf("SRDWR "));
        !          1324: Sought:
        !          1325:                /*
        !          1326:                 * Disable watchdog timer while waiting to lock DMA controller.
        !          1327:                 */
        !          1328:                fl.fl_time[fl.fl_unit] = -1;
        !          1329: 
        !          1330:                /*
        !          1331:                 * Next state will be DMA locked state.
        !          1332:                 */
        !          1333:                fl.fl_state = SLOCK;
        !          1334: 
        !          1335:                /*
        !          1336:                 * If DMA controller locked by someone else, exit for now.
        !          1337:                 */
        !          1338:                if (dmalock(&fldmalck, flfsm, 0) != 0)
        !          1339:                        return;
        !          1340: 
        !          1341:        case SLOCK:
        !          1342: T_HAL(0x40000, printf("SLOCK "));
        !          1343:                /*
        !          1344:                 * Reset watchdog timer to restart timeout sequence.
        !          1345:                 */
        !          1346: 
        !          1347:                fl.fl_time[fl.fl_unit] = 0;
        !          1348: 
        !          1349:                flcmd = CMDRDAT;
        !          1350:                fl.fl_wflag = 0;
        !          1351: 
        !          1352:                if (fl_clrng_cd == 0)
        !          1353:                        if (bp->b_req == BWRITE) {
        !          1354:                                fl.fl_wflag = 1;
        !          1355:                                flcmd = CMDWDAT;
        !          1356:                        }
        !          1357: 
        !          1358:                        else if (bp->b_req == BFLFMT) {
        !          1359:                                fl.fl_wflag = 1;
        !          1360:                                flcmd = CMDFMT;
        !          1361: 
        !          1362: #ifdef _I386
        !          1363:                                if(!dmaon(2, P2P(fl.fl_addr),bp->b_count,
        !          1364:                                  fl.fl_wflag))
        !          1365: #else
        !          1366:                                if(dmaon(2, fl.fl_addr, bp->b_count,
        !          1367:                                  fl.fl_wflag) == 0)
        !          1368: #endif
        !          1369:                                        goto straddle;
        !          1370: 
        !          1371:                                else
        !          1372:                                        goto command;
        !          1373:                        }
        !          1374: 
        !          1375: #ifdef _I386
        !          1376:                if (dmaon(2, P2P(fl.fl_addr), 512, fl.fl_wflag) == 0)
        !          1377: #else
        !          1378:                if (dmaon(2, fl.fl_addr, 512, fl.fl_wflag) == 0)
        !          1379: #endif
        !          1380:                {
        !          1381: straddle:
        !          1382:                        devmsg(bp->b_dev, "fd: DMA page straddle at %x:%x",
        !          1383:                                fl.fl_addr);
        !          1384:                        dmaunlock(&fldmalck);
        !          1385:                        bp->b_flag |= BFERR;
        !          1386:                        fldone(bp);
        !          1387:                        goto again;
        !          1388:                }
        !          1389: command:
        !          1390:                dmago(2);
        !          1391:                flput(flcmd);
        !          1392:                flput((fl.fl_head<<2) | fl.fl_unit);
        !          1393: 
        !          1394:                if (bp->b_req == BFLFMT) {
        !          1395:                        flput(fl.fl_fd[fl.fl_unit].fd_N);       /* N */
        !          1396:                        flput(fl.fl_fd[fl.fl_unit].fd_nspt);    /* SC */
        !          1397:                        flput(fl.fl_fd[fl.fl_unit].fd_FGPL);    /* GPL */
        !          1398:                        flput(0xF6);                            /* D */
        !          1399:                }
        !          1400:                
        !          1401:                else {
        !          1402:                        flput(fl.fl_fcyl);
        !          1403:                        flput(fl.fl_head);
        !          1404:                        flput(fl.fl_fsec);
        !          1405:                        flput(fl.fl_fd[fl.fl_unit].fd_N);       /* N */
        !          1406:                        flput(fl.fl_fd[fl.fl_unit].fd_nspt);    /* EOT */
        !          1407:                        flput(fl.fl_fd[fl.fl_unit].fd_GPL[fl.fl_rate_set]);
        !          1408:                                                                /* GPL */
        !          1409:                        flput(0xFF);                            /* DTL */
        !          1410:                }
        !          1411: 
        !          1412:                fl.fl_state = SENDIO;
        !          1413:                break;
        !          1414: 
        !          1415:        case SENDIO:
        !          1416: T_HAL(0x40000, printf("SENDIO "));
        !          1417:                fl.fl_time[fl.fl_unit] = 0;
        !          1418:                dmaoff(2);
        !          1419:                dmaunlock(&fldmalck);
        !          1420: 
        !          1421:                if (fl_clrng_cd) {
        !          1422:                        fl.fl_state = SIDLE;
        !          1423:                        wakeup(&fl.fl_state);
        !          1424:                        goto again;
        !          1425:                }
        !          1426: 
        !          1427:                /*
        !          1428:                 * We now check for errors.  If the error is a data
        !          1429:                 * CRC error, we KNOW we're on the correct track, and
        !          1430:                 * we just retry the read once before recalibrating.
        !          1431:                 * We recalibrate for all other errors.
        !          1432:                 */
        !          1433:                if ((fl.fl_cmdstat[0] & ST0_IC) != ST0_NT) {
        !          1434:                        if (++fl.fl_nerr < 5) {
        !          1435:                                if (fl.fl_cmdstat[2] & ST2_DD) {
        !          1436:                                        if (fl.fl_nerr & 1)
        !          1437:                                          goto SetSEEKState;
        !          1438:                                        else
        !          1439:                                          goto Ask4Recal;
        !          1440:                                } else {
        !          1441: Ask4Recal:
        !          1442:                                        fl.fl_incal[fl.fl_unit] = -1;
        !          1443: SetSEEKState:
        !          1444:                                        fl.fl_state = SSEEK;
        !          1445:                                }
        !          1446:                        } else {
        !          1447:                                flstatus();             /* Total failure; */
        !          1448:                                bp->b_flag |= BFERR;    /* we give up.    */
        !          1449:                                fldone(bp);
        !          1450:                        }
        !          1451:                }
        !          1452: 
        !          1453:                else if (--fl.fl_nsec == 0) {
        !          1454:                        bp->b_resid = 0;
        !          1455:                        fldone(bp);
        !          1456:                }
        !          1457:                
        !          1458:                else {
        !          1459:                        ++fl.fl_secn;
        !          1460:                        fl.fl_addr += 512;      /* 512 == fl.fl_fd.fd_nbps */
        !          1461:                        fl.fl_state = SSEEK;
        !          1462:                }
        !          1463: 
        !          1464:                /*
        !          1465:                 * Delay for minimum 1.5 msecs after writing before seek.
        !          1466:                 */
        !          1467:                if (fl.fl_wflag) {
        !          1468:                        timeout(&fltim, 2, fldelay, fl.fl_state);
        !          1469:                        fl.fl_state = SDELAY;
        !          1470:                        break;
        !          1471:                }
        !          1472: 
        !          1473:                goto again;
        !          1474: 
        !          1475:        case SDELAY:
        !          1476: T_HAL(0x40000, printf("SDELAY "));
        !          1477:                /*
        !          1478:                 * Ignore interrupts until timeout occurs.
        !          1479:                 */
        !          1480:                break;
        !          1481: 
        !          1482:        default:
        !          1483:                panic("fds");
        !          1484:        }
        !          1485: }
        !          1486: 
        !          1487: /*
        !          1488:  * Delay before initiating next operation.
        !          1489:  * This allows the floppy motor to turn on,
        !          1490:  * the head to settle before writing,
        !          1491:  * the erase head to turn off after writing, etc.
        !          1492:  */
        !          1493: static
        !          1494: fldelay(state)
        !          1495: int state;
        !          1496: {
        !          1497:        int s;
        !          1498: 
        !          1499:        s = sphi();
        !          1500:        if (fl.fl_state == SDELAY) {
        !          1501:                fl.fl_state = state;
        !          1502:                flfsm();
        !          1503:        }
        !          1504:        spl(s);
        !          1505: }
        !          1506: 
        !          1507: /*
        !          1508:  * The flrate function returns the data rate for the flopen and flfsm routines.
        !          1509:  */
        !          1510: static int
        !          1511: flrate(dev)
        !          1512: register dev_t dev;
        !          1513: {
        !          1514:        register int unit = funit(dev);
        !          1515:        register int rate = frates[fl.fl_type[unit]].fl_hi_rate;
        !          1516: 
        !          1517:        if ((rate == -1) || (fdata[fkind(dev)].fd_nspt < 15))
        !          1518:                rate = frates[fl.fl_type[unit]].fl_lo_rate;
        !          1519: 
        !          1520:        return(rate);
        !          1521: }
        !          1522: 
        !          1523: /*
        !          1524:  * This routine is called by the
        !          1525:  * clock handler every second. If the drive
        !          1526:  * has been idle for a long time it turns off
        !          1527:  * the motor and shuts off the timeouts.
        !          1528:  */
        !          1529: 
        !          1530: static
        !          1531: fltimeout()
        !          1532: {
        !          1533:        register int    unit;
        !          1534:        register int    mask;
        !          1535:        register int    s;
        !          1536: 
        !          1537:        s = sphi();
        !          1538: 
        !          1539:        /*
        !          1540:         * Scan all drives, looking for motor timeouts.
        !          1541:         */
        !          1542:        for (unit=0, mask=0x10; unit < MAXDRVS; unit++, mask <<= 1) {
        !          1543: 
        !          1544:                /*
        !          1545:                 * Ignore drives which aren't spinning.
        !          1546:                 */
        !          1547:                if ((fl.fl_mstatus & mask) == 0)
        !          1548:                        continue;
        !          1549: 
        !          1550:                /*
        !          1551:                 * If timer is disabled (i.e. we are waiting for the DMA
        !          1552:                 * controller), go on to the next drive.
        !          1553:                 */
        !          1554:                if (fl.fl_time[unit] < 0)
        !          1555:                        continue;
        !          1556: 
        !          1557:                /*
        !          1558:                 * Leave recently accessed (in last 4 seconds) drives spinning.
        !          1559:                 */
        !          1560:                if (++fl.fl_time[unit] < MTIMER)
        !          1561:                        continue;
        !          1562: 
        !          1563:                /*
        !          1564:                 * Timeout drives which have been inactive for 5 seconds.
        !          1565:                 */
        !          1566:                fl.fl_mstatus &= ~mask;
        !          1567:                if (unit == fl.fl_selected_unit)
        !          1568:                        fl.fl_selected_unit = -1;
        !          1569: 
        !          1570:                /*
        !          1571:                 * Not selected drive, or selected drive is idle.
        !          1572:                 */
        !          1573:                if ((unit != fl.fl_unit) || (fl.fl_state == SIDLE))
        !          1574:                        continue;
        !          1575: 
        !          1576:                /*
        !          1577:                 * Active drive did not complete operation within 5 seconds.
        !          1578:                 * Attempt recovery.
        !          1579:                 */
        !          1580:                flrecov();
        !          1581: 
        !          1582:                /*
        !          1583:                 * Initiate next block request.
        !          1584:                 */
        !          1585:                if (fl.fl_state == SIDLE)
        !          1586:                        flfsm();
        !          1587:        }
        !          1588: 
        !          1589:        /*
        !          1590:         * Physically turn off drives which timed out.
        !          1591:         */
        !          1592:        outb(FDCDOR, DORNMR | DORIEN | fl.fl_mstatus | fl.fl_unit);
        !          1593: 
        !          1594:        /*
        !          1595:         * Stop checking once all drives have been stopped.
        !          1596:         */
        !          1597:        if (fl.fl_mstatus == 0)
        !          1598:                drvl[FL_MAJOR].d_time = 0;
        !          1599: 
        !          1600:        spl(s);
        !          1601: }
        !          1602: 
        !          1603: /*
        !          1604:  * Remove all pending requests for a device from the queue
        !          1605:  * (used after errors).
        !          1606:  */
        !          1607: static
        !          1608: clrQ(dev)
        !          1609: register int dev;
        !          1610: {
        !          1611:        register BUF *bp, *bp2;
        !          1612:        int s;
        !          1613: 
        !          1614:        s = sphi();
        !          1615: 
        !          1616:        while ((bp = fl.fl_actf) && (bp->b_dev == dev)) {
        !          1617:                bp->b_flag |= BFERR;            /* Strip BUFs from front */
        !          1618:                fl.fl_actf = bp->b_actf;        /* of queue.             */
        !          1619:                bdone(bp);
        !          1620:        }
        !          1621:        while (bp) {
        !          1622:                fl.fl_actl = bp;
        !          1623:                if ((bp2 = bp->b_actf) && (bp2->b_dev == dev)) {
        !          1624:                        bp2->b_flag |= BFERR;     /* Strip BUFs from rest  */
        !          1625:                        bp->b_actf = bp2->b_actf; /* rest of queue.        */
        !          1626:                        bdone(bp2);
        !          1627:                } else
        !          1628:                        bp = bp2;
        !          1629:        }
        !          1630:        fl.fl_state = SIDLE;
        !          1631:        wakeup(&fl.fl_state);
        !          1632:        spl(s);
        !          1633: }
        !          1634: 
        !          1635: /*
        !          1636:  * The recovery routine resets and reprograms the floppy controller,
        !          1637:  * and discards any queued requests on the current drive.
        !          1638:  * This is required if the floppy door is open, or diskette is missing.
        !          1639:  */
        !          1640: static
        !          1641: flrecov()
        !          1642: {
        !          1643:        register int    x;
        !          1644: 
        !          1645:        if (FL_DSK_CH_PROB)
        !          1646:                jopen = 2;
        !          1647: 
        !          1648:        /*
        !          1649:         * Disable DMA transfer.
        !          1650:         * Reset floppy controller.
        !          1651:         */
        !          1652:        dmaoff(2);
        !          1653: 
        !          1654:        /*
        !          1655:         * Unlock the controller if locked by us.
        !          1656:         */
        !          1657: 
        !          1658:        outb(FDCDOR, 0);
        !          1659:        fl.fl_state = SIDLE;
        !          1660:        wakeup(&fl.fl_state);
        !          1661:        dmaunlock(&fldmalck);           /* Ensures 14 clock cycles */
        !          1662:        outb(FDCDOR, DORNMR | DORIEN);
        !          1663: 
        !          1664:        fl.fl_mstatus = 0;                      /* No motors on */
        !          1665:        fl.fl_selected_unit = -1;               /* No unit selected */
        !          1666: 
        !          1667:        /*
        !          1668:         * Program floppy controller.
        !          1669:         */
        !          1670:        flspecify();                            /* Forces wait */
        !          1671: 
        !          1672:        /*
        !          1673:         * Program transfer bps.
        !          1674:         */
        !          1675:        outb(FDCRATE, fl.fl_rate_set);
        !          1676: 
        !          1677:        /*
        !          1678:         * Drives are no longer in calibration.
        !          1679:         */
        !          1680:        for (x = 0; x < MAXDRVS; x++)
        !          1681:                fl.fl_incal[1] = -1;
        !          1682: 
        !          1683:        /*
        !          1684:         * Abort all block requests on current drive after 1st recov attempt.
        !          1685:         */
        !          1686:        if (fl.fl_actf) {
        !          1687:                printf("fd%d: <Door Open>\n", fl.fl_unit);     /* Message    */
        !          1688:                clrQ(fl.fl_actf->b_dev);                /* Dump pending reqs. */
        !          1689:                fl.fl_dsk_chngd[fl.fl_unit] = 1;        /* Make disk changed. */
        !          1690:        }
        !          1691: 
        !          1692:        /*
        !          1693:         * Delay before setting controller state to idle.
        !          1694:         * This gives time for spurious floppy interrupts to occur.
        !          1695:         * NOTE: Can't call flfsm(), since it may call us (future revision).
        !          1696:         */
        !          1697:        timeout(&fltim, HZ/4, fldelay, SIDLE);
        !          1698:        fl.fl_state = SDELAY;
        !          1699: }
        !          1700: 
        !          1701: /*
        !          1702:  * The interrupt routine gets all
        !          1703:  * the status bytes the controller chip
        !          1704:  * will give it, then issues a sense interrupt
        !          1705:  * status command (which is necessary for a seek
        !          1706:  * to complete!) and throws all of the status
        !          1707:  * bytes away.
        !          1708:  */
        !          1709: 
        !          1710: static
        !          1711: flintr()
        !          1712: {
        !          1713:        register int s;
        !          1714: 
        !          1715:        s = sphi();
        !          1716:        flsense();
        !          1717: 
        !          1718:        if (fl.fl_state != SIDLE)
        !          1719:                flfsm();
        !          1720: 
        !          1721:        spl(s);
        !          1722: }
        !          1723: 
        !          1724: /*
        !          1725:  * Fldone() returns current request to operating system.
        !          1726:  */
        !          1727: fldone(bp)
        !          1728: register BUF * bp;
        !          1729: {
        !          1730:        fl.fl_actf  = bp->b_actf;
        !          1731:        fl.fl_state = SIDLE;
        !          1732:        bdone(bp);
        !          1733:        wakeup(&fl.fl_state);
        !          1734: }
        !          1735: 
        !          1736: /*
        !          1737:  * Send Specify command and data bytes to FDC
        !          1738:  */
        !          1739: 
        !          1740: static
        !          1741: flspecify()
        !          1742: {
        !          1743:        flput(CMDSPEC);
        !          1744:        flput((fl_srt << 4) | fl_hut);
        !          1745:        flput(fl_hlt << 1);
        !          1746: }
        !          1747: 
        !          1748: /*
        !          1749:  * Flsense() issues a sense interrupt status command
        !          1750:  * to restore the controller to a quiescent state.
        !          1751:  */
        !          1752: 
        !          1753: static
        !          1754: flsense()
        !          1755: {
        !          1756:        flcmdstatus();                  /* Get command status. */
        !          1757:        flintstatus();                  /* Get int status, just in case. */
        !          1758: }
        !          1759: 
        !          1760: /*
        !          1761:  * Get status (if any) from last command
        !          1762:  */
        !          1763: static
        !          1764: flcmdstatus()
        !          1765: {
        !          1766:        register int    b;
        !          1767:        register int    n = 0;          /* # of status bytes read */
        !          1768:        register int    i = 0;          /* Timeout count */
        !          1769:        register int    s;
        !          1770: 
        !          1771:        s = sphi();
        !          1772: 
        !          1773:        /*
        !          1774:         * Read all the status bytes the controller will give us.
        !          1775:         */
        !          1776: 
        !          1777:        for (;;) {
        !          1778:                while (((b=inb(FDCMSR)) & MSRRQM) == 0) {
        !          1779:                        if (--i == 0) {
        !          1780:                                printf("flintr: timeout\n");
        !          1781:                                break;
        !          1782:                        }
        !          1783:                }
        !          1784: 
        !          1785:                if ((b & MSRDIO) == 0)
        !          1786:                        break;
        !          1787: 
        !          1788:                b = inb(FDCDAT);
        !          1789:                if (n < sizeof(fl.fl_cmdstat))
        !          1790:                        fl.fl_cmdstat[n++] = b;
        !          1791:        }
        !          1792: 
        !          1793:        fl.fl_ncmdstat = n;
        !          1794:        spl(s);
        !          1795: }
        !          1796: 
        !          1797: /*
        !          1798:  * Get Inteerupt status
        !          1799:  */
        !          1800: static
        !          1801: flintstatus()
        !          1802: {
        !          1803:        register int    b;
        !          1804:        register int    n = 0;          /* # of status bytes read */
        !          1805:        register int    i = 0;          /* Timeout count */
        !          1806:        register int    s;
        !          1807: 
        !          1808:        s = sphi();
        !          1809: 
        !          1810:        /*
        !          1811:         * Issue a sense interrupt command and stash result.
        !          1812:         */
        !          1813:        flput(CMDSINT);
        !          1814: 
        !          1815:        n = 0;
        !          1816:        for (;;) {
        !          1817:                while (((b=inb(FDCMSR)) & MSRRQM) == 0)
        !          1818:                        if (--i == 0) {
        !          1819:                                printf("flintsense: timeout\n");
        !          1820:                                break;
        !          1821:                        }
        !          1822: 
        !          1823:                if ((b & MSRDIO) == 0)
        !          1824:                        break;
        !          1825: 
        !          1826:                b = inb(FDCDAT);
        !          1827:                if (n < sizeof(fl.fl_intstat))
        !          1828:                        fl.fl_intstat[n++] = b;
        !          1829:        }
        !          1830:        fl.fl_nintstat = n;
        !          1831:        spl(s);
        !          1832: }
        !          1833: 
        !          1834: /*
        !          1835:  * Get the drive status
        !          1836:  */
        !          1837: static void
        !          1838: fldrvstatus()
        !          1839: {
        !          1840:        register int    b;
        !          1841:        register int    n = 0;          /* # of status bytes read */
        !          1842:        register int    i = 0;          /* Timeout count */
        !          1843:        register int    s;
        !          1844: 
        !          1845:        s = sphi();
        !          1846: 
        !          1847:        fldrvselect();                  /* Be sure drive is selected */
        !          1848: 
        !          1849:        /*
        !          1850:         * Issue a sense drive status command and stash result.
        !          1851:         */
        !          1852:        flput(CMDSDRV);
        !          1853:        flput(fl.fl_unit);
        !          1854: 
        !          1855:        for (;;) {
        !          1856:                spl(s);
        !          1857:                if (SELF->p_ssig && nondsig()) {  /* signal? */
        !          1858:                        u.u_error = EINTR;
        !          1859:                        break;
        !          1860:                }
        !          1861:                s = sphi();
        !          1862:                while (((b=inb(FDCMSR)) & MSRRQM) == 0)
        !          1863:                        if (--i == 0) {
        !          1864:                                printf("fldrvsense: timeout\n");
        !          1865:                                break;
        !          1866:                        }
        !          1867: 
        !          1868:                if ((b & MSRDIO) == 0)
        !          1869:                        break;
        !          1870: 
        !          1871:                b = inb(FDCDAT);
        !          1872:                if (n < sizeof(fl.fl_drvstat))
        !          1873:                        fl.fl_drvstat[n++] = b;
        !          1874:        }
        !          1875:        fl.fl_ndrvstat = n;
        !          1876:        spl(s);
        !          1877: }
        !          1878: 
        !          1879: /*
        !          1880:  *     fldrvselect() selects the drive and turns its motor on
        !          1881:  */
        !          1882: static
        !          1883: fldrvselect()
        !          1884: {
        !          1885:        fl.fl_time[fl.fl_unit] = 0;             /* Start motor-on timeout. */
        !          1886:        fl.fl_mstatus |= fl.fl_mask;
        !          1887:        outb(FDCDOR, DORNMR | DORIEN | fl.fl_mstatus | fl.fl_unit);
        !          1888:        fl.fl_selected_unit = fl.fl_unit;       /* This unit is running. */
        !          1889:        flsense();                              /* Just in case --- */
        !          1890: }
        !          1891: 
        !          1892: /*
        !          1893:  * Send a command byte to the
        !          1894:  * NEC chip, first waiting until the chip
        !          1895:  * says that it is ready. No timeout is
        !          1896:  * performed; if the chip dies, we do too!
        !          1897:  */
        !          1898: 
        !          1899: static
        !          1900: flput(b)
        !          1901: 
        !          1902: int    b;
        !          1903: 
        !          1904: {
        !          1905:        register int i = 0;
        !          1906: 
        !          1907:        while ((inb(FDCMSR) & (MSRRQM | MSRDIO)) != MSRRQM) {
        !          1908:                if (--i == 0) {
        !          1909:                        printf("flput(%x): timeout\n",b);
        !          1910:                        return;
        !          1911:                }
        !          1912:        }
        !          1913: 
        !          1914:        outb(FDCDAT, b);
        !          1915: }
        !          1916: 
        !          1917: /*
        !          1918:  * Dissassemble the floppy error status for user reference.
        !          1919:  */
        !          1920: 
        !          1921: static
        !          1922: flstatus()
        !          1923: {
        !          1924:        printf("fd%d: head=%u cyl=%u",
        !          1925:                fl.fl_cmdstat[0] & 3,
        !          1926:                fl.fl_head, fl.fl_fcyl);
        !          1927: 
        !          1928:        /*
        !          1929:         * Report on ST0 bits.
        !          1930:         */
        !          1931:        if (fl.fl_ncmdstat >= 1) {
        !          1932:                if (fl.fl_cmdstat[0] & ST0_NR)
        !          1933:                        printf(" <Not Ready>");
        !          1934: 
        !          1935:                if (fl.fl_cmdstat[0] & ST0_EC)
        !          1936:                        printf(" <Equipment Check>");
        !          1937:        }
        !          1938: 
        !          1939:        /*
        !          1940:         * Report on ST1 bits.
        !          1941:         */
        !          1942:        if (fl.fl_ncmdstat >= 2) {
        !          1943:                if (fl.fl_cmdstat[1] & ST1_MA)
        !          1944:                        printf(" <Missing Address Mark>");
        !          1945: 
        !          1946:                if (fl.fl_cmdstat[1] & ST1_NW)
        !          1947:                        printf(" <Write Protected>");
        !          1948: 
        !          1949:                if (fl.fl_cmdstat[1] & ST1_ND)
        !          1950:                        printf(" <No Data>");
        !          1951: 
        !          1952:                if (fl.fl_cmdstat[1] & ST1_OR)
        !          1953:                        printf(" <Overrun>");
        !          1954: 
        !          1955:                if (fl.fl_cmdstat[1] & ST1_DE)
        !          1956:                        printf(" <Data Error>");
        !          1957: 
        !          1958:                if (fl.fl_cmdstat[1] & ST1_EN)
        !          1959:                        printf(" <End of Cyl>");
        !          1960:        }
        !          1961: 
        !          1962:        /*
        !          1963:         * Report on ST2 bits.
        !          1964:         */
        !          1965:        if (fl.fl_ncmdstat >= 3) {
        !          1966:                if (fl.fl_cmdstat[2] & ST2_MD)
        !          1967:                        printf(" <Missing Data Address Mark>");
        !          1968: 
        !          1969:                if (fl.fl_cmdstat[2] & ST2_BC)
        !          1970:                        printf(" <Bad Cylinder>");
        !          1971: 
        !          1972:                if (fl.fl_cmdstat[2] & ST2_WC)
        !          1973:                        printf(" <Wrong Cylinder>");
        !          1974: 
        !          1975:                if (fl.fl_cmdstat[2] & ST2_DD)
        !          1976:                        printf(" <Bad Data CRC>");
        !          1977: 
        !          1978:                if (fl.fl_cmdstat[2] & ST2_CM)
        !          1979:                        printf(" <Data Deleted>");
        !          1980:        }
        !          1981: 
        !          1982:        printf("\n");
        !          1983: }
        !          1984: /*                       * * * * End of FL.C * * * *                   */

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.