|
|
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: * @(#)tm.c 7.13 (Berkeley) 7/1/90
7: */
8:
9: #include "te.h"
10: #include "ts.h"
11: #if NTE > 0
12: /*
13: * TM11/TE10 tape driver
14: *
15: * TODO:
16: * test driver with more than one controller
17: * test reset code
18: * what happens if you offline tape during rewind?
19: * test using file system on tape
20: */
21: #include "param.h"
22: #include "systm.h"
23: #include "buf.h"
24: #include "conf.h"
25: #include "user.h"
26: #include "file.h"
27: #include "map.h"
28: #include "vm.h"
29: #include "ioctl.h"
30: #include "mtio.h"
31: #include "cmap.h"
32: #include "uio.h"
33: #include "kernel.h"
34: #include "syslog.h"
35: #include "tprintf.h"
36:
37: #include "machine/pte.h"
38: #include "../vax/cpu.h"
39: #include "ubareg.h"
40: #include "ubavar.h"
41: #include "tmreg.h"
42:
43: /*
44: * There is a ctmbuf per tape controller.
45: * It is used as the token to pass to the internal routines
46: * to execute tape ioctls, and also acts as a lock on the slaves
47: * on the controller, since there is only one per controller.
48: * In particular, when the tape is rewinding on close we release
49: * the user process but any further attempts to use the tape drive
50: * before the rewind completes will hang waiting for ctmbuf.
51: */
52: struct buf ctmbuf[NTM];
53:
54: /*
55: * Driver unibus interface routines and variables.
56: */
57: int tmprobe(), tmslave(), tmattach(), tmdgo(), tmintr();
58: struct uba_ctlr *tmminfo[NTM];
59: struct uba_device *tedinfo[NTE];
60: struct buf teutab[NTE];
61: short tetotm[NTE];
62: u_short tmstd[] = { 0772520, 0 };
63: struct uba_driver tmdriver =
64: { tmprobe, tmslave, tmattach, tmdgo, tmstd, "te", tedinfo, "tm", tmminfo, 0 };
65:
66: /* bits in minor device */
67: #define TEUNIT(dev) (minor(dev)&03)
68: #define TMUNIT(dev) (tetotm[TEUNIT(dev)])
69: #define T_NOREWIND 04
70: #define T_1600BPI 0x8
71:
72: #define INF (daddr_t)1000000L
73:
74: /*
75: * Software state per tape transport.
76: *
77: * 1. A tape drive is a unique-open device; we refuse opens when it is already.
78: * 2. We keep track of the current position on a block tape and seek
79: * before operations by forward/back spacing if necessary.
80: * 3. We remember if the last operation was a write on a tape, so if a tape
81: * is open read write and the last thing done is a write we can
82: * write a standard end of tape mark (two eofs).
83: * 4. We remember the status registers after the last command, using
84: * then internally and returning them to the SENSE ioctl.
85: * 5. We remember the last density the tape was used at. If it is
86: * not a BOT when we start using it and we are writing, we don't
87: * let the density be changed.
88: */
89: struct te_softc {
90: char sc_openf; /* lock against multiple opens */
91: char sc_lastiow; /* last op was a write */
92: daddr_t sc_blkno; /* block number, for block device tape */
93: daddr_t sc_nxrec; /* position of end of tape, if known */
94: u_short sc_erreg; /* copy of last erreg */
95: u_short sc_ioerreg; /* copy of last erreg for I/O command */
96: u_short sc_dsreg; /* copy of last dsreg */
97: short sc_resid; /* copy of last bc */
98: #ifdef unneeded
99: short sc_lastcmd; /* last command to handle direction changes */
100: #endif
101: u_short sc_dens; /* prototype command with density info */
102: short sc_tact; /* timeout is active */
103: daddr_t sc_timo; /* time until timeout expires */
104: int sc_blks; /* number of I/O operations since open */
105: int sc_softerrs; /* number of soft I/O errors since open */
106: tpr_t sc_tpr; /* tprintf handle */
107: } te_softc[NTE];
108: #ifdef unneeded
109: int tmgapsdcnt; /* DEBUG */
110: #endif
111:
112: /*
113: * States for um->um_tab.b_active, the per controller state flag.
114: * This is used to sequence control in the driver.
115: */
116: #define SSEEK 1 /* seeking */
117: #define SIO 2 /* doing seq i/o */
118: #define SCOM 3 /* sending control command */
119: #define SREW 4 /* sending a drive rewind */
120:
121: /*
122: * Determine if there is a controller for
123: * a tm at address reg. Our goal is to make the
124: * device interrupt.
125: */
126: tmprobe(reg)
127: caddr_t reg;
128: {
129: register int br, cvec; /* must be r11,r10; value-result */
130:
131: #ifdef lint
132: br = 0; cvec = br; br = cvec;
133: tmintr(0);
134: #endif
135: ((struct tmdevice *)reg)->tmcs = TM_IE;
136: /*
137: * If this is a tm11, it ought to have interrupted
138: * by now, if it isn't (ie: it is a ts04) then we just
139: * hope that it didn't interrupt, so autoconf will ignore it.
140: * Just in case, we will reference one
141: * of the more distant registers, and hope for a machine
142: * check, or similar disaster if this is a ts.
143: *
144: * Note: on an 11/780, badaddr will just generate
145: * a uba error for a ts; but our caller will notice that
146: * so we won't check for it.
147: */
148: if (badaddr((caddr_t)&((struct tmdevice *)reg)->tmrd, 2))
149: return (0);
150: return (sizeof (struct tmdevice));
151: }
152:
153: /*
154: * Due to a design flaw, we cannot ascertain if the tape
155: * exists or not unless it is on line - ie: unless a tape is
156: * mounted. This is too servere a restriction to bear,
157: * so all units are assumed to exist.
158: */
159: /*ARGSUSED*/
160: tmslave(ui, reg)
161: struct uba_device *ui;
162: caddr_t reg;
163: {
164:
165: return (1);
166: }
167:
168: /*
169: * Record attachment of the unit to the controller.
170: */
171: /*ARGSUSED*/
172: tmattach(ui)
173: struct uba_device *ui;
174: {
175: /*
176: * Tetotm is used in TMUNIT to index the ctmbuf
177: * array given a te unit number.
178: */
179: tetotm[ui->ui_unit] = ui->ui_mi->um_ctlr;
180: }
181:
182: int tmtimer();
183: /*
184: * Open the device. Tapes are unique open
185: * devices, so we refuse if it is already open.
186: * We also check that a tape is available, and
187: * don't block waiting here; if you want to wait
188: * for a tape you should timeout in user code.
189: */
190:
191: #ifdef AVIV
192: int tmdens[4] = { 0x6000, 0x0000, 0x2000, 0 };
193: #endif AVIV
194: int tmdiag;
195:
196: tmopen(dev, flag)
197: dev_t dev;
198: int flag;
199: {
200: register int teunit;
201: register struct uba_device *ui;
202: register struct te_softc *sc;
203: int olddens, dens, error;
204: int s;
205:
206: teunit = TEUNIT(dev);
207: if (teunit>=NTE || (ui = tedinfo[teunit]) == 0 || ui->ui_alive == 0)
208: return (ENXIO);
209: if ((sc = &te_softc[teunit])->sc_openf)
210: return (EBUSY);
211: sc->sc_openf = 1;
212: olddens = sc->sc_dens;
213: dens = TM_IE | TM_GO | (ui->ui_slave << 8);
214: #ifndef AVIV
215: if ((minor(dev) & T_1600BPI) == 0)
216: dens |= TM_D800;
217: #else AVIV
218: dens |= tmdens[(minor(dev)>>3)&03];
219: #endif AVIV
220: sc->sc_dens = dens;
221: get:
222: tmcommand(dev, TM_SENSE, 1);
223: if (sc->sc_erreg&TMER_SDWN) {
224: if (error = tsleep((caddr_t)&lbolt, (PZERO+1) | PCATCH,
225: devopn, 0))
226: return (error);
227: goto get;
228: }
229: sc->sc_dens = olddens;
230: if ((sc->sc_erreg&(TMER_SELR|TMER_TUR)) != (TMER_SELR|TMER_TUR)) {
231: uprintf("te%d: not online\n", teunit);
232: sc->sc_openf = 0;
233: return (EIO);
234: }
235: if ((flag&FWRITE) && (sc->sc_erreg&TMER_WRL)) {
236: uprintf("te%d: no write ring\n", teunit);
237: sc->sc_openf = 0;
238: return (EIO);
239: }
240: if ((sc->sc_erreg&TMER_BOT) == 0 && (flag&FWRITE) &&
241: dens != sc->sc_dens) {
242: uprintf("te%d: can't change density in mid-tape\n", teunit);
243: sc->sc_openf = 0;
244: return (EIO);
245: }
246: sc->sc_blkno = (daddr_t)0;
247: sc->sc_nxrec = INF;
248: sc->sc_lastiow = 0;
249: sc->sc_dens = dens;
250: sc->sc_blks = 0;
251: sc->sc_softerrs = 0;
252: sc->sc_tpr = tprintf_open();
253: s = splclock();
254: if (sc->sc_tact == 0) {
255: sc->sc_timo = INF;
256: sc->sc_tact = 1;
257: timeout(tmtimer, (caddr_t)dev, 5*hz);
258: }
259: splx(s);
260: return (0);
261: }
262:
263: /*
264: * Close tape device.
265: *
266: * If tape was open for writing or last operation was
267: * a write, then write two EOF's and backspace over the last one.
268: * Unless this is a non-rewinding special file, rewind the tape.
269: * Make the tape available to others.
270: */
271: tmclose(dev, flag)
272: register dev_t dev;
273: register flag;
274: {
275: register struct te_softc *sc = &te_softc[TEUNIT(dev)];
276:
277: if (flag == FWRITE || (flag&FWRITE) && sc->sc_lastiow) {
278: tmcommand(dev, TM_WEOF, 1);
279: tmcommand(dev, TM_WEOF, 1);
280: tmcommand(dev, TM_SREV, 1);
281: }
282: if ((minor(dev)&T_NOREWIND) == 0)
283: /*
284: * 0 count means don't hang waiting for rewind complete
285: * rather ctmbuf stays busy until the operation completes
286: * preventing further opens from completing by
287: * preventing a TM_SENSE from completing.
288: */
289: tmcommand(dev, TM_REW, 0);
290: if (sc->sc_blks > 100 && sc->sc_softerrs > sc->sc_blks / 100)
291: log(LOG_INFO, "te%d: %d soft errors in %d blocks\n",
292: TEUNIT(dev), sc->sc_softerrs, sc->sc_blks);
293: tprintf_close(sc->sc_tpr);
294: sc->sc_openf = 0;
295: return (0);
296: }
297:
298: /*
299: * Execute a command on the tape drive
300: * a specified number of times.
301: */
302: tmcommand(dev, com, count)
303: dev_t dev;
304: int com, count;
305: {
306: register struct buf *bp;
307: register int s;
308:
309: bp = &ctmbuf[TMUNIT(dev)];
310: s = spl5();
311: while (bp->b_flags&B_BUSY) {
312: /*
313: * This special check is because B_BUSY never
314: * gets cleared in the non-waiting rewind case.
315: */
316: if (bp->b_repcnt == 0 && (bp->b_flags&B_DONE))
317: break;
318: bp->b_flags |= B_WANTED;
319: sleep((caddr_t)bp, PRIBIO);
320: }
321: bp->b_flags = B_BUSY|B_READ;
322: splx(s);
323: bp->b_dev = dev;
324: bp->b_repcnt = -count;
325: bp->b_command = com;
326: bp->b_blkno = 0;
327: tmstrategy(bp);
328: /*
329: * In case of rewind from close, don't wait.
330: * This is the only case where count can be 0.
331: */
332: if (count == 0)
333: return;
334: iowait(bp);
335: if (bp->b_flags&B_WANTED)
336: wakeup((caddr_t)bp);
337: bp->b_flags &= B_ERROR;
338: }
339:
340: /*
341: * Queue a tape operation.
342: */
343: tmstrategy(bp)
344: register struct buf *bp;
345: {
346: int teunit = TEUNIT(bp->b_dev);
347: register struct uba_ctlr *um;
348: register struct buf *dp;
349: int s;
350:
351: /*
352: * Put transfer at end of unit queue
353: */
354: dp = &teutab[teunit];
355: bp->av_forw = NULL;
356: s = spl5();
357: um = tedinfo[teunit]->ui_mi;
358: if (dp->b_actf == NULL) {
359: dp->b_actf = bp;
360: /*
361: * Transport not already active...
362: * put at end of controller queue.
363: */
364: dp->b_forw = NULL;
365: if (um->um_tab.b_actf == NULL)
366: um->um_tab.b_actf = dp;
367: else
368: um->um_tab.b_actl->b_forw = dp;
369: um->um_tab.b_actl = dp;
370: } else
371: dp->b_actl->av_forw = bp;
372: dp->b_actl = bp;
373: /*
374: * If the controller is not busy, get
375: * it going.
376: */
377: if (um->um_tab.b_active == 0)
378: tmstart(um);
379: splx(s);
380: }
381:
382: /*
383: * Start activity on a tm controller.
384: */
385: tmstart(um)
386: register struct uba_ctlr *um;
387: {
388: register struct buf *bp, *dp;
389: register struct tmdevice *addr = (struct tmdevice *)um->um_addr;
390: register struct te_softc *sc;
391: register struct uba_device *ui;
392: int teunit, cmd;
393: daddr_t blkno;
394:
395: /*
396: * Look for an idle transport on the controller.
397: */
398: loop:
399: if ((dp = um->um_tab.b_actf) == NULL)
400: return;
401: if ((bp = dp->b_actf) == NULL) {
402: um->um_tab.b_actf = dp->b_forw;
403: goto loop;
404: }
405: teunit = TEUNIT(bp->b_dev);
406: ui = tedinfo[teunit];
407: /*
408: * Record pre-transfer status (e.g. for TM_SENSE)
409: */
410: sc = &te_softc[teunit];
411: addr = (struct tmdevice *)um->um_addr;
412: addr->tmcs = (ui->ui_slave << 8);
413: sc->sc_dsreg = addr->tmcs;
414: sc->sc_erreg = addr->tmer;
415: sc->sc_resid = addr->tmbc;
416: /*
417: * Default is that last command was NOT a write command;
418: * if we do a write command we will notice this in tmintr().
419: */
420: sc->sc_lastiow = 0;
421: if (sc->sc_openf < 0 || (addr->tmcs&TM_CUR) == 0) {
422: /*
423: * Have had a hard error on a non-raw tape
424: * or the tape unit is now unavailable
425: * (e.g. taken off line).
426: */
427: bp->b_flags |= B_ERROR;
428: goto next;
429: }
430: if (bp == &ctmbuf[TMUNIT(bp->b_dev)]) {
431: /*
432: * Execute control operation with the specified count.
433: */
434: if (bp->b_command == TM_SENSE)
435: goto next;
436: /*
437: * Set next state; give 5 minutes to complete
438: * rewind, or 10 seconds per iteration (minimum 60
439: * seconds and max 5 minutes) to complete other ops.
440: */
441: if (bp->b_command == TM_REW) {
442: um->um_tab.b_active = SREW;
443: sc->sc_timo = 5 * 60;
444: } else {
445: um->um_tab.b_active = SCOM;
446: sc->sc_timo =
447: imin(imax(10*(int)-bp->b_repcnt,60),5*60);
448: }
449: if (bp->b_command == TM_SFORW || bp->b_command == TM_SREV)
450: addr->tmbc = bp->b_repcnt;
451: goto dobpcmd;
452: }
453: /*
454: * For raw I/O, fudge the current block number
455: * so we don't seek except on a retry.
456: * For raw I/O, fudge the current block number
457: * so we don't seek except on a retry.
458: */
459: if (bp->b_flags & B_RAW) {
460: if (um->um_tab.b_errcnt == 0) {
461: sc->sc_blkno = bdbtofsb(bp->b_blkno);
462: sc->sc_nxrec = sc->sc_blkno + 1;
463: }
464: } else {
465: /*
466: * Handle boundary cases for operation
467: * on non-raw tapes.
468: */
469: if (bdbtofsb(bp->b_blkno) > sc->sc_nxrec) {
470: /*
471: * Can't read past known end-of-file.
472: */
473: bp->b_flags |= B_ERROR;
474: bp->b_error = ENXIO;
475: goto next;
476: }
477: if (bdbtofsb(bp->b_blkno) == sc->sc_nxrec &&
478: bp->b_flags&B_READ) {
479: /*
480: * Reading at end of file returns 0 bytes.
481: */
482: bp->b_resid = bp->b_bcount;
483: clrbuf(bp);
484: goto next;
485: }
486: if ((bp->b_flags&B_READ) == 0)
487: /*
488: * Writing sets EOF
489: */
490: sc->sc_nxrec = bdbtofsb(bp->b_blkno) + 1;
491: }
492: /*
493: * If the data transfer command is in the correct place,
494: * set up all the registers except the csr, and give
495: * control over to the UNIBUS adapter routines, to
496: * wait for resources to start the i/o.
497: */
498: if ((blkno = sc->sc_blkno) == bdbtofsb(bp->b_blkno)) {
499: addr->tmbc = -bp->b_bcount;
500: if ((bp->b_flags&B_READ) == 0) {
501: if (um->um_tab.b_errcnt &&
502: (sc->sc_ioerreg&(TMER_HARD|TMER_SOFT)) != TMER_BGL)
503: cmd = TM_WIRG;
504: else
505: cmd = TM_WCOM;
506: } else
507: cmd = TM_RCOM;
508: um->um_tab.b_active = SIO;
509: um->um_cmd = sc->sc_dens|cmd;
510: #ifdef notdef
511: if (tmreverseop(sc->sc_lastcmd))
512: while (addr->tmer & TMER_SDWN)
513: DELAY(10),tmgapsdcnt++;
514: sc->sc_lastcmd = TM_RCOM; /* will serve */
515: #endif
516: sc->sc_timo = 60; /* premature, but should serve */
517: (void) ubago(ui);
518: return;
519: }
520: /*
521: * Tape positioned incorrectly;
522: * set to seek forwards or backwards to the correct spot.
523: * This happens for raw tapes only on error retries.
524: */
525: um->um_tab.b_active = SSEEK;
526: if (blkno < bdbtofsb(bp->b_blkno)) {
527: bp->b_command = TM_SFORW;
528: addr->tmbc = blkno - bdbtofsb(bp->b_blkno);
529: } else {
530: bp->b_command = TM_SREV;
531: addr->tmbc = bdbtofsb(bp->b_blkno) - blkno;
532: }
533: sc->sc_timo = imin(imax(10 * -addr->tmbc, 60), 5 * 60);
534: dobpcmd:
535: #ifdef notdef
536: /*
537: * It is strictly necessary to wait for the tape
538: * to stop before changing directions, but the TC11
539: * handles this for us.
540: */
541: if (tmreverseop(sc->sc_lastcmd) != tmreverseop(bp->b_command))
542: while (addr->tmer & TM_SDWN)
543: DELAY(10),tmgapsdcnt++;
544: sc->sc_lastcmd = bp->b_command;
545: #endif
546: /*
547: * Do the command in bp.
548: */
549: addr->tmcs = (sc->sc_dens | bp->b_command);
550: return;
551:
552: next:
553: /*
554: * Done with this operation due to error or
555: * the fact that it doesn't do anything.
556: * Release UBA resources (if any), dequeue
557: * the transfer and continue processing this slave.
558: */
559: if (um->um_ubinfo)
560: ubadone(um);
561: um->um_tab.b_errcnt = 0;
562: dp->b_actf = bp->av_forw;
563: iodone(bp);
564: goto loop;
565: }
566:
567: /*
568: * The UNIBUS resources we needed have been
569: * allocated to us; start the device.
570: */
571: tmdgo(um)
572: register struct uba_ctlr *um;
573: {
574: register struct tmdevice *addr = (struct tmdevice *)um->um_addr;
575:
576: addr->tmba = um->um_ubinfo;
577: addr->tmcs = um->um_cmd | ((um->um_ubinfo >> 12) & 0x30);
578: }
579:
580: /*
581: * Tm interrupt routine.
582: */
583: /*ARGSUSED*/
584: tmintr(tm11)
585: int tm11;
586: {
587: struct buf *dp;
588: register struct buf *bp;
589: register struct uba_ctlr *um = tmminfo[tm11];
590: register struct tmdevice *addr;
591: register struct te_softc *sc;
592: int teunit;
593: register state;
594:
595: if ((dp = um->um_tab.b_actf) == NULL)
596: return;
597: bp = dp->b_actf;
598: teunit = TEUNIT(bp->b_dev);
599: addr = (struct tmdevice *)tedinfo[teunit]->ui_addr;
600: sc = &te_softc[teunit];
601: /*
602: * If last command was a rewind, and tape is still
603: * rewinding, wait for the rewind complete interrupt.
604: */
605: if (um->um_tab.b_active == SREW) {
606: um->um_tab.b_active = SCOM;
607: if (addr->tmer&TMER_RWS) {
608: sc->sc_timo = 5*60; /* 5 minutes */
609: return;
610: }
611: }
612: /*
613: * An operation completed... record status
614: */
615: sc->sc_timo = INF;
616: if (um->um_tab.b_active == SIO)
617: if (um->um_tab.b_active == SIO)
618: sc->sc_ioerreg = addr->tmer;
619: sc->sc_dsreg = addr->tmcs;
620: sc->sc_erreg = addr->tmer;
621: sc->sc_resid = addr->tmbc;
622: if ((bp->b_flags & B_READ) == 0)
623: sc->sc_lastiow = 1;
624: state = um->um_tab.b_active;
625: um->um_tab.b_active = 0;
626: /*
627: * Check for errors.
628: */
629: if (addr->tmcs&TM_ERR) {
630: while (addr->tmer & TMER_SDWN)
631: DELAY(10); /* await settle down */
632: /*
633: * If we hit the end of the tape file, update our position.
634: */
635: if (addr->tmer&TMER_EOF) {
636: tmseteof(bp); /* set blkno and nxrec */
637: state = SCOM; /* force completion */
638: /*
639: * Stuff bc so it will be unstuffed correctly
640: * later to get resid.
641: */
642: addr->tmbc = -bp->b_bcount;
643: goto opdone;
644: }
645: /*
646: * If we were reading raw tape and the only error was that the
647: * record was too long, then we don't consider this an error.
648: */
649: if ((bp->b_flags & (B_READ|B_RAW)) == (B_READ|B_RAW) &&
650: (addr->tmer&(TMER_HARD|TMER_SOFT)) == TMER_RLE)
651: goto ignoreerr;
652: /*
653: * If error is not hard, and this was an i/o operation
654: * retry up to 8 times.
655: */
656: if ((addr->tmer&TMER_HARD)==0 && state==SIO) {
657: if (um->um_tab.b_errcnt++ < 8) {
658: if (tmdiag)
659: log(LOG_DEBUG,
660: "te%d: soft error bn%d er=%b\n",
661: minor(bp->b_dev)&03,
662: bp->b_blkno, sc->sc_erreg,
663: TMER_BITS);
664: sc->sc_blkno++; /* force backspace */
665: ubadone(um);
666: goto opcont;
667: }
668: } else
669: /*
670: * Hard or non-i/o errors on non-raw tape
671: * cause it to close.
672: */
673: if ((bp->b_flags&B_RAW) == 0 && sc->sc_openf > 0)
674: sc->sc_openf = -1;
675: /*
676: * Couldn't recover error
677: */
678: tprintf(sc->sc_tpr,
679: "te%d: hard error bn%d er=%b\n", minor(bp->b_dev)&03,
680: bp->b_blkno, sc->sc_erreg, TMER_BITS);
681: #ifdef AVIV
682: if (tmdiag) {
683: addr->tmmr = DAB;
684: printf("reject code 0%o", addr->tmmr & DAB_MASK);
685: addr->tmmr = DTS;
686: if (addr->tmmr & DTS_MASK)
687: printf(", dead track 0%o", addr->tmmr & DTS_MASK);
688: addr->tmmr = RWERR;
689: printf(", read/write errors %b\n",
690: addr->tmmr & RWERR_MASK,
691: RWERR_BITS);
692: addr->tmmr = DRSENSE;
693: printf("drive sense %b, ", addr->tmmr & DRSENSE_MASK,
694: DRSENSE_BITS);
695: printf("fsr %b\n", addr->tmfsr, FSR_BITS);
696: }
697: #endif AVIV
698: bp->b_flags |= B_ERROR;
699: goto opdone;
700: }
701: /*
702: * Advance tape control FSM.
703: */
704: ignoreerr:
705: switch (state) {
706:
707: case SIO:
708: /*
709: * Read/write increments tape block number
710: */
711: sc->sc_blkno++;
712: sc->sc_blks++;
713: if (um->um_tab.b_errcnt)
714: sc->sc_softerrs++;
715: goto opdone;
716:
717: case SCOM:
718: /*
719: * For forward/backward space record update current position.
720: */
721: if (bp == &ctmbuf[TMUNIT(bp->b_dev)])
722: switch ((int)bp->b_command) {
723:
724: case TM_SFORW:
725: sc->sc_blkno -= bp->b_repcnt;
726: break;
727:
728: case TM_SREV:
729: sc->sc_blkno += bp->b_repcnt;
730: break;
731: }
732: goto opdone;
733:
734: case SSEEK:
735: sc->sc_blkno = bdbtofsb(bp->b_blkno);
736: goto opcont;
737:
738: default:
739: panic("tmintr");
740: }
741: opdone:
742: /*
743: * Reset error count and remove
744: * from device queue.
745: */
746: um->um_tab.b_errcnt = 0;
747: dp->b_actf = bp->av_forw;
748: /*
749: * Check resid; watch out for resid >32767 (tmbc not negative).
750: */
751: bp->b_resid = ((int) -addr->tmbc) & 0xffff;
752: ubadone(um);
753: iodone(bp);
754: /*
755: * Circulate slave to end of controller
756: * queue to give other slaves a chance.
757: */
758: um->um_tab.b_actf = dp->b_forw;
759: if (dp->b_actf) {
760: dp->b_forw = NULL;
761: if (um->um_tab.b_actf == NULL)
762: um->um_tab.b_actf = dp;
763: else
764: um->um_tab.b_actl->b_forw = dp;
765: um->um_tab.b_actl = dp;
766: }
767: if (um->um_tab.b_actf == 0)
768: return;
769: opcont:
770: tmstart(um);
771: }
772:
773: tmtimer(dev)
774: int dev;
775: {
776: register struct te_softc *sc = &te_softc[TEUNIT(dev)];
777: register short x;
778:
779: if (sc->sc_timo != INF && (sc->sc_timo -= 5) < 0) {
780: printf("te%d: lost interrupt\n", TEUNIT(dev));
781: sc->sc_timo = INF;
782: x = spl5();
783: tmintr(TMUNIT(dev));
784: (void) splx(x);
785: }
786: timeout(tmtimer, (caddr_t)dev, 5*hz);
787: }
788:
789: tmseteof(bp)
790: register struct buf *bp;
791: {
792: register int teunit = TEUNIT(bp->b_dev);
793: register struct tmdevice *addr =
794: (struct tmdevice *)tedinfo[teunit]->ui_addr;
795: register struct te_softc *sc = &te_softc[teunit];
796:
797: if (bp == &ctmbuf[TMUNIT(bp->b_dev)]) {
798: if (sc->sc_blkno > bdbtofsb(bp->b_blkno)) {
799: /* reversing */
800: sc->sc_nxrec = bdbtofsb(bp->b_blkno) - addr->tmbc;
801: sc->sc_blkno = sc->sc_nxrec;
802: } else {
803: /* spacing forward */
804: sc->sc_blkno = bdbtofsb(bp->b_blkno) + addr->tmbc;
805: sc->sc_nxrec = sc->sc_blkno - 1;
806: }
807: return;
808: }
809: /* eof on read */
810: sc->sc_nxrec = bdbtofsb(bp->b_blkno);
811: }
812:
813: tmreset(uban)
814: int uban;
815: {
816: register struct uba_ctlr *um;
817: register tm11, teunit;
818: register struct uba_device *ui;
819: register struct buf *dp;
820:
821: for (tm11 = 0; tm11 < NTM; tm11++) {
822: if ((um = tmminfo[tm11]) == 0 || um->um_alive == 0 ||
823: um->um_ubanum != uban)
824: continue;
825: printf(" tm%d", tm11);
826: um->um_tab.b_active = 0;
827: um->um_tab.b_actf = um->um_tab.b_actl = 0;
828: if (um->um_ubinfo) {
829: printf("<%d>", (um->um_ubinfo>>28)&0xf);
830: um->um_ubinfo = 0;
831: }
832: ((struct tmdevice *)(um->um_addr))->tmcs = TM_DCLR;
833: for (teunit = 0; teunit < NTE; teunit++) {
834: if ((ui = tedinfo[teunit]) == 0 || ui->ui_mi != um ||
835: ui->ui_alive == 0)
836: continue;
837: dp = &teutab[teunit];
838: dp->b_active = 0;
839: dp->b_forw = 0;
840: if (um->um_tab.b_actf == NULL)
841: um->um_tab.b_actf = dp;
842: else
843: um->um_tab.b_actl->b_forw = dp;
844: um->um_tab.b_actl = dp;
845: if (te_softc[teunit].sc_openf > 0)
846: te_softc[teunit].sc_openf = -1;
847: }
848: tmstart(um);
849: }
850: }
851:
852: /*ARGSUSED*/
853: tmioctl(dev, cmd, data, flag)
854: caddr_t data;
855: dev_t dev;
856: {
857: int teunit = TEUNIT(dev);
858: register struct te_softc *sc = &te_softc[teunit];
859: register struct buf *bp = &ctmbuf[TMUNIT(dev)];
860: register callcount;
861: int fcount, error = 0;
862: struct mtop *mtop;
863: struct mtget *mtget;
864: /* we depend of the values and order of the MT codes here */
865: static tmops[] =
866: {TM_WEOF,TM_SFORW,TM_SREV,TM_SFORW,TM_SREV,TM_REW,TM_OFFL,TM_SENSE};
867:
868: switch (cmd) {
869:
870: case MTIOCTOP: /* tape operation */
871: mtop = (struct mtop *)data;
872: switch (mtop->mt_op) {
873:
874: case MTWEOF:
875: callcount = mtop->mt_count;
876: fcount = 1;
877: break;
878:
879: case MTFSF: case MTBSF:
880: callcount = mtop->mt_count;
881: fcount = INF;
882: break;
883:
884: case MTFSR: case MTBSR:
885: callcount = 1;
886: fcount = mtop->mt_count;
887: break;
888:
889: case MTREW: case MTOFFL: case MTNOP:
890: callcount = 1;
891: fcount = 1;
892: break;
893:
894: default:
895: return (ENXIO);
896: }
897: if (callcount <= 0 || fcount <= 0)
898: return (EINVAL);
899: while (--callcount >= 0) {
900: tmcommand(dev, tmops[mtop->mt_op], fcount);
901: if ((mtop->mt_op == MTFSR || mtop->mt_op == MTBSR) &&
902: bp->b_resid)
903: return (EIO);
904: if ((bp->b_flags&B_ERROR) || sc->sc_erreg&TMER_BOT)
905: break;
906: }
907: if (bp->b_flags&B_ERROR)
908: if ((error = bp->b_error)==0)
909: return (EIO);
910: return (error);
911:
912: case MTIOCGET:
913: mtget = (struct mtget *)data;
914: mtget->mt_dsreg = sc->sc_dsreg;
915: mtget->mt_erreg = sc->sc_erreg;
916: mtget->mt_resid = sc->sc_resid;
917: mtget->mt_type = MT_ISTM;
918: break;
919:
920: default:
921: return (ENXIO);
922: }
923: return (0);
924: }
925:
926: #define DBSIZE 20
927:
928: tmdump()
929: {
930: register struct uba_device *ui;
931: register struct uba_regs *up;
932: register struct tmdevice *addr;
933: int blk, num;
934: int start;
935:
936: start = 0;
937: num = maxfree;
938: #define phys(a,b) ((b)((int)(a)&0x7fffffff))
939: if (tedinfo[0] == 0)
940: return (ENXIO);
941: ui = phys(tedinfo[0], struct uba_device *);
942: up = phys(ui->ui_hd, struct uba_hd *)->uh_physuba;
943: ubainit(up);
944: DELAY(1000000);
945: addr = (struct tmdevice *)ui->ui_physaddr;
946: tmwait(addr);
947: addr->tmcs = TM_DCLR | TM_GO;
948: while (num > 0) {
949: blk = num > DBSIZE ? DBSIZE : num;
950: tmdwrite(start, blk, addr, up);
951: start += blk;
952: num -= blk;
953: }
954: tmeof(addr);
955: tmeof(addr);
956: tmwait(addr);
957: if (addr->tmcs&TM_ERR)
958: return (EIO);
959: addr->tmcs = TM_REW | TM_GO;
960: tmwait(addr);
961: return (0);
962: }
963:
964: tmdwrite(dbuf, num, addr, up)
965: register dbuf, num;
966: register struct tmdevice *addr;
967: struct uba_regs *up;
968: {
969: register struct pte *io;
970: register int npf;
971:
972: tmwait(addr);
973: io = up->uba_map;
974: npf = num+1;
975: while (--npf != 0)
976: *(int *)io++ = (dbuf++ | (1<<UBAMR_DPSHIFT) | UBAMR_MRV);
977: *(int *)io = 0;
978: addr->tmbc = -(num*NBPG);
979: addr->tmba = 0;
980: addr->tmcs = TM_WCOM | TM_GO;
981: }
982:
983: tmwait(addr)
984: register struct tmdevice *addr;
985: {
986: register s;
987:
988: do
989: s = addr->tmcs;
990: while ((s & TM_CUR) == 0);
991: }
992:
993: tmeof(addr)
994: struct tmdevice *addr;
995: {
996:
997: tmwait(addr);
998: addr->tmcs = TM_WEOF | TM_GO;
999: }
1000: #endif
This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.