Annotation of coherent/d/286_KERNEL/USRSRC/io/at.c, revision 1.1.1.1

1.1       root        1: /* (-lgl
                      2:  *     COHERENT Driver Kit Version 1.1.0
                      3:  *     Copyright (c) 1982, 1990 by Mark Williams Company.
                      4:  *     All rights reserved. May not be copied without permission.
                      5:  *
                      6:  * $Log:       at.c,v $
                      7:  * Revision 1.11  92/01/17  11:38:56  bin
                      8:  * update by hal... looks like final 321 version
                      9:  * 
                     10:  * Revision 1.12  92/01/17  03:50:53  hal
                     11:  * Cleanup for 3.2.1.
                     12:  * 
                     13:  * Revision 1.11  91/12/11  16:41:10  hal
                     14:  * Add ATSREG patchable kernel variable.
                     15:  * 
                     16:  * Revision 1.10  91/11/11  12:29:03  hal
                     17:  * Use n_atdr.
                     18:  * 
                     19:  * Revision 1.9  91/10/30  10:47:46  hal
                     20:  * Get atparms from tboot.
                     21:  *
                     22:  * Revision 1.8  91/10/24  12:36:25  hal
                     23:  * Bump ATSECS from 4 to 6.
                     24:  * Poll HF_REG (3F6) rather than CSR_REG (1F7).
                     25:  * COH 3.2.03.
                     26:  *
                     27:  * Revision 1.7  91/09/11  14:45:38  hal
                     28:  * Trial patch for Seagate 157A problems.
                     29:  *
                     30:  * Revision 1.6  91/09/11  13:23:12  hal
                     31:  * Explicit sys in include paths.  AT_MAJOR.
                     32:  *
                     33:  * Revision 1.5  91/05/22  15:06:59  hal
                     34:  * Don't force 8's bit of control byte.
                     35:  *
                     36:  * Revision 1.4        91/03/14  14:22:32      hal
                     37:  *
                     38:  -lgl) */
                     39: /*
                     40:  * This is a driver for the
                     41:  * hard disk on the AT.
                     42:  *
                     43:  * Reads drive characteristics from ROM (thru interrupt vector 0x41 and 0x46).
                     44:  * Reads partition information from disk.
                     45:  */
                     46: #include       <sys/coherent.h>
                     47: #include       <sys/fdisk.h>
                     48: #include       <sys/hdioctl.h>
                     49: #include       <sys/buf.h>
                     50: #include       <sys/con.h>
                     51: #include       <sys/devices.h>
                     52: #include       <sys/stat.h>
                     53: #include       <sys/uproc.h>
                     54: #include       <sys/typed.h>
                     55: #include       <errno.h>
                     56: 
                     57: extern saddr_t sds;            /* System Data Selector */
                     58: extern short   n_atdr;         /* Number of "at" drives */
                     59: 
                     60: /*
                     61:  * Configurable parameters
                     62:  */
                     63: #define        HDIRQ   14                      /* Level 14 */
                     64: #define        HDBASE  0x01F0                  /* Port base */
                     65: #define NDRIVE 2                       /* only two drives supported */
                     66: #define        SOFTLIM 6                       /*  (7) num of retrys before diag */
                     67: #define        HARDLIM 8                       /* number of retrys before fail */
                     68: #define        BADLIM  100                     /* num to stop recov if flagged bad */
                     69: 
                     70: #define        BIT(n)          (1 << (n))
                     71: 
                     72: #define        CMOSA   0x70                    /* write cmos address to this port */
                     73: #define        CMOSD   0x71                    /* read cmos data through this port */
                     74: 
                     75: #ifndef        ATCACHE
                     76: #      if VERBOSE > 0
                     77: #              define  ATCACHE 2       /* local cache size in blocks */
                     78: #      else
                     79: #              define  ATCACHE 0       /* no cache for small code */
                     80: #      endif
                     81: #endif
                     82: 
                     83: /*
                     84:  * Driver configuration.
                     85:  */
                     86: void   atload();
                     87: void   atunload();
                     88: void   atopen();
                     89: void   atread();
                     90: void   atwrite();
                     91: int    atioctl();
                     92: void   atwatch();
                     93: void   atblock();
                     94: int    nulldev();
                     95: int    nonedev();
                     96: 
                     97: CON    atcon   = {
                     98:        DFBLK|DFCHR,                    /* Flags */
                     99:        AT_MAJOR,                       /* Major index */
                    100:        atopen,                         /* Open */
                    101:        nulldev,                        /* Close */
                    102:        atblock,                        /* Block */
                    103:        atread,                         /* Read */
                    104:        atwrite,                        /* Write */
                    105:        atioctl,                        /* Ioctl */
                    106:        nulldev,                        /* Powerfail */
                    107:        atwatch,                        /* Timeout */
                    108:        atload,                         /* Load */
                    109:        atunload                        /* Unload */
                    110: };
                    111: 
                    112: /*
                    113:  * Forward Referenced Functions.
                    114:  */
                    115: static int atreset();
                    116: int    atdequeue();
                    117: void   atstart();
                    118: void   atintr();
                    119: void   atdefer();
                    120: int    aterror();
                    121: void   atrecov();
                    122: void   atdone();
                    123: 
                    124: /*
                    125:  * I/O Port Addresses
                    126:  */
                    127: #define        DATA_REG        (HDBASE+0)      /* data (r/w) */
                    128: #define        AUX_REG         (HDBASE+1)      /* error(r), write precomp cyl/4 (w) */
                    129: #define        NSEC_REG        (HDBASE+2)      /* sector count (r/w) */
                    130: #define        SEC_REG         (HDBASE+3)      /* sector number (r/w) */
                    131: #define        LCYL_REG        (HDBASE+4)      /* low cylinder (r/w) */
                    132: #define        HCYL_REG        (HDBASE+5)      /* high cylinder (r/w) */
                    133: #define        HDRV_REG        (HDBASE+6)      /* drive/head (r/w) (D<<4)+(1<<H) */
                    134: #define        CSR_REG         (HDBASE+7)      /* status (r), command (w) */
                    135: #define        HF_REG          (HDBASE+0x206)  /* secondary status(r)/control byte(w)*/
                    136: 
                    137: /*
                    138:  * Error from AUX_REG (r)
                    139:  */
                    140: #define        DAM_ERR         BIT(0)          /* data address mark not found */
                    141: #define        TR0_ERR         BIT(1)          /* track 000 not found */
                    142: #define        ABT_ERR         BIT(2)          /* aborted command */
                    143: #define        ID_ERR          BIT(4)          /* id not found */
                    144: #define        ECC_ERR         BIT(6)          /* data ecc error */
                    145: #define        BAD_ERR         BIT(7)          /* bad block detect */
                    146: 
                    147: /*
                    148:  * Status from CSR_REG (r)
                    149:  */
                    150: #define        ERR_ST          BIT(0)          /* error occurred */
                    151: #define        INDEX_ST        BIT(1)          /* index pulse */
                    152: #define        SOFT_ST         BIT(2)          /* soft (corrected) ECC error */
                    153: #define        DRQ_ST          BIT(3)          /* data request */
                    154: #define        SKC_ST          BIT(4)          /* seek complete */
                    155: #define        WFLT_ST         BIT(5)          /* improper drive operation */
                    156: #define        RDY_ST          BIT(6)          /* drive is ready */
                    157: #define        BSY_ST          BIT(7)          /* controller is busy */
                    158: 
                    159: 
                    160: /*
                    161:  * Commands to CSR_REG (w)
                    162:  */
                    163: #define        RESTORE(rate)   (0x10+(rate))   /* X */
                    164: #define        SEEK(rate)      (0x70+(rate))   /* X */
                    165: #define        READ_CMD        (0x20)          /* X */
                    166: #define        WRITE_CMD       (0x30)          /* X */
                    167: #define        FORMAT_CMD      (0x50)          /* X */
                    168: #define        VERIFY_CMD      (0x40)          /* X */
                    169: #define        DIAGNOSE_CMD    (0x90)          /* X */
                    170: #define        SETPARM_CMD     (0x91)          /* X */
                    171: 
                    172: /*
                    173:  * Device States.
                    174:  */
                    175: #define        SIDLE   0                       /* controller idle */
                    176: #define        SRETRY  1                       /* seeking */
                    177: #define        SREAD   2                       /* reading */
                    178: #define        SWRITE  3                       /* writing */
                    179: 
                    180: /*
                    181:  * Drive Parameters - copied from ROM.
                    182:  * If patched, use the given values instead of reading from the ROM.
                    183:  * NOTE: Exactly duplicates hdparm_s struct.
                    184:  */
                    185: struct dparm_s {
                    186:        unsigned short  d_ncyl;         /* number of cylinders */
                    187:        unsigned char   d_nhead;        /* number of heads */
                    188:        unsigned short  d_rwcc;         /* reduced write current cyl */
                    189:        unsigned short  d_wpcc;         /* write pre-compensation cyl */
                    190:        unsigned char   d_eccl;         /* max ecc data length */
                    191:        unsigned char   d_ctrl;         /* control byte */
                    192:        unsigned char   d_fill2[3];
                    193:        unsigned short  d_landc;        /* landing zone cylinder */
                    194:        unsigned char   d_nspt;         /* number of sectors per track */
                    195:        unsigned char   d_fill3;
                    196: 
                    197: }      atparm[ NDRIVE ] = {
                    198:        0                               /* Initialized to allow patching */
                    199: };
                    200: 
                    201: /*
                    202:  * Partition Parameters - copied from disk.
                    203:  *
                    204:  *     There are NDRIVE * NPARTN positions for the user partitions,
                    205:  *     plus NDRIVE additional partitions to span each drive.
                    206:  *
                    207:  *     Aligning partitions on cylinder boundaries:
                    208:  *     Optimal partition size: 2 * 3 * 4 * 5 * 7 * 17 = 14280 blocks
                    209:  *     Acceptable partition size:  3 * 4 * 5 * 7 * 17 =  7140 blocks
                    210:  */
                    211: static
                    212: struct fdisk_s pparm[NDRIVE*NPARTN + NDRIVE];
                    213: 
                    214: /*
                    215:  * Per disk controller data.
                    216:  * Only one controller; no more, no less.
                    217:  */
                    218: static
                    219: struct at      {
                    220:        BUF             *at_actf;       /* Link to first */
                    221:        BUF             *at_actl;       /* Link to last */
                    222:        faddr_t         at_faddr;       /* Source/Dest virtual address */
                    223:        daddr_t         at_bno;         /* Block # on disk */
                    224:        unsigned        at_nsec;        /* # of sectors on current transfer */
                    225:        unsigned        at_drv;
                    226:        unsigned        at_head;
                    227:        unsigned        at_cyl;
                    228:        unsigned        at_sec;
                    229:        unsigned        at_partn;
                    230:        unsigned char   at_dtype[ NDRIVE ];     /* drive type, 0 if unused */
                    231:        unsigned char   at_tries;
                    232:        unsigned char   at_state;
                    233:        unsigned char   at_caching;             /* caching in progress */
                    234: #if    ATCACHE > 0
                    235:        unsigned char   at_cdrv[ ATCACHE ];     /* cached drive */
                    236:        daddr_t         at_cbno[ ATCACHE ];     /* cached block number */
                    237:        unsigned char * at_cbuf[ ATCACHE ];     /* cached block */
                    238: #endif
                    239:        unsigned        at_bad_drv;
                    240:        unsigned        at_bad_head;
                    241:        unsigned        at_bad_cyl;
                    242: }      at;
                    243: 
                    244: static BUF     dbuf;                   /* For raw I/O */
                    245: 
                    246: /*
                    247:  * Patchable variables.
                    248:  *     ATBSYW is a loop count for busy-waiting after issuing commands.
                    249:  *     ATSECS is number of seconds to wait for an expected interrupt.
                    250:  *     ATSREG needs to be 3F6 for most new IDE drives;  needs to be
                    251:  *             1F7 for Perstor controllers and some old IDE drives.
                    252:  *             Either value works with many drives.
                    253:  */
                    254: int    ATBSYW = 50;                    /* patchable */
                    255: int    ATSECS = 6;                     /* patchable */
                    256: int    ATSREG = HF_REG;                /* patchable */
                    257: static char timeout_msg[] = "at%d: TO\n";
                    258: 
                    259: /**
                    260:  *
                    261:  * void
                    262:  * atload()    - load routine.
                    263:  *
                    264:  *     Action: The controller is reset and the interrupt vector is grabbed.
                    265:  *             The drive characteristics are set up at this time.
                    266:  */
                    267: static void
                    268: atload()
                    269: {
                    270:        unsigned int u;
                    271:        struct dparm_s * dp;
                    272:        struct { unsigned off, seg; } p;
                    273: 
                    274:        if (n_atdr == 0)
                    275:                return;
                    276: 
                    277:        /*
                    278:         * Obtain Drive Types.
                    279:         *
                    280:         *      High nibble of CMOS 0x12 is drive 0's type.
                    281:         *      Low  nibble of CMOS 0x12 is drive 1's type.
                    282:         */
                    283:        outb(CMOSA, 0x12);
                    284:        /* delay */
                    285:        u = inb(CMOSD);
                    286:        at.at_dtype[0] = u >> 4;
                    287:        at.at_dtype[1] = u & 15;
                    288: 
                    289: 
                    290:        /*
                    291:         * Obtain Drive Characteristics.
                    292:         */
                    293:        for (u = 0, dp = &atparm[0]; u < n_atdr; ++dp, ++u) {
                    294:                struct dparm_s int_dp;
                    295: 
                    296:                if (dp->d_ncyl == 0) {
                    297:                        /*
                    298:                         * Not patched.
                    299:                         *
                    300:                         * If tertiary boot sent us parameters,
                    301:                         *   Use "fifo" routines to fetch them.
                    302:                         *   This only gives us ncyl, nhead, and nspt.
                    303:                         *   Make educated guesses for other parameters:
                    304:                         *   Set landc to ncyl, wpcc to -1.
                    305:                         *   Set ctrl to 0 or 8 depending on head count.
                    306:                         *
                    307:                         * Follow INT 0x41/46 to get drive static BIOS drive
                    308:                         * parameters, if any.
                    309:                         *
                    310:                         * If there were no parameters from tertiary boot,
                    311:                         * or if INT 0x4? nhead and nspt match tboot parms,
                    312:                         *   use "INT" parameters (will give better match on
                    313:                         *   wpcc, landc, and ctrl fields, which tboot can't
                    314:                         *   give us).
                    315:                         */
                    316: 
                    317:                        FIFO *ffp;
                    318:                        typed_space *tp;
                    319:                        int found, parm_int;
                    320:                        extern typed_space boot_gift;
                    321: 
                    322:                if (F_NULL != (ffp = fifo_open(&boot_gift, 0))) {
                    323: 
                    324:                        for (found = 0;
                    325:                        !found && T_NULL != (tp = fifo_read(ffp));
                    326:                        ) {
                    327:                                BIOS_DISK *bdp = (BIOS_DISK *)tp->ts_data;
                    328:                                if ((T_BIOS_DISK == tp->ts_type) &&
                    329:                                    (u == bdp->dp_drive) ) {
                    330:                                        found = 1;
                    331:                                        dp->d_ncyl = bdp->dp_cylinders;
                    332:                                        dp->d_nhead = bdp->dp_heads;
                    333:                                        dp->d_nspt = bdp->dp_sectors;
                    334:                                        dp->d_wpcc = 0xffff;
                    335:                                        dp->d_landc = dp->d_ncyl;
                    336:                                        if (dp->d_nhead > 8)
                    337:                                                dp->d_ctrl |= 8;
                    338:                                }
                    339:                        }
                    340:                        fifo_close(ffp);
                    341:                }
                    342: 
                    343:                        if (u == 0)
                    344:                                parm_int = 0x41;
                    345:                        else /* (u == 1) */
                    346:                                parm_int = 0x46;
                    347:                        pkcopy((paddr_t)(parm_int*4), &p, sizeof p);
                    348:                        pkcopy((paddr_t) (p.seg << 4L) + p.off,
                    349:                                &int_dp, sizeof(int_dp));
                    350:                        if (!found ||
                    351:                            (dp->d_nhead == int_dp.d_nhead
                    352:                             && dp->d_nspt == int_dp.d_nspt)) {
                    353:                             *dp = int_dp;
                    354:                                printf("Using INT 0x%x",parm_int);
                    355:                        } else
                    356:                                printf("Using INT 0x13(08)");
                    357:                } else {
                    358:                        printf("Using patched");
                    359:                        /*
                    360:                         * Avoid incomplete patching.
                    361:                         */
                    362:                        if (at.at_dtype[u] == 0)
                    363:                                at.at_dtype[u] = 1;
                    364:                        if (dp->d_nspt == 0)
                    365:                                dp->d_nspt = 17;
                    366: #if FORCE_CTRL_8
                    367:                        if (dp->d_nhead > 8)
                    368:                                dp->d_ctrl |= 8;
                    369: #endif
                    370: 
                    371:                }
                    372: #if VERBOSE > 0
                    373:        printf(" drive %d parameters\n", u);
                    374: 
                    375:        /* intersegment printf only gets 6 words of arguments */        
                    376:        printf( "at%d: ncyl=%d nhead=%d wpcc=%d ",
                    377:          u, dp->d_ncyl, dp->d_nhead, dp->d_wpcc);
                    378:        printf(" eccl=%d ctrl=%d landc=%d nspt=%d\n",
                    379:          dp->d_eccl, dp->d_ctrl, dp->d_landc, dp->d_nspt);
                    380: #endif
                    381:        }
                    382: 
                    383:        /*
                    384:         * Initialize Drive Size.
                    385:         */
                    386:        for (u = 0, dp = &atparm[0]; u < n_atdr; ++dp, ++u) {
                    387: 
                    388:                if (at.at_dtype[u] == 0)
                    389:                        continue;
                    390: 
                    391:                pparm[NDRIVE*NPARTN + u].p_size =
                    392:                        (long) dp->d_ncyl * dp->d_nhead * dp->d_nspt;
                    393:        }
                    394: 
                    395:        /*
                    396:         * Initialize Drive Controller.
                    397:         */
                    398:        atreset();
                    399: 
                    400:        setivec(HDIRQ, atintr);
                    401: 
                    402: #if ATCACHE > 0
                    403:        at.at_cdrv[0] = -1;
                    404:        at.at_cbuf[0] = kalloc(BSIZE);
                    405: #endif
                    406: 
                    407: #if ATCACHE > 1
                    408:        at.at_cdrv[1] = -1;
                    409:        at.at_cbuf[1] = kalloc(BSIZE);
                    410: #endif
                    411: 
                    412:        at.at_bad_drv = -1;
                    413: }
                    414: 
                    415: /**
                    416:  *
                    417:  * void
                    418:  * atunload()  - unload routine.
                    419:  */
                    420: static void
                    421: atunload()
                    422: {
                    423:        clrivec(HDIRQ);
                    424: }
                    425: 
                    426: /*
                    427:  * atreset()   -- reset hard disk controller, define drive characteristics.
                    428:  *
                    429:  * Return 0 if controller apparently not found, else return 0.
                    430:  */
                    431: static int atreset()
                    432: {
                    433:        register int u;
                    434:        register struct dparm_s * dp;
                    435:        int ret = 1;
                    436: 
                    437:        /*
                    438:         * Reset controller for a minimum of 4.8 microseconds.
                    439:         */
                    440:        outb(HF_REG, 4);
                    441:        for (u = 100; --u != 0;)
                    442:                ;
                    443:        outb(HF_REG, atparm[0].d_ctrl & 0x0F);
                    444:        myatbsyw(0);
                    445:        if (inb(AUX_REG) != 0x01) {
                    446:                /*
                    447:                 * Some IDE drives always timeout on initial reset.
                    448:                 * So don't report first timeout.
                    449:                 */
                    450:                static one_bad;
                    451: 
                    452:                if (one_bad) {
                    453:                        printf("at: hd controller reset timeout\n");
                    454:                } else
                    455:                        one_bad = 1;
                    456:                ret = 0;
                    457:        }
                    458: 
                    459:        /*
                    460:         * Initialize drive parameters.
                    461:         */
                    462:        for (u = 0, dp = &atparm[0]; u < n_atdr; ++dp, ++u) {
                    463: 
                    464:                if (at.at_dtype[u] == 0)
                    465:                        continue;
                    466: 
                    467:                myatbsyw(u);
                    468: 
                    469:                /*
                    470:                 * Set drive characteristics.
                    471:                 * 0x1F1 - AUX_REG
                    472:                 * 0x1F2 - NSEC_REG
                    473:                 * 0x1F3 - SEC_REG
                    474:                 * 0x1F4 - LCYL_REG
                    475:                 * 0x1F5 - HCYL_REG
                    476:                 * 0x1F6 - HDRV_REG
                    477:                 * 0x1F7 - CSR_REG
                    478:                 */
                    479:                outb(HF_REG,    dp->d_ctrl);
                    480:                outb(AUX_REG,  dp->d_wpcc / 4);
                    481:                outb(NSEC_REG, dp->d_nspt);
                    482:                outb(SEC_REG, 0x01);
                    483:                outb(LCYL_REG, (char)(dp->d_ncyl));
                    484:                outb(HCYL_REG, (char)(dp->d_ncyl >> 8));
                    485:                outb(HDRV_REG, 0xA0 + (u<<4) + dp->d_nhead - 1);
                    486:                outb(CSR_REG,  SETPARM_CMD);
                    487:                myatbsyw(u);
                    488: 
                    489:                /*
                    490:                 * Restore heads.
                    491:                 */
                    492:                outb(CSR_REG, RESTORE(0));
                    493:                myatbsyw(u);
                    494:        }
                    495:        return ret;
                    496: }
                    497: 
                    498: /**
                    499:  *
                    500:  * void
                    501:  * atopen(dev, mode)
                    502:  * dev_t dev;
                    503:  * int mode;
                    504:  *
                    505:  *     Input:  dev = disk device to be opened.
                    506:  *             mode = access mode [IPR,IPW, IPR+IPW].
                    507:  *
                    508:  *     Action: Validate the minor device.
                    509:  *             Update the paritition table if necessary.
                    510:  */
                    511: static void
                    512: atopen(dev, mode)
                    513: register dev_t dev;
                    514: {
                    515:        register int d;         /* drive */
                    516:        register int p;         /* partition */
                    517: 
                    518:        p = minor(dev) % (NDRIVE*NPARTN);
                    519: 
                    520:        if (minor(dev) & SDEV) {
                    521:                d = minor(dev) % NDRIVE;
                    522:                p += NDRIVE * NPARTN;
                    523:        }
                    524:        else
                    525:                d = minor(dev) / NPARTN;
                    526: 
                    527:        if ((d >= NDRIVE) || (at.at_dtype[d] == 0)) {
                    528:                u.u_error = ENXIO;
                    529:                return;
                    530:        }
                    531: 
                    532:        if (minor(dev) & SDEV)
                    533:                return;
                    534: 
                    535:        /*
                    536:         * If partition not defined read partition characteristics.
                    537:         */
                    538:        if (pparm[p].p_size == 0)
                    539:                fdisk(makedev(major(dev), SDEV + d), &pparm[ d * NPARTN ]);
                    540: 
                    541:        /*
                    542:         * Ensure partition lies within drive boundaries and is non-zero size.
                    543:         */
                    544:        if ((pparm[p].p_base+pparm[p].p_size) > pparm[d+NDRIVE*NPARTN].p_size)
                    545:                u.u_error = EBADFMT;
                    546:        else if (pparm[p].p_size == 0)
                    547:                u.u_error = ENODEV;
                    548: }
                    549: 
                    550: /**
                    551:  *
                    552:  * void
                    553:  * atread(dev, iop)    - write a block to the raw disk
                    554:  * dev_t dev;
                    555:  * IO * iop;
                    556:  *
                    557:  *     Input:  dev = disk device to be written to.
                    558:  *             iop = pointer to source I/O structure.
                    559:  *
                    560:  *     Action: Invoke the common raw I/O processing code.
                    561:  */
                    562: static void
                    563: atread(dev, iop)
                    564: dev_t  dev;
                    565: IO     *iop;
                    566: {
                    567:        ioreq(&dbuf, iop, dev, BREAD, BFRAW|BFBLK|BFIOC);
                    568: }
                    569: 
                    570: /**
                    571:  *
                    572:  * void
                    573:  * atwrite(dev, iop)   - write a block to the raw disk
                    574:  * dev_t dev;
                    575:  * IO * iop;
                    576:  *
                    577:  *     Input:  dev = disk device to be written to.
                    578:  *             iop = pointer to source I/O structure.
                    579:  *
                    580:  *     Action: Invoke the common raw I/O processing code.
                    581:  */
                    582: static void
                    583: atwrite(dev, iop)
                    584: dev_t  dev;
                    585: IO     *iop;
                    586: {
                    587:        ioreq(&dbuf, iop, dev, BWRITE, BFRAW|BFBLK|BFIOC);
                    588: }
                    589: 
                    590: /**
                    591:  *
                    592:  * int
                    593:  * atioctl(dev, cmd, arg)
                    594:  * dev_t dev;
                    595:  * int cmd;
                    596:  * char * vec;
                    597:  *
                    598:  *     Input:  dev = disk device to be operated on.
                    599:  *             cmd = input/output request to be performed.
                    600:  *             vec = (pointer to) optional argument.
                    601:  *
                    602:  *     Action: Validate the minor device.
                    603:  *             Update the paritition table if necessary.
                    604:  */
                    605: static int
                    606: atioctl(dev, cmd, vec)
                    607: register dev_t dev;
                    608: int cmd;
                    609: char * vec;
                    610: {
                    611:        int d;
                    612: 
                    613:        /*
                    614:         * Identify drive number.
                    615:         */
                    616:        if (minor(dev) & SDEV)
                    617:                d = minor(dev) % NDRIVE;
                    618:        else
                    619:                d = minor(dev) / NPARTN;
                    620: 
                    621:        /*
                    622:         * Identify input/output request.
                    623:         */
                    624:        switch (cmd) {
                    625: 
                    626:        case HDGETA:
                    627:                /*
                    628:                 * Get hard disk attributes.
                    629:                 */
                    630:                kucopy(&atparm[d], vec, sizeof(atparm[0]));
                    631:                return(0);
                    632: 
                    633:        case HDSETA:
                    634:                /* Set hard disk attributes. */
                    635:                ukcopy(vec, &atparm[d], sizeof(atparm[0]));
                    636:                at.at_dtype[d] = 1;             /* set drive type nonzero */
                    637:                pparm[NDRIVE * NPARTN + d].p_size =
                    638:                        (long) atparm[d].d_ncyl * atparm[d].d_nhead * atparm[d].d_nspt;
                    639:                atreset();
                    640:                return 0;
                    641: 
                    642:        default:
                    643:                u.u_error = EINVAL;
                    644:                return(-1);
                    645:        }
                    646: }
                    647: 
                    648: /**
                    649:  *
                    650:  * void
                    651:  * atwatch()           - guard against lost interrupt
                    652:  *
                    653:  *     Action: If drvl[AT_MAJOR] is greater than zero, decrement it.
                    654:  *             If it decrements to zero, simulate a hardware interrupt.
                    655:  */
                    656: static void
                    657: atwatch()
                    658: {
                    659:        register BUF * bp = at.at_actf;
                    660:        register int s;
                    661: 
                    662:        s = sphi();
                    663:        if (--drvl[AT_MAJOR].d_time > 0) {
                    664:                spl(s);
                    665:                return;
                    666:        }
                    667:        printf("at%d%c: bno=%U head=%u cyl=%u <Watchdog Timeout>\n",
                    668:                at.at_drv,
                    669:                (bp->b_dev & SDEV) ? 'x' : at.at_partn % NPARTN + 'a',
                    670:                bp->b_bno, at.at_head, at.at_cyl);
                    671: 
                    672:        /*
                    673:         * Reset hard disk controller.
                    674:         *
                    675:         * Mark current cylinder as bad so atstart() will fail.
                    676:         * Otherwise would lock up if this track NEVER gives enough IRQ's.
                    677:         */
                    678:        at.at_bad_drv   = at.at_drv;
                    679:        at.at_bad_head  = at.at_head;
                    680:        at.at_bad_cyl   = at.at_cyl;
                    681:        atreset();
                    682:        atstart();
                    683:        spl(s);
                    684: }
                    685: 
                    686: /**
                    687:  *
                    688:  * void
                    689:  * atblock(bp) - queue a block to the disk
                    690:  *
                    691:  *     Input:  bp = pointer to block to be queued.
                    692:  *
                    693:  *     Action: Queue a block to the disk.
                    694:  *             Make sure that the transfer is within the disk partition.
                    695:  */
                    696: static void
                    697: atblock(bp)
                    698: register BUF   *bp;
                    699: {
                    700:        register struct fdisk_s *pp;
                    701:        int partn = minor(bp->b_dev) % (NDRIVE*NPARTN);
                    702: 
                    703:        bp->b_resid = bp->b_count;
                    704: 
                    705:        if (minor(bp->b_dev) & SDEV)
                    706:                partn += NDRIVE * NPARTN;
                    707: 
                    708:        pp = &pparm[ partn ];
                    709: 
                    710:        /*
                    711:         * Check for read at end of partition.
                    712:         */
                    713:        if ((bp->b_req == BREAD) && (bp->b_bno == pp->p_size)) {
                    714:                bdone(bp);
                    715:                return;
                    716:        }
                    717: 
                    718:        /*
                    719:         * Range check disk region.
                    720:         */
                    721:        if (((bp->b_bno + (bp->b_count/BSIZE)) > pp->p_size)
                    722:        || (bp->b_count % BSIZE) || bp->b_count == 0) {
                    723:                bp->b_flag |= BFERR;
                    724:                bdone(bp);
                    725:                return;
                    726:        }
                    727: 
                    728:        bp->b_actf = NULL;
                    729:        if (at.at_actf == NULL)
                    730:                at.at_actf = bp;
                    731:        else
                    732:                at.at_actl->b_actf = bp;
                    733:        at.at_actl = bp;
                    734: 
                    735:        if (at.at_state == SIDLE)
                    736:                if (atdequeue())
                    737:                        atstart();
                    738: }
                    739: 
                    740: /**
                    741:  *
                    742:  * int
                    743:  * atdequeue()         - obtain next disk read/write operation
                    744:  *
                    745:  *     Action: Pull some work from the disk queue.
                    746:  *
                    747:  *     Return: 0 = no work.
                    748:  *             * = work to do.
                    749:  */
                    750: static int
                    751: atdequeue()
                    752: {
                    753:        register BUF * bp;
                    754:        register struct fdisk_s * pp;
                    755:        unsigned int nspt;
                    756: 
                    757:        for (;;) {
                    758:                at.at_caching = 0;
                    759:                at.at_tries   = 0;
                    760: 
                    761:                if ((bp = at.at_actf) == NULL)
                    762:                        return (0);
                    763: 
                    764:                at.at_partn = minor(bp->b_dev) % (NDRIVE*NPARTN);
                    765: 
                    766:                if (minor(bp->b_dev) & SDEV) {
                    767:                        at.at_partn += (NDRIVE*NPARTN);
                    768:                        at.at_drv  = minor(bp->b_dev) % NDRIVE;
                    769:                }
                    770:                else
                    771:                        at.at_drv = minor(bp->b_dev) / NPARTN;
                    772:                nspt = atparm[at.at_drv].d_nspt;
                    773: 
                    774:                pp = &pparm[ at.at_partn ];
                    775:                at.at_bno   = pp->p_base + bp->b_bno;
                    776:                at.at_nsec  = bp->b_count / BSIZE;
                    777:                at.at_faddr = bp->b_faddr;
                    778: 
                    779: #if ATCACHE > 0
                    780:                if (bp->b_req == BWRITE) {
                    781: 
                    782:                        /*
                    783:                         * Invalidate cache if write might overlap.
                    784:                         */
                    785:                        if (at.at_nsec > 1) {
                    786:                                at.at_cdrv[0] = -1;
                    787: #if ATCACHE > 1
                    788:                                at.at_cdrv[1] = -1;
                    789: #endif
                    790:                        }
                    791:                        else if (at.at_bno == at.at_cbno[0])
                    792:                                at.at_cdrv[0] = -1;
                    793: #if ATCACHE > 1
                    794:                        else if (at.at_bno == at.at_cbno[1])
                    795:                                at.at_cdrv[1] = -1;
                    796: #endif
                    797:                }
                    798:                else if (at.at_nsec == 1) {
                    799: 
                    800:                        /*
                    801:                         * Test for cache hit on block 0.
                    802:                         */
                    803:                        if ((at.at_drv == at.at_cdrv[0])
                    804:                        &&   (at.at_bno == at.at_cbno[0])) {
                    805: 
                    806:                                kpcopy(at.at_cbuf[0],
                    807:                                        bp->b_paddr, BSIZE);
                    808:                                at.at_actf  = bp->b_actf;
                    809:                                bp->b_resid = 0;
                    810:                                bdone(bp);
                    811:                                continue;
                    812:                        }
                    813: 
                    814: #if ATCACHE > 1
                    815:                        /*
                    816:                         * Test for cache hit on block 1.
                    817:                         */
                    818:                        if ((at.at_drv == at.at_cdrv[1])
                    819:                        &&   (at.at_bno == at.at_cbno[1])) {
                    820: 
                    821:                                kpcopy(at.at_cbuf[1],
                    822:                                        bp->b_paddr, BSIZE);
                    823:                                at.at_actf  = bp->b_actf;
                    824:                                bp->b_resid = 0;
                    825:                                bdone(bp);
                    826:                                continue;
                    827:                        }
                    828: #endif
                    829: 
                    830:                        /*
                    831:                         * Enable caching if no backlog for disk i/o.
                    832:                         */
                    833:                        if (bp->b_actf == NULL) {
                    834:                                /*
                    835:                                 * Enable caching on single block reads
                    836:                                 * when at least one block left on same track.
                    837:                                 */
                    838:                                at.at_caching = nspt - 1 - (at.at_bno % nspt);
                    839: #if ATCACHE > 1
                    840:                                if (at.at_caching >= 2) {
                    841:                                        at.at_caching   = 2;
                    842:                                        at.at_cdrv[2-1] = -1;
                    843:                                }
                    844: #endif
                    845: 
                    846:                                if (at.at_caching) {
                    847:                                        at.at_nsec  += at.at_caching;
                    848:                                        at.at_cdrv[1-1] = -1;
                    849:                                }
                    850:                        }
                    851:                }
                    852: #endif
                    853: 
                    854:                return (1);
                    855:        }
                    856: }
                    857: 
                    858: /**
                    859:  *
                    860:  * void
                    861:  * atstart()   - start or restart next disk read/write operation.
                    862:  *
                    863:  *     Action: Initiate disk read/write operation.
                    864:  */
                    865: static void
                    866: atstart()
                    867: {
                    868:        register struct dparm_s *dp;
                    869: 
                    870:        dp = &atparm[ at.at_drv ];
                    871: 
                    872:        at.at_cyl  = (at.at_bno / dp->d_nspt) / dp->d_nhead;
                    873:        at.at_head = (at.at_bno / dp->d_nspt) % dp->d_nhead;
                    874:        at.at_sec  = (at.at_bno % dp->d_nspt) + 1;
                    875: 
                    876:        /*
                    877:         * Check for repeated access to most recently identified bad track.
                    878:         */
                    879:        if ((at.at_drv  == at.at_bad_drv)
                    880:          && (at.at_cyl  == at.at_bad_cyl)
                    881:          && (at.at_head == at.at_bad_head)) {
                    882:                BUF * bp = at.at_actf;
                    883:                printf("at%d%c: bno=%U head=%u cyl=%u <Track Flagged Bad>\n",
                    884:                        at.at_drv,
                    885:                        (bp->b_dev & SDEV) ? 'x' : at.at_partn % NPARTN + 'a',
                    886:                        bp->b_bno,
                    887:                        at.at_head,
                    888:                        at.at_cyl);
                    889:                bp->b_flag |= BFERR;
                    890:                atdone(bp);
                    891:                return;
                    892:        }
                    893: 
                    894:        myatbsyw(at.at_drv);
                    895: 
                    896:        outb(HF_REG,   dp->d_ctrl);
                    897:        outb(AUX_REG,  dp->d_wpcc / 4);
                    898:        outb(NSEC_REG, at.at_nsec);
                    899:        outb(SEC_REG,  at.at_sec);
                    900:        outb(LCYL_REG, at.at_cyl);
                    901:        outb(HCYL_REG, at.at_cyl >> 8);
                    902:        outb(HDRV_REG, (at.at_drv << 4) + at.at_head + 0xA0);
                    903: 
                    904:        if (at.at_actf->b_req == BWRITE) {
                    905: 
                    906:                outb(CSR_REG, WRITE_CMD);
                    907: 
                    908:                while (atdrqw() == 0)
                    909:                        printf(timeout_msg, at.at_drv);
                    910: 
                    911:                atsend(at.at_faddr);
                    912:                at.at_state = SWRITE;
                    913:        }
                    914:        else {
                    915:                outb(CSR_REG, READ_CMD);
                    916:                at.at_state = SREAD;
                    917:        }
                    918:        drvl[AT_MAJOR].d_time = ATSECS;
                    919: }
                    920: 
                    921: /**
                    922:  *
                    923:  * void
                    924:  * atintr()    - Interrupt routine.
                    925:  *
                    926:  *     Clear interrupt then defer actual processing.
                    927:  */
                    928: static void
                    929: atintr()
                    930: {
                    931:        inb(CSR_REG);           /* clears controller interrupt */
                    932:        defer(atdefer, 0);
                    933: }
                    934: 
                    935: /**
                    936:  *
                    937:  * void
                    938:  * atdefer()   - Deferred service of hard disk interrupt.
                    939:  *
                    940:  *     Action: Service disk interrupt.
                    941:  *             Transfer required data.
                    942:  *             Update state.
                    943:  */
                    944: static void
                    945: atdefer()
                    946: {
                    947:        register BUF * bp = at.at_actf;
                    948: 
                    949:        switch (at.at_state) {
                    950: 
                    951:        case SRETRY:
                    952:                atstart();
                    953:                break;
                    954: 
                    955:        case SREAD:
                    956:                /*
                    957:                 * Check for I/O error before waiting for data.
                    958:                 */
                    959:                if (aterror()) {
                    960:                        atrecov();
                    961:                        break;
                    962:                }
                    963: 
                    964:                /*
                    965:                 * Wait for data, or forever.
                    966:                 */
                    967:                if (atdrqw() == 0)
                    968:                        printf(timeout_msg, at.at_drv);
                    969: 
                    970: #if ATCACHE > 0
                    971:                /*
                    972:                 * Cache data block.
                    973:                 */
                    974:                if (at.at_caching == at.at_nsec)
                    975:                        atrecv(at.at_cbuf[ at.at_nsec - 1 ], sds);
                    976:                else
                    977: #endif
                    978: 
                    979:                /*
                    980:                 * Read data block.
                    981:                 */
                    982:                        atrecv(at.at_faddr);
                    983: 
                    984:                /*
                    985:                 * Check for I/O error after reading data.
                    986:                 */
                    987:                if (aterror()) {
                    988:                        atrecov();
                    989:                        break;
                    990:                }
                    991: 
                    992: #if ATCACHE > 0
                    993:                /*
                    994:                 * Validate cached blocks.
                    995:                 */
                    996:                if (at.at_caching == at.at_nsec) {
                    997:                        at.at_cbno[ at.at_nsec - 1 ] = at.at_bno;
                    998:                        at.at_cdrv[ at.at_nsec - 1 ] = at.at_drv;
                    999:                        at.at_caching--;
                   1000:                }
                   1001:                else
                   1002: #endif
                   1003:                {
                   1004:                        FP_OFF(at.at_faddr) += BSIZE;
                   1005:                        bp->b_resid -= BSIZE;
                   1006:                }
                   1007: 
                   1008:                at.at_tries = 0;
                   1009:                at.at_bno++;
                   1010: 
                   1011:                /*
                   1012:                 * Check for end of transfer.
                   1013:                 */
                   1014:                if (--at.at_nsec == 0)
                   1015:                        atdone(bp);
                   1016:                break;
                   1017: 
                   1018:        case SWRITE:
                   1019:                /*
                   1020:                 * Check for I/O error.
                   1021:                 */
                   1022:                if (aterror()) {
                   1023:                        atrecov();
                   1024:                        break;
                   1025:                }
                   1026: 
                   1027:                FP_OFF(at.at_faddr) += BSIZE;
                   1028:                bp->b_resid -= BSIZE;
                   1029:                at.at_tries  = 0;
                   1030:                at.at_bno++;
                   1031: 
                   1032:                /*
                   1033:                 * Check for end of transfer.
                   1034:                 */
                   1035:                if (--at.at_nsec == 0) {
                   1036:                        atdone(bp);
                   1037:                        break;
                   1038:                }
                   1039: 
                   1040:                /*
                   1041:                 * Wait for ability to send data, or forever.
                   1042:                 */
                   1043:                while (atdrqw() == 0)
                   1044:                        printf(timeout_msg, at.at_drv);
                   1045: 
                   1046:                /*
                   1047:                 * Send data block.
                   1048:                 */
                   1049:                atsend(at.at_faddr);
                   1050:        }
                   1051: }
                   1052: 
                   1053: /**
                   1054:  *
                   1055:  * int
                   1056:  * aterror()
                   1057:  *
                   1058:  *     Action: Check for drive error.
                   1059:  *             If found, increment error count and report it.
                   1060:  *
                   1061:  *     Return: 0 = No error found.
                   1062:  *             1 = Error occurred.
                   1063:  */
                   1064: static int
                   1065: aterror()
                   1066: {
                   1067:        register BUF * bp = at.at_actf;
                   1068:        register int csr;
                   1069:        register int aux;
                   1070: 
                   1071:        if ((csr = inb(ATSREG)) & (ERR_ST|WFLT_ST)) {
                   1072: 
                   1073:                aux = inb(AUX_REG);
                   1074: 
                   1075:                /*
                   1076:                 * Don't retry or report failures on cache reads.
                   1077:                 */
                   1078: #if ATCACHE > 0
                   1079:                if ((at.at_state == SREAD) && (at.at_caching == at.at_nsec)) {
                   1080:                        at.at_tries = BADLIM;
                   1081:                        return 1;
                   1082:                }
                   1083: #endif
                   1084: 
                   1085:                if (aux & BAD_ERR) {
                   1086:                        at.at_tries     = BADLIM;
                   1087:                        at.at_bad_drv   = at.at_drv;
                   1088:                        at.at_bad_head  = at.at_head;
                   1089:                        at.at_bad_cyl   = at.at_cyl;
                   1090:                }
                   1091:                else if (++at.at_tries < SOFTLIM)
                   1092:                        return 1;
                   1093: 
                   1094:                printf("at%d%c: bno=%U head=%u cyl=%u",
                   1095:                        at.at_drv,
                   1096:                        (bp->b_dev & SDEV) ? 'x' : at.at_partn % NPARTN + 'a',
                   1097:                        (bp->b_count/BSIZE) + bp->b_bno
                   1098:                                + at.at_caching - at.at_nsec,
                   1099:                        at.at_head, at.at_cyl);
                   1100: 
                   1101: #if VERBOSE > 0
                   1102:                if ((csr & RDY_ST) == 0)
                   1103:                        printf(" <Drive Not Ready>");
                   1104:                if (csr & WFLT_ST)
                   1105:                        printf(" <Write Fault>");
                   1106: 
                   1107:                if (aux & DAM_ERR)
                   1108:                        printf(" <No Data Addr Mark>");
                   1109:                if (aux & TR0_ERR)
                   1110:                        printf(" <Track 0 Not Found>");
                   1111:                if (aux & ID_ERR)
                   1112:                        printf(" <ID Not Found>");
                   1113:                if (aux & ECC_ERR)
                   1114:                        printf(" <Bad Data Checksum>");
                   1115:                if (aux & ABT_ERR)
                   1116:                        printf(" <Command Aborted>");
                   1117: #else
                   1118:                if ((csr & (RDY_ST|WFLT_ST)) != RDY_ST)
                   1119:                        printf(" csr=%x", csr);
                   1120:                if (aux & (DAM_ERR|TR0_ERR|ID_ERR|ECC_ERR|ABT_ERR))
                   1121:                        printf(" aux=%x", aux);
                   1122: #endif
                   1123:                if (aux & BAD_ERR)
                   1124:                        printf(" <Block Flagged Bad>");
                   1125: 
                   1126:                if (at.at_tries < HARDLIM)
                   1127:                        printf(" retrying...");
                   1128:                printf("\n");
                   1129:                return 1;
                   1130:        }
                   1131:        return 0;
                   1132: }
                   1133: 
                   1134: /**
                   1135:  *
                   1136:  * void
                   1137:  * atrecov()
                   1138:  *
                   1139:  *     Action: Attempt recovery.
                   1140:  */
                   1141: static void
                   1142: atrecov()
                   1143: {
                   1144:        register BUF *bp = at.at_actf;
                   1145:        register int cmd = SEEK(0);
                   1146:        register int cyl = at.at_cyl;
                   1147: 
                   1148:        switch (at.at_tries) {
                   1149: 
                   1150:        case 1:
                   1151:        case 2:
                   1152:                /*
                   1153:                 * Move in 1 cylinder, then retry operation
                   1154:                 */
                   1155:                if (--cyl < 0)
                   1156:                        cyl += 2;
                   1157:                break;
                   1158: 
                   1159:        case 3:
                   1160:        case 4:
                   1161:                /*
                   1162:                 * Move out 1 cylinder, then retry operation
                   1163:                 */
                   1164:                if (++cyl >= atparm[ at.at_drv ].d_ncyl)
                   1165:                        cyl -= 2;
                   1166:                break;
                   1167: 
                   1168:        case 5:
                   1169:        case 6:
                   1170:                /*
                   1171:                 * Seek to cylinder 0, then retry operation
                   1172:                 */
                   1173:                cyl = 0;
                   1174:                break;
                   1175: 
                   1176:        default:
                   1177:                /*
                   1178:                 * Restore drive, then retry operation
                   1179:                 */
                   1180:                cmd = RESTORE(0);
                   1181:                cyl = 0;
                   1182:                break;
                   1183:        }
                   1184: 
                   1185:        /*
                   1186:         * Retry operation [after repositioning head]
                   1187:         */
                   1188:        if (at.at_tries < HARDLIM) {
                   1189:                drvl[AT_MAJOR].d_time = (cmd == RESTORE(0))
                   1190:                        ? (ATSECS * 2) : ATSECS;
                   1191:                outb(LCYL_REG, cyl);
                   1192:                outb(HCYL_REG, cyl >> 8);
                   1193:                outb(HDRV_REG, (at.at_drv << 4) + 0xA0);
                   1194:                outb(CSR_REG, cmd);
                   1195:                at.at_state = SRETRY;
                   1196:        }
                   1197: 
                   1198:        /*
                   1199:         * Give up on block.
                   1200:         */
                   1201:        else {
                   1202:                /*
                   1203:                 * Not a cache-read error.
                   1204:                 */
                   1205: #if ATCACHE > 0
                   1206:                if ((at.at_state != SREAD) || (at.at_caching != at.at_nsec))
                   1207: #endif
                   1208:                        bp->b_flag |= BFERR;
                   1209: 
                   1210:                atdone(bp);
                   1211:        }
                   1212: }
                   1213: 
                   1214: /**
                   1215:  *
                   1216:  * void
                   1217:  * atdone(bp)
                   1218:  * BUF * bp;
                   1219:  *
                   1220:  *     Action: Release current i/o buffer to the O/S.
                   1221:  */
                   1222: static void
                   1223: atdone(bp)
                   1224: register BUF * bp;
                   1225: {
                   1226:        drvl[AT_MAJOR].d_time = 0;
                   1227:        at.at_state = SIDLE;
                   1228:        at.at_actf  = bp->b_actf;
                   1229:        bdone(bp);
                   1230: 
                   1231:        if (atdequeue())
                   1232:                atstart();
                   1233: }
                   1234: 
                   1235: int
                   1236: myatbsyw(unit) int unit;
                   1237: {
                   1238:        register int n, status;
                   1239: 
                   1240:        for (n = ATBSYW; n > 0; --n)
                   1241:                if ((status = atbsyw()) != 0)
                   1242:                        return status;
                   1243:        printf(timeout_msg, unit);
                   1244:        return 0;
                   1245: }

unix.superglobalmegacorp.com

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