|
|
1.1 ! root 1: /* ! 2: * Copyright (c) 1982, 1986 Regents of the University of California. ! 3: * All rights reserved. The Berkeley software License Agreement ! 4: * specifies the terms and conditions for redistribution. ! 5: * ! 6: * @(#)ht.c 7.11 (Berkeley) 4/12/90 ! 7: */ ! 8: ! 9: #include "tu.h" ! 10: #if NHT > 0 ! 11: /* ! 12: * TM03/TU?? tape driver ! 13: * ! 14: * TODO: ! 15: * cleanup messages on errors ! 16: * test ioctl's ! 17: * see how many rewind interrups we get if we kick when not at BOT ! 18: * fixup rle error on block tape code ! 19: */ ! 20: #include "param.h" ! 21: #include "systm.h" ! 22: #include "buf.h" ! 23: #include "conf.h" ! 24: #include "file.h" ! 25: #include "user.h" ! 26: #include "proc.h" ! 27: #include "map.h" ! 28: #include "ioctl.h" ! 29: #include "mtio.h" ! 30: #include "cmap.h" ! 31: #include "tty.h" ! 32: #include "syslog.h" ! 33: ! 34: #include "machine/pte.h" ! 35: #include "../vax/cpu.h" ! 36: #include "mbareg.h" ! 37: #include "mbavar.h" ! 38: #include "htreg.h" ! 39: ! 40: struct buf chtbuf[NHT]; ! 41: ! 42: short httypes[] = ! 43: { MBDT_TM03, MBDT_TE16, MBDT_TU45, MBDT_TU77, 0 }; ! 44: struct mba_device *htinfo[NHT]; ! 45: struct mba_slave *tuinfo[NTU]; ! 46: int htattach(), htslave(), htustart(), htndtint(), htdtint(); ! 47: struct mba_driver htdriver = ! 48: { htattach, htslave, htustart, 0, htdtint, htndtint, ! 49: httypes, "ht", "tu", htinfo }; ! 50: ! 51: #define MASKREG(r) ((r) & 0xffff) ! 52: ! 53: /* bits in minor device */ ! 54: #define TUUNIT(dev) (minor(dev)&03) ! 55: #define H_NOREWIND 04 ! 56: #define H_DENS(dev) ((minor(dev) >> 3) & 03) ! 57: ! 58: #define HTUNIT(dev) (tuinfo[TUUNIT(dev)]->ms_ctlr) ! 59: ! 60: #define INF (daddr_t)1000000L /* a block number that wont exist */ ! 61: ! 62: struct tu_softc { ! 63: char sc_openf; ! 64: char sc_flags; ! 65: daddr_t sc_blkno; ! 66: daddr_t sc_nxrec; ! 67: u_short sc_erreg; ! 68: u_short sc_dsreg; ! 69: short sc_resid; ! 70: short sc_dens; ! 71: caddr_t sc_ctty; /* record user's tty for errors */ ! 72: int sc_blks; /* number of I/O operations since open */ ! 73: int sc_softerrs; /* number of soft I/O errors since open */ ! 74: } tu_softc[NTU]; ! 75: ! 76: /* ! 77: * Bits for sc_flags. ! 78: */ ! 79: #define H_WRITTEN 1 /* last operation was a write */ ! 80: #define H_ERASED 2 /* last write retry was an erase gap */ ! 81: #define H_REWIND 4 /* last unit start was a rewind */ ! 82: ! 83: char hter_bits[] = HTER_BITS; ! 84: char htds_bits[] = HTDS_BITS; ! 85: ! 86: /*ARGSUSED*/ ! 87: htattach(mi) ! 88: struct mba_device *mi; ! 89: { ! 90: ! 91: } ! 92: ! 93: htslave(mi, ms, sn) ! 94: struct mba_device *mi; ! 95: struct mba_slave *ms; ! 96: int sn; ! 97: { ! 98: register struct htdevice *htaddr = (struct htdevice *)mi->mi_drv; ! 99: ! 100: htaddr->httc = sn; ! 101: if (htaddr->htdt & HTDT_SPR) { ! 102: tuinfo[ms->ms_unit] = ms; ! 103: return (1); ! 104: } else ! 105: return (0); ! 106: } ! 107: ! 108: int htdens[4] = { HTTC_800BPI, HTTC_1600BPI, HTTC_6250BPI, HTTC_800BPI }; ! 109: ! 110: htopen(dev, flag) ! 111: dev_t dev; ! 112: int flag; ! 113: { ! 114: register int tuunit; ! 115: register struct tu_softc *sc; ! 116: register struct mba_slave *ms; ! 117: int olddens, dens; ! 118: ! 119: tuunit = TUUNIT(dev); ! 120: if (tuunit >= NTU || (ms = tuinfo[tuunit]) == NULL || ! 121: ms->ms_alive == 0 || htinfo[ms->ms_ctlr]->mi_alive == 0) ! 122: return (ENXIO); ! 123: if ((sc = &tu_softc[tuunit])->sc_openf) ! 124: return (EBUSY); ! 125: sc->sc_openf = 1; ! 126: olddens = sc->sc_dens; ! 127: dens = sc->sc_dens = htdens[H_DENS(dev)] | HTTC_PDP11 | ms->ms_slave; ! 128: htcommand(dev, HT_SENSE, 1); ! 129: sc->sc_dens = olddens; ! 130: if ((sc->sc_dsreg & HTDS_MOL) == 0) { ! 131: sc->sc_openf = 0; ! 132: uprintf("tu%d: not online\n", tuunit); ! 133: return (EIO); ! 134: } ! 135: if ((flag&FWRITE) && (sc->sc_dsreg&HTDS_WRL)) { ! 136: sc->sc_openf = 0; ! 137: uprintf("tu%d: no write ring\n", tuunit); ! 138: return (EIO); ! 139: } ! 140: if ((sc->sc_dsreg & HTDS_BOT) == 0 && (flag&FWRITE) && ! 141: dens != sc->sc_dens) { ! 142: sc->sc_openf = 0; ! 143: uprintf("tu%d: can't change density in mid-tape\n", tuunit); ! 144: return (EIO); ! 145: } ! 146: sc->sc_blkno = (daddr_t)0; ! 147: sc->sc_nxrec = INF; ! 148: sc->sc_flags = 0; ! 149: sc->sc_dens = dens; ! 150: sc->sc_blks = 0; ! 151: sc->sc_softerrs = 0; ! 152: sc->sc_ctty = (caddr_t)(u.u_procp->p_flag&SCTTY ? ! 153: u.u_procp->p_session->s_ttyp : 0); ! 154: return (0); ! 155: } ! 156: ! 157: htclose(dev, flag) ! 158: register dev_t dev; ! 159: register flag; ! 160: { ! 161: register struct tu_softc *sc = &tu_softc[TUUNIT(dev)]; ! 162: ! 163: if (flag == FWRITE || ((flag&FWRITE) && (sc->sc_flags&H_WRITTEN))) { ! 164: htcommand(dev, HT_WEOF, 1); ! 165: htcommand(dev, HT_WEOF, 1); ! 166: htcommand(dev, HT_SREV, 1); ! 167: } ! 168: if ((minor(dev)&H_NOREWIND) == 0) ! 169: htcommand(dev, HT_REW, 0); ! 170: if (sc->sc_blks > 100 && sc->sc_softerrs > sc->sc_blks / 100) ! 171: log(LOG_INFO, "tu%d: %d soft errors in %d blocks\n", ! 172: TUUNIT(dev), sc->sc_softerrs, sc->sc_blks); ! 173: sc->sc_openf = 0; ! 174: return (0); ! 175: } ! 176: ! 177: htcommand(dev, com, count) ! 178: dev_t dev; ! 179: int com, count; ! 180: { ! 181: register struct buf *bp; ! 182: register int s; ! 183: ! 184: bp = &chtbuf[HTUNIT(dev)]; ! 185: s = spl5(); ! 186: while (bp->b_flags&B_BUSY) { ! 187: if(bp->b_repcnt == 0 && (bp->b_flags&B_DONE)) ! 188: break; ! 189: bp->b_flags |= B_WANTED; ! 190: sleep((caddr_t)bp, PRIBIO); ! 191: } ! 192: bp->b_flags = B_BUSY|B_READ; ! 193: splx(s); ! 194: bp->b_dev = dev; ! 195: bp->b_command = com; ! 196: bp->b_repcnt = count; ! 197: bp->b_blkno = 0; ! 198: htstrategy(bp); ! 199: if (count == 0) ! 200: return; ! 201: iowait(bp); ! 202: if (bp->b_flags&B_WANTED) ! 203: wakeup((caddr_t)bp); ! 204: bp->b_flags &= B_ERROR; ! 205: } ! 206: ! 207: htstrategy(bp) ! 208: register struct buf *bp; ! 209: { ! 210: register struct mba_device *mi = htinfo[HTUNIT(bp->b_dev)]; ! 211: register struct buf *dp; ! 212: register int s; ! 213: ! 214: bp->av_forw = NULL; ! 215: dp = &mi->mi_tab; ! 216: s = spl5(); ! 217: if (dp->b_actf == NULL) ! 218: dp->b_actf = bp; ! 219: else ! 220: dp->b_actl->av_forw = bp; ! 221: dp->b_actl = bp; ! 222: if (dp->b_active == 0) ! 223: mbustart(mi); ! 224: splx(s); ! 225: } ! 226: ! 227: htustart(mi) ! 228: register struct mba_device *mi; ! 229: { ! 230: register struct htdevice *htaddr = ! 231: (struct htdevice *)mi->mi_drv; ! 232: register struct buf *bp = mi->mi_tab.b_actf; ! 233: register struct tu_softc *sc = &tu_softc[TUUNIT(bp->b_dev)]; ! 234: daddr_t blkno; ! 235: ! 236: htaddr->httc = sc->sc_dens; ! 237: #ifdef notdef ! 238: /* unneeded, may hang controller */ ! 239: if (bp == &chtbuf[HTUNIT(bp->b_dev)] && bp->b_command == HT_SENSE) { ! 240: htaddr->htcs1 = HT_SENSE|HT_GO; ! 241: mbclrattn(mi); ! 242: } ! 243: #endif ! 244: sc->sc_dsreg = htaddr->htds; ! 245: sc->sc_erreg = htaddr->hter; ! 246: sc->sc_resid = htaddr->htfc; ! 247: sc->sc_flags &= ~(H_WRITTEN|H_REWIND); ! 248: if ((htaddr->htdt & HTDT_SPR) == 0 || (htaddr->htds & HTDS_MOL) == 0) ! 249: if (sc->sc_openf > 0) ! 250: sc->sc_openf = -1; ! 251: if (sc->sc_openf < 0) { ! 252: bp->b_flags |= B_ERROR; ! 253: return (MBU_NEXT); ! 254: } ! 255: if (bp != &chtbuf[HTUNIT(bp->b_dev)]) { ! 256: /* transfer: check positioning */ ! 257: if (bp->b_flags & B_RAW) { ! 258: /* raw transfer: record position for retry */ ! 259: if (mi->mi_tab.b_errcnt == 0) { ! 260: sc->sc_blkno = bdbtofsb(bp->b_blkno); ! 261: sc->sc_nxrec = sc->sc_blkno + 1; ! 262: } ! 263: } else { ! 264: if (bdbtofsb(bp->b_blkno) > sc->sc_nxrec) { ! 265: bp->b_flags |= B_ERROR; ! 266: bp->b_error = ENXIO; ! 267: return (MBU_NEXT); ! 268: } ! 269: if (bdbtofsb(bp->b_blkno) == sc->sc_nxrec && ! 270: bp->b_flags&B_READ) { ! 271: bp->b_resid = bp->b_bcount; ! 272: clrbuf(bp); ! 273: return (MBU_NEXT); ! 274: } ! 275: if ((bp->b_flags&B_READ)==0) ! 276: sc->sc_nxrec = bdbtofsb(bp->b_blkno) + 1; ! 277: } ! 278: } else { ! 279: if (bp->b_command == HT_SENSE) ! 280: return (MBU_NEXT); ! 281: if (bp->b_command == HT_REW) ! 282: sc->sc_flags |= H_REWIND; ! 283: else ! 284: htaddr->htfc = -bp->b_bcount; ! 285: htaddr->htcs1 = bp->b_command|HT_GO; ! 286: return (MBU_STARTED); ! 287: } ! 288: if ((blkno = sc->sc_blkno) == bdbtofsb(bp->b_blkno)) { ! 289: htaddr->htfc = -bp->b_bcount; ! 290: if ((bp->b_flags&B_READ) == 0) { ! 291: if (mi->mi_tab.b_errcnt) { ! 292: if ((sc->sc_flags & H_ERASED) == 0) { ! 293: sc->sc_flags |= H_ERASED; ! 294: htaddr->htcs1 = HT_ERASE | HT_GO; ! 295: return (MBU_STARTED); ! 296: } ! 297: sc->sc_flags &= ~H_ERASED; ! 298: } ! 299: if (htaddr->htds & HTDS_EOT) { ! 300: bp->b_resid = bp->b_bcount; ! 301: bp->b_flags |= B_ERROR; ! 302: return (MBU_NEXT); ! 303: } ! 304: } ! 305: return (MBU_DODATA); ! 306: } ! 307: if (blkno < bdbtofsb(bp->b_blkno)) { ! 308: htaddr->htfc = blkno - bdbtofsb(bp->b_blkno); ! 309: htaddr->htcs1 = HT_SFORW|HT_GO; ! 310: } else { ! 311: htaddr->htfc = bdbtofsb(bp->b_blkno) - blkno; ! 312: htaddr->htcs1 = HT_SREV|HT_GO; ! 313: } ! 314: return (MBU_STARTED); ! 315: } ! 316: ! 317: htdtint(mi, mbsr) ! 318: register struct mba_device *mi; ! 319: int mbsr; ! 320: { ! 321: register struct htdevice *htaddr = (struct htdevice *)mi->mi_drv; ! 322: register struct buf *bp = mi->mi_tab.b_actf; ! 323: register struct tu_softc *sc; ! 324: int ds, er, mbs; ! 325: ! 326: sc = &tu_softc[TUUNIT(bp->b_dev)]; ! 327: ds = sc->sc_dsreg = MASKREG(htaddr->htds); ! 328: er = sc->sc_erreg = MASKREG(htaddr->hter); ! 329: sc->sc_resid = MASKREG(htaddr->htfc); ! 330: mbs = mbsr; ! 331: sc->sc_blkno++; ! 332: if((bp->b_flags & B_READ) == 0) ! 333: sc->sc_flags |= H_WRITTEN; ! 334: if ((ds&(HTDS_ERR|HTDS_MOL)) != HTDS_MOL || mbs & MBSR_EBITS) { ! 335: htaddr->htcs1 = HT_DCLR|HT_GO; ! 336: mbclrattn(mi); ! 337: if (bp->b_flags & B_RAW) { ! 338: er &= ~HTER_FCE; ! 339: mbs &= ~(MBSR_DTABT|MBSR_MBEXC); ! 340: } ! 341: if (bp->b_flags & B_READ && ds & HTDS_PES) ! 342: er &= ~(HTER_CSITM|HTER_CORCRC); ! 343: if (er&HTER_HARD || mbs&MBSR_EBITS || (ds&HTDS_MOL) == 0 || ! 344: er && ++mi->mi_tab.b_errcnt >= 7) { ! 345: if ((ds & HTDS_MOL) == 0 && sc->sc_openf > 0) ! 346: sc->sc_openf = -1; ! 347: if ((er&HTER_HARD) == HTER_FCE && ! 348: (mbs&MBSR_EBITS) == (MBSR_DTABT|MBSR_MBEXC) && ! 349: (ds&HTDS_MOL)) ! 350: goto noprint; ! 351: tprintf(sc->sc_ctty, "tu%d: hard error bn%d mbsr=%b er=%b ds=%b\n", ! 352: TUUNIT(bp->b_dev), bp->b_blkno, ! 353: mbsr, mbsr_bits, ! 354: sc->sc_erreg, hter_bits, ! 355: sc->sc_dsreg, htds_bits); ! 356: noprint: ! 357: bp->b_flags |= B_ERROR; ! 358: return (MBD_DONE); ! 359: } ! 360: if (er) ! 361: return (MBD_RETRY); ! 362: } ! 363: bp->b_resid = 0; ! 364: sc->sc_blks++; ! 365: if (mi->mi_tab.b_errcnt) ! 366: sc->sc_softerrs++; ! 367: if (bp->b_flags & B_READ) ! 368: if (ds&HTDS_TM) { /* must be a read, right? */ ! 369: bp->b_resid = bp->b_bcount; ! 370: sc->sc_nxrec = bdbtofsb(bp->b_blkno); ! 371: } else if(bp->b_bcount > MASKREG(htaddr->htfc)) ! 372: bp->b_resid = bp->b_bcount - MASKREG(htaddr->htfc); ! 373: return (MBD_DONE); ! 374: } ! 375: ! 376: htndtint(mi) ! 377: register struct mba_device *mi; ! 378: { ! 379: register struct htdevice *htaddr = (struct htdevice *)mi->mi_drv; ! 380: register struct buf *bp = mi->mi_tab.b_actf; ! 381: register struct tu_softc *sc; ! 382: int er, ds, fc; ! 383: ! 384: ds = MASKREG(htaddr->htds); ! 385: er = MASKREG(htaddr->hter); ! 386: fc = MASKREG(htaddr->htfc); ! 387: if (er) { ! 388: htaddr->htcs1 = HT_DCLR|HT_GO; ! 389: mbclrattn(mi); ! 390: } ! 391: if (bp == 0) ! 392: return (MBN_SKIP); ! 393: sc = &tu_softc[TUUNIT(bp->b_dev)]; ! 394: sc->sc_dsreg = ds; ! 395: sc->sc_erreg = er; ! 396: sc->sc_resid = fc; ! 397: if (bp == &chtbuf[HTUNIT(bp->b_dev)]) { ! 398: switch ((int)bp->b_command) { ! 399: case HT_REWOFFL: ! 400: /* offline is on purpose; don't do anything special */ ! 401: ds |= HTDS_MOL; ! 402: break; ! 403: case HT_SREV: ! 404: /* if backspace file hit bot, its not an error */ ! 405: if (er == (HTER_NEF|HTER_FCE) && ds&HTDS_BOT && ! 406: bp->b_repcnt == INF) ! 407: er &= ~HTER_NEF; ! 408: break; ! 409: } ! 410: er &= ~HTER_FCE; ! 411: if (er == 0) ! 412: ds &= ~HTDS_ERR; ! 413: } ! 414: if ((ds & (HTDS_ERR|HTDS_MOL)) != HTDS_MOL) { ! 415: if ((ds & HTDS_MOL) == 0 && sc->sc_openf > 0) ! 416: sc->sc_openf = -1; ! 417: tprintf(sc->sc_ctty, "tu%d: hard error bn%d er=%b ds=%b\n", ! 418: TUUNIT(bp->b_dev), bp->b_blkno, ! 419: sc->sc_erreg, hter_bits, sc->sc_dsreg, htds_bits); ! 420: bp->b_flags |= B_ERROR; ! 421: return (MBN_DONE); ! 422: } ! 423: if (bp == &chtbuf[HTUNIT(bp->b_dev)]) { ! 424: if (sc->sc_flags & H_REWIND) ! 425: return (ds & HTDS_BOT ? MBN_DONE : MBN_RETRY); ! 426: bp->b_resid = -sc->sc_resid; ! 427: return (MBN_DONE); ! 428: } ! 429: if (ds & HTDS_TM) ! 430: if (sc->sc_blkno > bdbtofsb(bp->b_blkno)) { ! 431: sc->sc_nxrec = bdbtofsb(bp->b_blkno) - fc; ! 432: sc->sc_blkno = sc->sc_nxrec; ! 433: } else { ! 434: sc->sc_blkno = bdbtofsb(bp->b_blkno) + fc; ! 435: sc->sc_nxrec = sc->sc_blkno - 1; ! 436: } ! 437: else ! 438: sc->sc_blkno = bdbtofsb(bp->b_blkno); ! 439: return (MBN_RETRY); ! 440: } ! 441: ! 442: /*ARGSUSED*/ ! 443: htioctl(dev, cmd, data, flag) ! 444: dev_t dev; ! 445: int cmd; ! 446: caddr_t data; ! 447: int flag; ! 448: { ! 449: register struct tu_softc *sc = &tu_softc[TUUNIT(dev)]; ! 450: register struct buf *bp = &chtbuf[HTUNIT(dev)]; ! 451: register callcount; ! 452: int fcount, error = 0; ! 453: struct mtop *mtop; ! 454: struct mtget *mtget; ! 455: /* we depend of the values and order of the MT codes here */ ! 456: static htops[] = ! 457: {HT_WEOF,HT_SFORW,HT_SREV,HT_SFORW,HT_SREV,HT_REW,HT_REWOFFL,HT_SENSE}; ! 458: ! 459: switch (cmd) { ! 460: ! 461: case MTIOCTOP: /* tape operation */ ! 462: mtop = (struct mtop *)data; ! 463: switch (mtop->mt_op) { ! 464: ! 465: case MTWEOF: ! 466: callcount = mtop->mt_count; ! 467: fcount = 1; ! 468: break; ! 469: ! 470: case MTFSF: case MTBSF: ! 471: callcount = mtop->mt_count; ! 472: fcount = INF; ! 473: break; ! 474: ! 475: case MTFSR: case MTBSR: ! 476: callcount = 1; ! 477: fcount = mtop->mt_count; ! 478: break; ! 479: ! 480: case MTREW: case MTOFFL: ! 481: callcount = 1; ! 482: fcount = 1; ! 483: break; ! 484: ! 485: default: ! 486: return (ENXIO); ! 487: } ! 488: if (callcount <= 0 || fcount <= 0) ! 489: return (EINVAL); ! 490: while (--callcount >= 0) { ! 491: htcommand(dev, htops[mtop->mt_op], fcount); ! 492: if ((mtop->mt_op == MTFSR || mtop->mt_op == MTBSR) && ! 493: bp->b_resid) ! 494: return (EIO); ! 495: if ((bp->b_flags&B_ERROR) || sc->sc_dsreg&HTDS_BOT) ! 496: break; ! 497: } ! 498: if (bp->b_flags&B_ERROR) ! 499: if ((error = bp->b_error)==0) ! 500: return (EIO); ! 501: return (error); ! 502: ! 503: case MTIOCGET: ! 504: mtget = (struct mtget *)data; ! 505: mtget->mt_dsreg = sc->sc_dsreg; ! 506: mtget->mt_erreg = sc->sc_erreg; ! 507: mtget->mt_resid = sc->sc_resid; ! 508: mtget->mt_type = MT_ISHT; ! 509: break; ! 510: ! 511: default: ! 512: return (ENXIO); ! 513: } ! 514: return (0); ! 515: } ! 516: ! 517: #define DBSIZE 20 ! 518: ! 519: htdump() ! 520: { ! 521: register struct mba_device *mi; ! 522: register struct mba_regs *mp; ! 523: register struct htdevice *htaddr; ! 524: int blk, num; ! 525: int start; ! 526: ! 527: start = 0; ! 528: num = maxfree; ! 529: #define phys(a,b) ((b)((int)(a)&0x7fffffff)) ! 530: if (htinfo[0] == 0) ! 531: return (ENXIO); ! 532: mi = phys(htinfo[0], struct mba_device *); ! 533: mp = phys(mi->mi_hd, struct mba_hd *)->mh_physmba; ! 534: mp->mba_cr = MBCR_IE; ! 535: htaddr = (struct htdevice *)&mp->mba_drv[mi->mi_drive]; ! 536: htaddr->httc = HTTC_PDP11|HTTC_1600BPI; ! 537: htaddr->htcs1 = HT_DCLR|HT_GO; ! 538: while (num > 0) { ! 539: blk = num > DBSIZE ? DBSIZE : num; ! 540: htdwrite(start, blk, htaddr, mp); ! 541: start += blk; ! 542: num -= blk; ! 543: } ! 544: hteof(htaddr); ! 545: hteof(htaddr); ! 546: htwait(htaddr); ! 547: if (htaddr->htds&HTDS_ERR) ! 548: return (EIO); ! 549: htaddr->htcs1 = HT_REW|HT_GO; ! 550: return (0); ! 551: } ! 552: ! 553: htdwrite(dbuf, num, htaddr, mp) ! 554: register dbuf, num; ! 555: register struct htdevice *htaddr; ! 556: struct mba_regs *mp; ! 557: { ! 558: register struct pte *io; ! 559: register int i; ! 560: ! 561: htwait(htaddr); ! 562: io = mp->mba_map; ! 563: for (i = 0; i < num; i++) ! 564: *(int *)io++ = dbuf++ | PG_V; ! 565: htaddr->htfc = -(num*NBPG); ! 566: mp->mba_sr = -1; ! 567: mp->mba_bcr = -(num*NBPG); ! 568: mp->mba_var = 0; ! 569: htaddr->htcs1 = HT_WCOM|HT_GO; ! 570: } ! 571: ! 572: htwait(htaddr) ! 573: struct htdevice *htaddr; ! 574: { ! 575: register s; ! 576: ! 577: do ! 578: s = htaddr->htds; ! 579: while ((s & HTDS_DRY) == 0); ! 580: } ! 581: ! 582: hteof(htaddr) ! 583: struct htdevice *htaddr; ! 584: { ! 585: ! 586: htwait(htaddr); ! 587: htaddr->htcs1 = HT_WEOF|HT_GO; ! 588: } ! 589: #endif
This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.