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

unix.superglobalmegacorp.com

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