Annotation of 43BSDReno/sys/hpdev/ct.c, revision 1.1

1.1     ! root        1: /*
        !             2:  * Copyright (c) 1982, 1990 The Regents of the University of California.
        !             3:  * All rights reserved.
        !             4:  *
        !             5:  * Redistribution is only permitted until one year after the first shipment
        !             6:  * of 4.4BSD by the Regents.  Otherwise, redistribution and use in source and
        !             7:  * binary forms are permitted provided that: (1) source distributions retain
        !             8:  * this entire copyright notice and comment, and (2) distributions including
        !             9:  * binaries display the following acknowledgement:  This product includes
        !            10:  * software developed by the University of California, Berkeley and its
        !            11:  * contributors'' in the documentation or other materials provided with the
        !            12:  * distribution and in all advertising materials mentioning features or use
        !            13:  * of this software.  Neither the name of the University nor the names of
        !            14:  * its contributors may be used to endorse or promote products derived from
        !            15:  * this software without specific prior written permission.
        !            16:  * THIS SOFTWARE IS PROVIDED AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
        !            17:  * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
        !            18:  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
        !            19:  *
        !            20:  *     @(#)ct.c        7.1 (Berkeley) 5/8/90
        !            21:  */
        !            22: 
        !            23: #include "ct.h"
        !            24: #if NCT > 0
        !            25: /*
        !            26:  * CS80 cartridge tape driver (9144, 88140, 9145)
        !            27:  *
        !            28:  * Reminder:
        !            29:  *     C_CC bit (character count option) when used in the CS/80 command
        !            30:  *     'set options' will cause the tape not to stream.
        !            31:  *
        !            32:  * TODO:
        !            33:  *     make filesystem compatible
        !            34:  *     make block mode work according to mtio(4) spec. (if possible)
        !            35:  *     merge with cs80 disk driver
        !            36:  *     finish support of 9145
        !            37:  */
        !            38: 
        !            39: #include "param.h"
        !            40: #include "buf.h"
        !            41: #include "ioctl.h"
        !            42: #include "mtio.h"
        !            43: #include "errno.h"
        !            44: #include "ctreg.h"
        !            45: #include "device.h"
        !            46: #include "user.h"
        !            47: #include "tty.h"
        !            48: #include "proc.h"
        !            49: 
        !            50: /* number of eof marks to remember */
        !            51: #define EOFS   128
        !            52: 
        !            53: int    ctinit(), ctstart(), ctgo(), ctintr();
        !            54: struct driver ctdriver = {
        !            55:        ctinit, "ct", ctstart, ctgo, ctintr,
        !            56: };
        !            57: 
        !            58: struct ct_softc {
        !            59:        struct  hp_device *sc_hd;
        !            60:        struct  ct_iocmd sc_ioc;
        !            61:        struct  ct_rscmd sc_rsc;
        !            62:        struct  ct_stat sc_stat;
        !            63:        struct  ct_ssmcmd sc_ssmc;
        !            64:        struct  ct_srcmd sc_src;
        !            65:        struct  ct_soptcmd sc_soptc;
        !            66:        struct  ct_ulcmd sc_ul;
        !            67:        struct  ct_wfmcmd sc_wfm;
        !            68:        struct  ct_clearcmd sc_clear;
        !            69:        struct  buf *sc_bp;
        !            70:        int     sc_blkno;
        !            71:        int     sc_cmd;
        !            72:        int     sc_resid;
        !            73:        char    *sc_addr;
        !            74:        int     sc_flags;
        !            75:        short   sc_type;
        !            76:        short   sc_punit;
        !            77:        caddr_t sc_ctty;
        !            78:        struct  devqueue sc_dq;
        !            79:        int     sc_eofp;
        !            80:        int     sc_eofs[EOFS];
        !            81: } ct_softc[NCT];
        !            82: 
        !            83: /* flags */
        !            84: #define        CTF_OPEN        0x01
        !            85: #define        CTF_ALIVE       0x02
        !            86: #define        CTF_WRT         0x04
        !            87: #define        CTF_CMD         0x08
        !            88: #define        CTF_IO          0x10
        !            89: #define        CTF_BEOF        0x20
        !            90: #define        CTF_AEOF        0x40
        !            91: #define        CTF_EOT         0x80
        !            92: #define        CTF_STATWAIT    0x100
        !            93: #define CTF_CANSTREAM  0x200
        !            94: #define        CTF_WRTTN       0x400
        !            95: 
        !            96: struct ctinfo {
        !            97:        short   hwid;
        !            98:        short   punit;
        !            99:        char    *desc;
        !           100: } ctinfo[] = {
        !           101:        CT7946ID,       1,      "7946A",
        !           102:        CT7912PID,      1,      "7912P",
        !           103:        CT7914PID,      1,      "7914P",
        !           104:        CT9144ID,       0,      "9144",
        !           105:        CT9145ID,       0,      "9145",
        !           106: };
        !           107: int    nctinfo = sizeof(ctinfo) / sizeof(ctinfo[0]);
        !           108: 
        !           109: struct buf cttab[NCT];
        !           110: struct buf ctbuf[NCT];
        !           111: 
        !           112: #define        CT_NOREW        4
        !           113: #define        CT_STREAM       8
        !           114: #define        UNIT(x)         (minor(x) & 3)
        !           115: #define        ctpunit(x)      ((x) & 7)
        !           116: 
        !           117: #ifdef DEBUG
        !           118: int ctdebug = 0;
        !           119: #define CDB_FILES      0x01
        !           120: #define CT_BSF         0x02
        !           121: #endif
        !           122: 
        !           123: ctinit(hd)
        !           124:        register struct hp_device *hd;
        !           125: {
        !           126:        register struct ct_softc *sc = &ct_softc[hd->hp_unit];
        !           127: 
        !           128:        sc->sc_hd = hd;
        !           129:        sc->sc_punit = ctpunit(hd->hp_flags);
        !           130:        if (ctident(sc, hd) < 0)
        !           131:                return(0);
        !           132:        ctreset(sc, hd);
        !           133:        sc->sc_dq.dq_ctlr = hd->hp_ctlr;
        !           134:        sc->sc_dq.dq_unit = hd->hp_unit;
        !           135:        sc->sc_dq.dq_slave = hd->hp_slave;
        !           136:        sc->sc_dq.dq_driver = &ctdriver;
        !           137:        sc->sc_flags |= CTF_ALIVE;
        !           138:        return(1);
        !           139: }
        !           140: 
        !           141: ctident(sc, hd)
        !           142:        register struct ct_softc *sc;
        !           143:        register struct hp_device *hd;
        !           144: {
        !           145:        struct ct_describe desc;
        !           146:        u_char stat, cmd[3];
        !           147:        char name[7];
        !           148:        int id, i;
        !           149: 
        !           150:        /*
        !           151:         * Read device id and verify that:
        !           152:         * 1. It is a CS80 device
        !           153:         * 2. It is one of our recognized tape devices
        !           154:         * 3. It has the proper physical unit number
        !           155:         */
        !           156:        id = hpibid(hd->hp_ctlr, hd->hp_slave);
        !           157:        if ((id & 0x200) == 0)
        !           158:                return(-1);
        !           159:        for (i = 0; i < nctinfo; i++)
        !           160:                if (id == ctinfo[i].hwid)
        !           161:                        break;
        !           162:        if (i == nctinfo || sc->sc_punit != ctinfo[i].punit)
        !           163:                return(-1);
        !           164:        id = i;
        !           165: 
        !           166:        /*
        !           167:         * Collect device description.
        !           168:         * Right now we only need this to differentiate 7945 from 7946.
        !           169:         * Note that we always issue the describe command to unit 0.
        !           170:         */
        !           171:        cmd[0] = C_SUNIT(0);
        !           172:        cmd[1] = C_SVOL(0);
        !           173:        cmd[2] = C_DESC;
        !           174:        hpibsend(hd->hp_ctlr, hd->hp_slave, C_CMD, cmd, sizeof(cmd));
        !           175:        hpibrecv(hd->hp_ctlr, hd->hp_slave, C_EXEC, &desc, 37);
        !           176:        hpibrecv(hd->hp_ctlr, hd->hp_slave, C_QSTAT, &stat, sizeof(stat));
        !           177:        bzero(name, sizeof(name));
        !           178:        if (!stat) {
        !           179:                register int n = desc.d_name;
        !           180:                for (i = 5; i >= 0; i--) {
        !           181:                        name[i] = (n & 0xf) + '0';
        !           182:                        n >>= 4;
        !           183:                }
        !           184:        }
        !           185:        switch (ctinfo[id].hwid) {
        !           186:        case CT7946ID:
        !           187:                if (bcmp(name, "079450", 6) == 0)
        !           188:                        return(-1);             /* not really a 7946 */
        !           189:                /* fall into... */
        !           190:        case CT9144ID:
        !           191:        case CT9145ID:
        !           192:                sc->sc_type = CT9144;
        !           193:                sc->sc_flags |= CTF_CANSTREAM;
        !           194:                break;
        !           195: 
        !           196:        case CT7912PID:
        !           197:        case CT7914PID:
        !           198:                sc->sc_type = CT88140;
        !           199:                break;
        !           200:        }
        !           201:        printf("ct%d: %s %stape\n", hd->hp_unit, ctinfo[id].desc,
        !           202:               (sc->sc_flags & CTF_CANSTREAM) ? "streaming " : " ");
        !           203:        return(id);
        !           204: }
        !           205: 
        !           206: ctreset(sc, hd)
        !           207:        register struct ct_softc *sc;
        !           208:        register struct hp_device *hd;
        !           209: {
        !           210:        u_char stat;
        !           211: 
        !           212:        sc->sc_clear.unit = C_SUNIT(sc->sc_punit);
        !           213:        sc->sc_clear.cmd = C_CLEAR;
        !           214:        hpibsend(hd->hp_ctlr, hd->hp_slave, C_TCMD, &sc->sc_clear,
        !           215:                sizeof(sc->sc_clear));
        !           216:        hpibswait(hd->hp_ctlr, hd->hp_slave);
        !           217:        hpibrecv(hd->hp_ctlr, hd->hp_slave, C_QSTAT, &stat, sizeof(stat));
        !           218:        sc->sc_src.unit = C_SUNIT(CTCTLR);
        !           219:        sc->sc_src.nop = C_NOP;
        !           220:        sc->sc_src.cmd = C_SREL;
        !           221:        sc->sc_src.param = C_REL;
        !           222:        hpibsend(hd->hp_ctlr, hd->hp_slave, C_CMD, &sc->sc_src,
        !           223:                sizeof(sc->sc_src));
        !           224:        hpibswait(hd->hp_ctlr, hd->hp_slave);
        !           225:        hpibrecv(hd->hp_ctlr, hd->hp_slave, C_QSTAT, &stat, sizeof(stat));
        !           226:        sc->sc_ssmc.unit = C_SUNIT(sc->sc_punit);
        !           227:        sc->sc_ssmc.cmd = C_SSM;
        !           228:        sc->sc_ssmc.refm = REF_MASK;
        !           229:        sc->sc_ssmc.fefm = FEF_MASK;
        !           230:        sc->sc_ssmc.aefm = AEF_MASK;
        !           231:        sc->sc_ssmc.iefm = IEF_MASK;
        !           232:        hpibsend(hd->hp_ctlr, hd->hp_slave, C_CMD, &sc->sc_ssmc,
        !           233:                sizeof(sc->sc_ssmc));
        !           234:        hpibswait(hd->hp_ctlr, hd->hp_slave);
        !           235:        hpibrecv(hd->hp_ctlr, hd->hp_slave, C_QSTAT, &stat, sizeof(stat));
        !           236:        sc->sc_soptc.unit = C_SUNIT(sc->sc_punit);
        !           237:        sc->sc_soptc.nop = C_NOP;
        !           238:        sc->sc_soptc.cmd = C_SOPT;
        !           239:        sc->sc_soptc.opt = C_SPAR;
        !           240:        hpibsend(hd->hp_ctlr, hd->hp_slave, C_CMD, &sc->sc_soptc,
        !           241:                sizeof(sc->sc_soptc));
        !           242:        hpibswait(hd->hp_ctlr, hd->hp_slave);
        !           243:        hpibrecv(hd->hp_ctlr, hd->hp_slave, C_QSTAT, &stat, sizeof(stat));
        !           244: }
        !           245: 
        !           246: /*ARGSUSED*/
        !           247: ctopen(dev, flag)
        !           248:        dev_t dev;
        !           249: {
        !           250:        register struct ct_softc *sc = &ct_softc[UNIT(dev)];
        !           251:        u_char stat;
        !           252:        int cc;
        !           253: 
        !           254:        if (UNIT(dev) >= NCT || (sc->sc_flags & CTF_ALIVE) == 0)
        !           255:                return(ENXIO);
        !           256:        if (sc->sc_flags & CTF_OPEN)
        !           257:                return(EBUSY);
        !           258:        sc->sc_soptc.unit = C_SUNIT(sc->sc_punit);
        !           259:        sc->sc_soptc.nop = C_NOP;
        !           260:        sc->sc_soptc.cmd = C_SOPT;
        !           261:        if ((dev & CT_STREAM) && (sc->sc_flags & CTF_CANSTREAM))
        !           262:                sc->sc_soptc.opt = C_SPAR | C_IMRPT;
        !           263:        else
        !           264:                sc->sc_soptc.opt = C_SPAR;
        !           265:        /* 
        !           266:         * Check the return of hpibsend() and hpibswait().
        !           267:         * Drive could be loading/unloading a tape. If not checked,
        !           268:         * driver hangs. 
        !           269:         */
        !           270:        cc = hpibsend(sc->sc_hd->hp_ctlr, sc->sc_hd->hp_slave,
        !           271:                      C_CMD, &sc->sc_soptc, sizeof(sc->sc_soptc));
        !           272:        if (cc != sizeof(sc->sc_soptc))
        !           273:                return(EBUSY);
        !           274:        hpibswait(sc->sc_hd->hp_ctlr, sc->sc_hd->hp_slave);
        !           275:        cc = hpibrecv(sc->sc_hd->hp_ctlr, sc->sc_hd->hp_slave, C_QSTAT, 
        !           276:                      &stat, sizeof(stat));
        !           277:        if (cc != sizeof(stat))
        !           278:                return(EBUSY);
        !           279:        sc->sc_ctty = (caddr_t)(u.u_procp->p_flag&SCTTY ? 
        !           280:                        u.u_procp->p_session->s_ttyvp : 0);
        !           281:        sc->sc_flags |= CTF_OPEN;
        !           282:        return(0);
        !           283: }
        !           284: 
        !           285: /*ARGSUSED*/
        !           286: ctclose(dev, flag)
        !           287:        dev_t dev;
        !           288: {
        !           289:        register struct ct_softc *sc = &ct_softc[UNIT(dev)];
        !           290: 
        !           291:        if ((sc->sc_flags & (CTF_WRT|CTF_WRTTN)) == (CTF_WRT|CTF_WRTTN) &&
        !           292:            (sc->sc_flags & CTF_EOT) == 0 ) { /* XXX return error if EOT ?? */
        !           293:                ctcommand(dev, MTWEOF, 2);
        !           294:                ctcommand(dev, MTBSR, 1);
        !           295:                if (sc->sc_eofp == EOFS - 1)
        !           296:                        sc->sc_eofs[EOFS - 1]--;
        !           297:                else
        !           298:                        sc->sc_eofp--;
        !           299: #ifdef DEBUG
        !           300:                if(ctdebug & CT_BSF)
        !           301:                        printf("ct%d: ctclose backup eofs prt %d blk %d\n",
        !           302:                               UNIT(dev), sc->sc_eofp, sc->sc_eofs[sc->sc_eofp]);
        !           303: #endif
        !           304:        }
        !           305:        if ((minor(dev) & CT_NOREW) == 0)
        !           306:                ctcommand(dev, MTREW, 1);
        !           307:        sc->sc_flags &= ~(CTF_OPEN | CTF_WRT | CTF_WRTTN);
        !           308:        sc->sc_ctty = NULL;
        !           309: #ifdef DEBUG
        !           310:        if (ctdebug & CDB_FILES)
        !           311:                printf("ctclose: flags %x\n", sc->sc_flags);
        !           312: #endif
        !           313:        return(0);      /* XXX */
        !           314: }
        !           315: 
        !           316: ctcommand(dev, cmd, cnt)
        !           317:        dev_t dev;
        !           318:        register int cnt;
        !           319: {
        !           320:        register struct ct_softc *sc = &ct_softc[UNIT(dev)];
        !           321:        register struct buf *bp = &ctbuf[UNIT(dev)];
        !           322:        register struct buf *nbp = 0;
        !           323: 
        !           324:        if (cmd == MTBSF && sc->sc_eofp == EOFS - 1) {
        !           325:                cnt = sc->sc_eofs[EOFS - 1] - cnt;
        !           326:                ctcommand(dev, MTREW, 1);
        !           327:                ctcommand(dev, MTFSF, cnt);
        !           328:                cnt = 2;
        !           329:                cmd = MTBSR;
        !           330:        }
        !           331: 
        !           332:        if (cmd == MTBSF && sc->sc_eofp - cnt < 0) {
        !           333:                cnt = 1;
        !           334:                cmd = MTREW;
        !           335:        }
        !           336: 
        !           337:        sc->sc_flags |= CTF_CMD;
        !           338:        sc->sc_bp = bp;
        !           339:        sc->sc_cmd = cmd;
        !           340:        bp->b_dev = dev;
        !           341:        if (cmd == MTFSF) {
        !           342:                nbp = (struct buf *)geteblk(MAXBSIZE);
        !           343:                bp->b_un.b_addr = nbp->b_un.b_addr;
        !           344:                bp->b_bcount = MAXBSIZE;
        !           345:        }
        !           346: again:
        !           347:        bp->b_flags = B_BUSY;
        !           348:        if (cmd == MTBSF) {
        !           349:                sc->sc_blkno = sc->sc_eofs[sc->sc_eofp];
        !           350:                sc->sc_eofp--;
        !           351: #ifdef DEBUG
        !           352:                if (ctdebug & CT_BSF)
        !           353:                        printf("ct%d: backup eof pos %d blk %d\n",
        !           354:                               UNIT(dev), sc->sc_eofp, 
        !           355:                               sc->sc_eofs[sc->sc_eofp]);
        !           356: #endif
        !           357:        }
        !           358:        ctstrategy(bp);
        !           359:        iowait(bp);
        !           360:        if (--cnt > 0)
        !           361:                goto again;
        !           362:        bp->b_flags = 0;
        !           363:        sc->sc_flags &= ~CTF_CMD;
        !           364:        if (nbp)
        !           365:                brelse(nbp);
        !           366: }
        !           367: 
        !           368: ctstrategy(bp)
        !           369:        register struct buf *bp;
        !           370: {
        !           371:        register struct buf *dp;
        !           372:        register int s, unit;
        !           373: 
        !           374:        unit = UNIT(bp->b_dev);
        !           375:        dp = &cttab[unit];
        !           376:        bp->av_forw = NULL;
        !           377:        s = splbio();
        !           378:        if (dp->b_actf == NULL)
        !           379:                dp->b_actf = bp;
        !           380:        else
        !           381:                dp->b_actl->av_forw = bp;
        !           382:        dp->b_actl = bp;
        !           383:        if (dp->b_active == 0) {
        !           384:                dp->b_active = 1;
        !           385:                ctustart(unit);
        !           386:        }
        !           387:        splx(s);
        !           388: }
        !           389: 
        !           390: ctustart(unit)
        !           391:        register int unit;
        !           392: {
        !           393:        register struct ct_softc *sc = &ct_softc[unit];
        !           394:        register struct buf *bp;
        !           395: 
        !           396:        bp = cttab[unit].b_actf;
        !           397:        sc->sc_addr = bp->b_un.b_addr;
        !           398:        sc->sc_resid = bp->b_bcount;
        !           399:        if (hpibreq(&sc->sc_dq))
        !           400:                ctstart(unit);
        !           401: }
        !           402: 
        !           403: ctstart(unit)
        !           404:        register int unit;
        !           405: {
        !           406:        register struct ct_softc *sc = &ct_softc[unit];
        !           407:        register struct buf *bp;
        !           408:        register int i;
        !           409: 
        !           410:        bp = cttab[unit].b_actf;
        !           411: again:
        !           412:        if ((sc->sc_flags & CTF_CMD) && sc->sc_bp == bp) {
        !           413:                switch(sc->sc_cmd) {
        !           414: 
        !           415:                case MTFSF:
        !           416:                        bp->b_flags |= B_READ;
        !           417:                        goto mustio;
        !           418: 
        !           419:                case MTBSF:
        !           420:                        goto gotaddr;
        !           421: 
        !           422:                case MTOFFL:
        !           423:                        sc->sc_blkno = 0;
        !           424:                        sc->sc_ul.unit = C_SUNIT(sc->sc_punit);
        !           425:                        sc->sc_ul.cmd = C_UNLOAD;
        !           426:                        hpibsend(sc->sc_hd->hp_ctlr, sc->sc_hd->hp_slave,
        !           427:                                C_CMD, &sc->sc_ul, sizeof(sc->sc_ul));
        !           428:                        break;
        !           429: 
        !           430:                case MTWEOF:
        !           431:                        sc->sc_blkno++;
        !           432:                        sc->sc_flags |= CTF_WRT;
        !           433:                        sc->sc_wfm.unit = C_SUNIT(sc->sc_punit);
        !           434:                        sc->sc_wfm.cmd = C_WFM;
        !           435:                        hpibsend(sc->sc_hd->hp_ctlr, sc->sc_hd->hp_slave,
        !           436:                                C_CMD, &sc->sc_wfm, sizeof(sc->sc_wfm));
        !           437:                        ctaddeof(unit);
        !           438:                        break;
        !           439: 
        !           440:                case MTBSR:
        !           441:                        sc->sc_blkno--;
        !           442:                        goto gotaddr;
        !           443: 
        !           444:                case MTFSR:
        !           445:                        sc->sc_blkno++;
        !           446:                        goto gotaddr;
        !           447: 
        !           448:                case MTREW:
        !           449:                        sc->sc_blkno = 0;
        !           450: #ifdef DEBUG
        !           451:                        if(ctdebug & CT_BSF)
        !           452:                                printf("ct%d: clearing eofs\n", unit);
        !           453: #endif
        !           454:                        for (i=0; i<EOFS; i++)
        !           455:                                sc->sc_eofs[i] = 0;
        !           456:                        sc->sc_eofp = 0;
        !           457: 
        !           458: gotaddr:
        !           459:                        sc->sc_ioc.saddr = C_SADDR;
        !           460:                        sc->sc_ioc.addr0 = 0;
        !           461:                        sc->sc_ioc.addr = sc->sc_blkno;
        !           462:                        sc->sc_ioc.unit = C_SUNIT(sc->sc_punit);
        !           463:                        sc->sc_ioc.nop2 = C_NOP;
        !           464:                        sc->sc_ioc.slen = C_SLEN;
        !           465:                        sc->sc_ioc.len = 0;
        !           466:                        sc->sc_ioc.nop3 = C_NOP;
        !           467:                        sc->sc_ioc.cmd = C_READ;
        !           468:                        hpibsend(sc->sc_hd->hp_ctlr, sc->sc_hd->hp_slave,
        !           469:                                C_CMD, &sc->sc_ioc, sizeof(sc->sc_ioc));
        !           470:                        break;
        !           471:                }
        !           472:        }
        !           473:        else {
        !           474: mustio:
        !           475:                if ((bp->b_flags & B_READ) &&
        !           476:                    sc->sc_flags & (CTF_BEOF|CTF_EOT)) {
        !           477: #ifdef DEBUG
        !           478:                        if (ctdebug & CDB_FILES)
        !           479:                                printf("ctstart: before flags %x\n", sc->sc_flags);
        !           480: #endif
        !           481:                        if (sc->sc_flags & CTF_BEOF) {
        !           482:                                sc->sc_flags &= ~CTF_BEOF;
        !           483:                                sc->sc_flags |= CTF_AEOF;
        !           484: #ifdef DEBUG
        !           485:                                if (ctdebug & CDB_FILES)
        !           486:                                        printf("ctstart: after flags %x\n", sc->sc_flags);
        !           487: #endif
        !           488:                        }
        !           489:                        bp->b_resid = bp->b_bcount;
        !           490:                        iodone(bp);
        !           491:                        hpibfree(&sc->sc_dq);
        !           492:                        cttab[unit].b_actf = bp = bp->av_forw;
        !           493:                        if (bp == NULL) {
        !           494:                                cttab[unit].b_active = 0;
        !           495:                                return;
        !           496:                        }
        !           497:                        sc->sc_addr = bp->b_un.b_addr;
        !           498:                        sc->sc_resid = bp->b_bcount;
        !           499:                        if (hpibreq(&sc->sc_dq))
        !           500:                                goto again;
        !           501:                        return;
        !           502:                }                       
        !           503:                sc->sc_flags |= CTF_IO;
        !           504:                sc->sc_ioc.unit = C_SUNIT(sc->sc_punit);
        !           505:                sc->sc_ioc.saddr = C_SADDR;
        !           506:                sc->sc_ioc.addr0 = 0;
        !           507:                sc->sc_ioc.addr = sc->sc_blkno;
        !           508:                sc->sc_ioc.nop2 = C_NOP;
        !           509:                sc->sc_ioc.slen = C_SLEN;
        !           510:                sc->sc_ioc.len = sc->sc_resid;
        !           511:                sc->sc_ioc.nop3 = C_NOP;
        !           512:                if (bp->b_flags & B_READ)
        !           513:                        sc->sc_ioc.cmd = C_READ;
        !           514:                else {
        !           515:                        sc->sc_ioc.cmd = C_WRITE;
        !           516:                        sc->sc_flags |= (CTF_WRT | CTF_WRTTN);
        !           517:                }
        !           518:                hpibsend(sc->sc_hd->hp_ctlr, sc->sc_hd->hp_slave, C_CMD,
        !           519:                        &sc->sc_ioc, sizeof(sc->sc_ioc));
        !           520:        }
        !           521:        hpibawait(sc->sc_hd->hp_ctlr);
        !           522: }
        !           523: 
        !           524: ctgo(unit)
        !           525:        register int unit;
        !           526: {
        !           527:        register struct ct_softc *sc = &ct_softc[unit];
        !           528:        register struct buf *bp;
        !           529: 
        !           530:        bp = cttab[unit].b_actf;
        !           531:        hpibgo(sc->sc_hd->hp_ctlr, sc->sc_hd->hp_slave, C_EXEC,
        !           532:                sc->sc_addr, sc->sc_resid, bp->b_flags & B_READ);
        !           533: }
        !           534: 
        !           535: /*
        !           536:  * Hideous grue to handle EOF/EOT (mostly for reads)
        !           537:  */
        !           538: cteof(sc, bp)
        !           539:        register struct ct_softc *sc;
        !           540:        register struct buf *bp;
        !           541: {
        !           542:        long blks;
        !           543: 
        !           544:        /*
        !           545:         * EOT on a write is an error.
        !           546:         */
        !           547:        if ((bp->b_flags & B_READ) == 0) {
        !           548:                bp->b_resid = bp->b_bcount;
        !           549:                bp->b_flags |= B_ERROR;
        !           550:                bp->b_error = ENOSPC;
        !           551:                sc->sc_flags |= CTF_EOT;
        !           552:                return;
        !           553:        }
        !           554:        /*
        !           555:         * Use returned block position to determine how many blocks
        !           556:         * we really read and update b_resid.
        !           557:         */
        !           558:        blks = sc->sc_stat.c_blk - sc->sc_blkno - 1;
        !           559: #ifdef DEBUG
        !           560:        if (ctdebug & CDB_FILES)
        !           561:                printf("cteof: bc %d oblk %d nblk %d read %d, resid %d\n",
        !           562:                       bp->b_bcount, sc->sc_blkno, sc->sc_stat.c_blk,
        !           563:                       blks, bp->b_bcount - CTKTOB(blks));
        !           564: #endif
        !           565:        if (blks == -1) { /* 9145 on EOF does not change sc_stat.c_blk */
        !           566:                blks = 0;
        !           567:                sc->sc_blkno++;
        !           568:        }
        !           569:        else {
        !           570:                sc->sc_blkno = sc->sc_stat.c_blk;
        !           571:        }
        !           572:        bp->b_resid = bp->b_bcount - CTKTOB(blks);
        !           573:        /*
        !           574:         * If we are at physical EOV or were after an EOF,
        !           575:         * we are now at logical EOT.
        !           576:         */
        !           577:        if ((sc->sc_stat.c_aef & AEF_EOV) ||
        !           578:            (sc->sc_flags & CTF_AEOF)) {
        !           579:                sc->sc_flags |= CTF_EOT;
        !           580:                sc->sc_flags &= ~(CTF_AEOF|CTF_BEOF);
        !           581:        }
        !           582:        /*
        !           583:         * If we were before an EOF or we have just completed a FSF,
        !           584:         * we are now after EOF.
        !           585:         */
        !           586:        else if ((sc->sc_flags & CTF_BEOF) ||
        !           587:                 (sc->sc_flags & CTF_CMD) && sc->sc_cmd == MTFSF) {
        !           588:                sc->sc_flags |= CTF_AEOF;
        !           589:                sc->sc_flags &= ~CTF_BEOF;
        !           590:        }
        !           591:        /*
        !           592:         * Otherwise if we read something we are now before EOF
        !           593:         * (and no longer after EOF).
        !           594:         */
        !           595:        else if (blks) {
        !           596:                sc->sc_flags |= CTF_BEOF;
        !           597:                sc->sc_flags &= ~CTF_AEOF;
        !           598:        }
        !           599:        /*
        !           600:         * Finally, if we didn't read anything we just passed an EOF
        !           601:         */
        !           602:        else
        !           603:                sc->sc_flags |= CTF_AEOF;
        !           604: #ifdef DEBUG
        !           605:        if (ctdebug & CDB_FILES)
        !           606:                printf("cteof: leaving flags %x\n", sc->sc_flags);
        !           607: #endif
        !           608: }
        !           609: 
        !           610: ctintr(unit)
        !           611:        register int unit;
        !           612: {
        !           613:        register struct ct_softc *sc = &ct_softc[unit];
        !           614:        register struct buf *bp;
        !           615:        u_char stat;
        !           616: 
        !           617:        bp = cttab[unit].b_actf;
        !           618:        if (bp == NULL) {
        !           619:                printf("ct%d: bp == NULL\n", unit);
        !           620:                return;
        !           621:        }
        !           622:        if (sc->sc_flags & CTF_IO) {
        !           623:                sc->sc_flags &= ~CTF_IO;
        !           624:                if (hpibustart(sc->sc_hd->hp_ctlr))
        !           625:                        ctgo(unit);
        !           626:                return;
        !           627:        }
        !           628:        if ((sc->sc_flags & CTF_STATWAIT) == 0) {
        !           629:                if (hpibpptest(sc->sc_hd->hp_ctlr, sc->sc_hd->hp_slave) == 0) {
        !           630:                        sc->sc_flags |= CTF_STATWAIT;
        !           631:                        hpibawait(sc->sc_hd->hp_ctlr);
        !           632:                        return;
        !           633:                }
        !           634:        } else
        !           635:                sc->sc_flags &= ~CTF_STATWAIT;
        !           636:        hpibrecv(sc->sc_hd->hp_ctlr, sc->sc_hd->hp_slave, C_QSTAT, &stat, 1);
        !           637: #ifdef DEBUG
        !           638:        if (ctdebug & CDB_FILES)
        !           639:                printf("ctintr: before flags %x\n", sc->sc_flags);
        !           640: #endif
        !           641:        if (stat) {
        !           642:                sc->sc_rsc.unit = C_SUNIT(sc->sc_punit);
        !           643:                sc->sc_rsc.cmd = C_STATUS;
        !           644:                hpibsend(sc->sc_hd->hp_ctlr, sc->sc_hd->hp_slave, C_CMD,
        !           645:                        &sc->sc_rsc, sizeof(sc->sc_rsc));
        !           646:                hpibrecv(sc->sc_hd->hp_ctlr, sc->sc_hd->hp_slave, C_EXEC,
        !           647:                        &sc->sc_stat, sizeof(sc->sc_stat));
        !           648:                hpibrecv(sc->sc_hd->hp_ctlr, sc->sc_hd->hp_slave, C_QSTAT,
        !           649:                        &stat, 1);
        !           650: #ifdef DEBUG
        !           651:                if (ctdebug & CDB_FILES)
        !           652:                        printf("ctintr: return stat 0x%x, A%x F%x blk %d\n",
        !           653:                               stat, sc->sc_stat.c_aef,
        !           654:                               sc->sc_stat.c_fef, sc->sc_stat.c_blk);
        !           655: #endif
        !           656:                if (stat == 0) {
        !           657:                        if (sc->sc_stat.c_aef & (AEF_EOF | AEF_EOV)) {
        !           658:                                cteof(sc, bp);
        !           659:                                ctaddeof(unit);
        !           660:                                goto done;
        !           661:                        }
        !           662:                        if (sc->sc_stat.c_fef & FEF_PF) {
        !           663:                                ctreset(sc, sc->sc_hd);
        !           664:                                ctstart(unit);
        !           665:                                return;
        !           666:                        }
        !           667:                        if (sc->sc_stat.c_fef & FEF_REXMT) {
        !           668:                                ctstart(unit);
        !           669:                                return;
        !           670:                        }
        !           671:                        if (sc->sc_stat.c_aef & 0x5800) {
        !           672:                                if (sc->sc_stat.c_aef & 0x4000)
        !           673:                                        tprintf(sc->sc_ctty,
        !           674:                                                "ct%d: uninitialized media\n",
        !           675:                                                unit);
        !           676:                                if (sc->sc_stat.c_aef & 0x1000)
        !           677:                                        tprintf(sc->sc_ctty,
        !           678:                                                "ct%d: not ready\n", unit);
        !           679:                                if (sc->sc_stat.c_aef & 0x0800)
        !           680:                                        tprintf(sc->sc_ctty,
        !           681:                                                "ct%d: write protect\n", unit);
        !           682:                        } else {
        !           683:                                printf("ct%d err: v%d u%d ru%d bn%d, ",
        !           684:                                       unit,
        !           685:                                       (sc->sc_stat.c_vu>>4)&0xF,
        !           686:                                       sc->sc_stat.c_vu&0xF,
        !           687:                                       sc->sc_stat.c_pend,
        !           688:                                       sc->sc_stat.c_blk);
        !           689:                                printf("R0x%x F0x%x A0x%x I0x%x\n",
        !           690:                                       sc->sc_stat.c_ref,
        !           691:                                       sc->sc_stat.c_fef,
        !           692:                                       sc->sc_stat.c_aef,
        !           693:                                       sc->sc_stat.c_ief);
        !           694:                        }
        !           695:                } else
        !           696:                        printf("ct%d: request status failed\n", unit);
        !           697:                bp->b_flags |= B_ERROR;
        !           698:                bp->b_error = EIO;
        !           699:                goto done;
        !           700:        } else
        !           701:                bp->b_resid = 0;
        !           702:        if (sc->sc_flags & CTF_CMD) {
        !           703:                switch (sc->sc_cmd) {
        !           704:                case MTFSF:
        !           705:                        sc->sc_flags &= ~(CTF_BEOF|CTF_AEOF);
        !           706:                        sc->sc_blkno += CTBTOK(sc->sc_resid);
        !           707:                        ctstart(unit);
        !           708:                        return;
        !           709:                case MTBSF:
        !           710:                        sc->sc_flags &= ~(CTF_AEOF|CTF_BEOF|CTF_EOT);
        !           711:                        break;
        !           712:                case MTBSR:
        !           713:                        sc->sc_flags &= ~CTF_BEOF;
        !           714:                        if (sc->sc_flags & CTF_EOT) {
        !           715:                                sc->sc_flags |= CTF_AEOF;
        !           716:                                sc->sc_flags &= ~CTF_EOT;
        !           717:                        } else if (sc->sc_flags & CTF_AEOF) {
        !           718:                                sc->sc_flags |= CTF_BEOF;
        !           719:                                sc->sc_flags &= ~CTF_AEOF;
        !           720:                        }
        !           721:                        break;
        !           722:                case MTWEOF:
        !           723:                        sc->sc_flags &= ~CTF_BEOF;
        !           724:                        if (sc->sc_flags & (CTF_AEOF|CTF_EOT)) {
        !           725:                                sc->sc_flags |= CTF_EOT;
        !           726:                                sc->sc_flags &= ~CTF_AEOF;
        !           727:                        } else
        !           728:                                sc->sc_flags |= CTF_AEOF;
        !           729:                        break;
        !           730:                case MTREW:
        !           731:                case MTOFFL:
        !           732:                        sc->sc_flags &= ~(CTF_BEOF|CTF_AEOF|CTF_EOT);
        !           733:                        break;
        !           734:                }
        !           735:        } else {
        !           736:                sc->sc_flags &= ~CTF_AEOF;
        !           737:                sc->sc_blkno += CTBTOK(sc->sc_resid);
        !           738:        }
        !           739: done:
        !           740: #ifdef DEBUG
        !           741:        if (ctdebug & CDB_FILES)
        !           742:                printf("ctintr: after flags %x\n", sc->sc_flags);
        !           743: #endif
        !           744:        cttab[unit].b_actf = bp->av_forw;
        !           745:        iodone(bp);
        !           746:        hpibfree(&sc->sc_dq);
        !           747:        if (cttab[unit].b_actf == NULL) {
        !           748:                cttab[unit].b_active = 0;
        !           749:                return;
        !           750:        }
        !           751:        ctustart(unit);
        !           752: }
        !           753: 
        !           754: ctread(dev, uio)
        !           755:        dev_t dev;
        !           756:        struct uio *uio;
        !           757: {
        !           758:        register int unit = UNIT(dev);
        !           759: 
        !           760:        return(physio(ctstrategy, &ctbuf[unit], dev, B_READ, minphys, uio));
        !           761: }
        !           762: 
        !           763: ctwrite(dev, uio)
        !           764:        dev_t dev;
        !           765:        struct uio *uio;
        !           766: {
        !           767:        register int unit = UNIT(dev);
        !           768: 
        !           769:        return(physio(ctstrategy, &ctbuf[unit], dev, B_WRITE, minphys, uio));
        !           770: }
        !           771: 
        !           772: /*ARGSUSED*/
        !           773: ctioctl(dev, cmd, data, flag)
        !           774:        dev_t dev;
        !           775:        caddr_t data;
        !           776: {
        !           777:        register struct mtop *op;
        !           778:        register int cnt;
        !           779: 
        !           780:        switch (cmd) {
        !           781: 
        !           782:        case MTIOCTOP:
        !           783:                op = (struct mtop *)data;
        !           784:                switch(op->mt_op) {
        !           785: 
        !           786:                case MTWEOF:
        !           787:                case MTFSF:
        !           788:                case MTBSR:
        !           789:                case MTBSF:
        !           790:                case MTFSR:
        !           791:                        cnt = op->mt_count;
        !           792:                        break;
        !           793: 
        !           794:                case MTREW:
        !           795:                case MTOFFL:
        !           796:                        cnt = 1;
        !           797:                        break;
        !           798: 
        !           799:                default:
        !           800:                        return(EINVAL);
        !           801:                }
        !           802:                ctcommand(dev, op->mt_op, cnt);
        !           803:                break;
        !           804: 
        !           805:        case MTIOCGET:
        !           806:                break;
        !           807: 
        !           808:        default:
        !           809:                return(EINVAL);
        !           810:        }
        !           811:        return(0);
        !           812: }
        !           813: 
        !           814: /*ARGSUSED*/
        !           815: ctdump(dev)
        !           816:        dev_t dev;
        !           817: {
        !           818:        return(ENXIO);
        !           819: }
        !           820: 
        !           821: ctaddeof(unit)
        !           822:        int unit;
        !           823: {
        !           824:        register struct ct_softc *sc = &ct_softc[unit];
        !           825:        
        !           826:        if (sc->sc_eofp == EOFS - 1)
        !           827:                sc->sc_eofs[EOFS - 1]++;
        !           828:        else {
        !           829:                sc->sc_eofp++;
        !           830:                if (sc->sc_eofp == EOFS - 1)
        !           831:                        sc->sc_eofs[EOFS - 1] = EOFS;
        !           832:                else
        !           833:                        /* save blkno */
        !           834:                        sc->sc_eofs[sc->sc_eofp] = sc->sc_blkno - 1;
        !           835:        }
        !           836: #ifdef DEBUG
        !           837:        if (ctdebug & CT_BSF)
        !           838:                printf("ct%d: add eof pos %d blk %d\n",
        !           839:                       unit, sc->sc_eofp,
        !           840:                       sc->sc_eofs[sc->sc_eofp]);
        !           841: #endif
        !           842: }
        !           843: #endif

unix.superglobalmegacorp.com

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