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

unix.superglobalmegacorp.com

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