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