Annotation of 43BSDReno/sys/vaxuba/tmscp.c, revision 1.1

1.1     ! root        1: /*     @(#)tmscp.c     7.14 (Berkeley) 7/1/90 */
        !             2: 
        !             3: #ifndef lint
        !             4: static char    *sccsid = "@(#)tmscp.c  1.24    (ULTRIX)        1/21/86";
        !             5: #endif lint
        !             6: 
        !             7: 
        !             8: /************************************************************************
        !             9:  *                                                                     *
        !            10:  *        Licensed from Digital Equipment Corporation                  *
        !            11:  *                       Copyright (c)                                         *
        !            12:  *               Digital Equipment Corporation                         *
        !            13:  *                   Maynard, Massachusetts                            *
        !            14:  *                         1985, 1986                                  *
        !            15:  *                    All rights reserved.                             *
        !            16:  *                                                                     *
        !            17:  *        The Information in this software is subject to change        *
        !            18:  *   without notice and should not be construed as a commitment        *
        !            19:  *   by  Digital  Equipment  Corporation.   Digital   makes  no        *
        !            20:  *   representations about the suitability of this software for        *
        !            21:  *   any purpose.  It is supplied "As Is" without expressed  or        *
        !            22:  *   implied  warranty.                                                *
        !            23:  *                                                                     *
        !            24:  *        If the Regents of the University of California or its        *
        !            25:  *   licensees modify the software in a manner creating                *
        !            26:  *   diriviative copyright rights, appropriate copyright               *
        !            27:  *   legends may be placed on  the drivative work in addition                  *
        !            28:  *   to that set forth above.                                          *
        !            29:  *                                                                     *
        !            30:  ************************************************************************
        !            31:  * 
        !            32:  * tmscp.c - TMSCP (TK50/TU81) tape device driver
        !            33:  * 
        !            34:  * Modification History:
        !            35:  *
        !            36:  * 06-Jan-86 - afd
        !            37:  *     Changed the probe routine to use DELAY (not TODR).  This now
        !            38:  *     works for MicroVAXen as well.  This eliminates the busy-wait
        !            39:  *     for MicroVAXen so a dead TK50 controller will not hang autoconf.
        !            40:  *
        !            41:  * 06-Dec-85 - afd
        !            42:  *     Fixed a bug in density selection.  The "set unit characteristics"
        !            43:  *     command to select density, was clearing the "unit flags" field
        !            44:  *     where the CACHE bit was for TU81-E.  Now the unit's "format" and
        !            45:  *     "unitflgs" are saved in tms_info struct.  And are used on STUNT
        !            46:  *     commands.
        !            47:  *
        !            48:  * 19-Oct-85 - afd
        !            49:  *     Added support to the open routine to allow drives to be opened
        !            50:  *     for low density (800 or 1600 bpi) use.  When the slave routine
        !            51:  *     initiates a "get-unit-char" cmd, the format menu for the unit
        !            52:  *     is saved in the tms_info structure. The format menu is used in the
        !            53:  *     start routine to select the proper low density.
        !            54:  *
        !            55:  * 02-Oct-85 - afd
        !            56:  *     When a tmscp-type controller is initializing, it is possible for
        !            57:  *     the sa reg to become 0 between states.  Thus the init code in
        !            58:  *     the interrupt routine had to be modified to reflect this.
        !            59:  *
        !            60:  * 21-Sep-85 - afd
        !            61:  *     The TK50 declares a serious exception when a tape mark is encountered.
        !            62:  *     This causes problems to dd (& other UN*X utilities).  So a flag
        !            63:  *     is set in the rsp() routine when a tape mark is encountered.  If
        !            64:  *     this flag is set, the start() routine appends the Clear Serious
        !            65:  *     Exception modifier to the next command.
        !            66:  *
        !            67:  * 03-Sep-85 -- jaw
        !            68:  *     messed up previous edit..
        !            69:  *
        !            70:  * 29-Aug-85 - jaw
        !            71:  *     fixed bugs in 8200 and 750 buffered datapath handling.
        !            72:  *
        !            73:  * 06-Aug-85 - afd
        !            74:  *   1. When repositioning records or files, the count of items skipped
        !            75:  *     does NOT HAVE to be returned by controllers (& the TU81 doesn't).
        !            76:  *     So tmscprsp() had to be modified to stop reporting
        !            77:  *     residual count errors on reposition commands.
        !            78:  *
        !            79:  *   2. Fixed bug in the open routine which allowed multiple opens.
        !            80:  *
        !            81:  * 18-Jul-85 - afd
        !            82:  *   1. Need to return status when mt status (or corresponding ioctl) is done.
        !            83:  *     Save resid, flags, endcode & status in tmscprsp() routine (except on
        !            84:  *     clear serious exception no-op).  Return these fields when status
        !            85:  *     ioctl is done (in tmscpcommand()).  How they are returned:
        !            86:  *             mt_resid = resid
        !            87:  *             mt_dsreg = flags|endcode
        !            88:  *             mt_erreg = status
        !            89:  *
        !            90:  *   2. Added latent support for enabling/disabling caching.  This is
        !            91:  *     handled along with all other ioctl commands.
        !            92:  *
        !            93:  *   3. Need to issue a no-op on unrecognized ioctl in tmscpstart(), since
        !            94:  *     we have already commited to issuing a command at that point.
        !            95:  *
        !            96:  *   4. In tmscprsp() routine if encode is 0200 (invalid command issued);
        !            97:  *     We need to: Unlink the buffer from the I/O wait queue,
        !            98:  *     and signal iodone, so the higher level command can exit!
        !            99:  *     Just as if it were a valid command.
        !           100:  *
        !           101:  * 11-jul-85 -- jaw
        !           102:  *     fix bua/bda map registers.
        !           103:  *
        !           104:  * 19-Jun-85 -- jaw
        !           105:  *     VAX8200 name change.
        !           106:  *
        !           107:  * 06-Jun-85 - jaw
        !           108:  *     fixes for 8200.
        !           109:  *
        !           110:  * 9-Apr-85 - afd
        !           111:  *     Added timeout code to the probe routine, so if the controller
        !           112:  *     fails to init in 10 seconds we return failed status.
        !           113:  *
        !           114:  * 13-Mar-85 -jaw
        !           115:  *     Changes for support of the VAX8200 were merged in.
        !           116:  *
        !           117:  * 27-Feb-85 -tresvik
        !           118:  *     Changes for support of the VAX8600 were merged in.
        !           119:  *
        !           120:  */
        !           121: 
        !           122: #include "tms.h"
        !           123: #if NTMSCP > 0
        !           124: 
        !           125: #include "param.h"
        !           126: #include "systm.h"
        !           127: #include "buf.h"
        !           128: #include "conf.h"
        !           129: #include "errno.h"
        !           130: #include "file.h"
        !           131: #include "map.h"
        !           132: #include "vm.h"
        !           133: #include "ioctl.h"
        !           134: #include "syslog.h"
        !           135: #include "mtio.h"
        !           136: #include "cmap.h"
        !           137: #include "uio.h"
        !           138: #include "tprintf.h"
        !           139: 
        !           140: #include "../vax/pte.h"
        !           141: #include "../vax/cpu.h"
        !           142: #include "../vax/mtpr.h"
        !           143: #include "ubareg.h"
        !           144: #include "ubavar.h"
        !           145: 
        !           146: #define TENSEC (1000)
        !           147: #define        TMS_PRI LOG_INFO
        !           148: 
        !           149: #define NRSPL2  3               /* log2 number of response packets */
        !           150: #define NCMDL2  3               /* log2 number of command packets */
        !           151: #define NRSP    (1<<NRSPL2)
        !           152: #define NCMD    (1<<NCMDL2)
        !           153: 
        !           154: #include "tmscpreg.h"
        !           155: #include "../vax/tmscp.h"
        !           156: 
        !           157: /* Software state per controller */
        !           158: 
        !           159: struct tmscp_softc {
        !           160:        short   sc_state;       /* state of controller */
        !           161:        short   sc_mapped;      /* Unibus map allocated for tmscp struct? */
        !           162:        int     sc_ubainfo;     /* Unibus mapping info */
        !           163:        struct  tmscp *sc_tmscp;   /* Unibus address of tmscp struct */
        !           164:        int     sc_ivec;        /* interrupt vector address */
        !           165:        short   sc_credits;     /* transfer credits */
        !           166:        short   sc_lastcmd;     /* pointer into command ring */
        !           167:        short   sc_lastrsp;     /* pointer into response ring */
        !           168:        short   sc_ipl;         /* interrupt priority (Q-bus) */
        !           169: } tmscp_softc[NTMSCP];
        !           170: 
        !           171: struct tmscp {
        !           172:        struct tmscpca  tmscp_ca;         /* communications area */
        !           173:        struct mscp     tmscp_rsp[NRSP];  /* response packets */
        !           174:        struct mscp     tmscp_cmd[NCMD];  /* command packets */
        !           175: } tmscp[NTMSCP];
        !           176: 
        !           177: /*
        !           178:  * Per drive-unit info
        !           179:  */
        !           180: struct tms_info {
        !           181:        daddr_t         tms_dsize;      /* Max user size from online pkt */
        !           182:        unsigned        tms_type;       /* Drive type int field  */
        !           183:        int             tms_resid;      /* residual from last xfer */
        !           184:        u_char          tms_endcode;    /* last command endcode */
        !           185:        u_char          tms_flags;      /* last command end flags */
        !           186:        unsigned        tms_status;     /* Command status from last command */
        !           187:        char            tms_openf;      /* lock against multiple opens */
        !           188:        char            tms_lastiow;    /* last op was a write */
        !           189:        char            tms_serex;      /* set when serious exception occurs */
        !           190:        char            tms_clserex;    /* set when serex being cleared by no-op */
        !           191:        short           tms_fmtmenu;    /* the unit's format (density) menu */
        !           192:        short           tms_unitflgs;   /* unit flag parameters */
        !           193:        short           tms_format;     /* the unit's current format (density) */
        !           194:        tpr_t           tms_tpr;        /* tprintf handle */
        !           195: } tms_info[NTMS];
        !           196: struct uba_ctlr *tmscpminfo[NTMSCP];
        !           197: struct uba_device *tmsdinfo[NTMS];
        !           198: /* 
        !           199:  * ifdef other tmscp devices here if they allow more than 1 unit/controller
        !           200:  */
        !           201: struct uba_device *tmscpip[NTMSCP][1];
        !           202: struct buf ctmscpbuf[NTMSCP];          /* internal cmd buffer (for ioctls) */
        !           203: struct buf tmsutab[NTMS];              /* Drive queue */
        !           204: struct buf tmscpwtab[NTMSCP];          /* I/O wait queue, per controller */
        !           205: int    tmscpmicro[NTMSCP];             /* to store microcode level */
        !           206: short  utoctlr[NTMS];                  /* Slave unit to controller mapping */
        !           207:                                        /* filled in by the slave routine */
        !           208: 
        !           209: /* Bits in minor device */
        !           210: #define        TMSUNIT(dev)    (minor(dev)&03)
        !           211: #define        T_NOREWIND      04
        !           212: #define        T_HIDENSITY     010
        !           213: 
        !           214: /* Slave unit to controller mapping */
        !           215: #define TMSCPCTLR(dev) (utoctlr[TMSUNIT(dev)])
        !           216: 
        !           217: /*
        !           218:  * Internal (ioctl) command codes (these must also be declared in the
        !           219:  * tmscpioctl routine).  These correspond to ioctls in mtio.h
        !           220:  */
        !           221: #define TMS_WRITM      0               /* write tape mark */
        !           222: #define TMS_FSF                1               /* forward space file */
        !           223: #define TMS_BSF                2               /* backward space file */
        !           224: #define TMS_FSR                3               /* forward space record */
        !           225: #define TMS_BSR                4               /* backward space record */
        !           226: #define TMS_REW                5               /* rewind tape */
        !           227: #define TMS_OFFL       6               /* rewind tape & mark unit offline */
        !           228: #define TMS_SENSE      7               /* noop - do a get unit status */
        !           229: #define TMS_CACHE      8               /* enable cache */
        !           230: #define TMS_NOCACHE    9               /* disable cache */
        !           231: /* These go last: after all real mt cmds, just bump the numbers up */
        !           232: #define TMS_CSE                10              /* clear serious exception */
        !           233: #define TMS_LOWDENSITY 11              /* set unit to low density */
        !           234: #define TMS_HIDENSITY  12              /* set unit to high density */
        !           235: 
        !           236: /*
        !           237:  * Controller states
        !           238:  */
        !           239: #define S_IDLE  0               /* hasn't been initialized */
        !           240: #define S_STEP1 1               /* doing step 1 init */
        !           241: #define S_STEP2 2               /* doing step 2 init */
        !           242: #define S_STEP3 3               /* doing step 3 init */
        !           243: #define S_SCHAR 4               /* doing "set controller characteristics" */
        !           244: #define S_RUN   5               /* running */
        !           245: 
        !           246: int     tmscperror = 0;                /* causes hex dump of packets */
        !           247: int    tmscp_cp_wait = 0;      /* Something to wait on for command */
        !           248:                                /* packets and or credits. */
        !           249: int    wakeup();
        !           250: extern int     hz;             /* Should find the right include */
        !           251: 
        !           252: #ifdef DEBUG
        !           253: #define printd if (tmscpdebug) printf
        !           254: int tmscpdebug = 1;
        !           255: #define        printd10 if(tmscpdebug >= 10) printf
        !           256: #endif 
        !           257: 
        !           258: int     tmscpprobe(), tmscpslave(), tmscpattach(), tmscpintr();
        !           259: struct  mscp *tmscpgetcp();
        !           260: 
        !           261: #define DRVNAME "tms"
        !           262: #define CTRLNAME "tmscp"
        !           263: 
        !           264: u_short tmscpstd[] = { 0174500, 0 };
        !           265: struct  uba_driver tmscpdriver =
        !           266: { tmscpprobe, tmscpslave, tmscpattach, 0, tmscpstd, DRVNAME, tmsdinfo, CTRLNAME
        !           267: , tmscpminfo, 0};
        !           268: 
        !           269: #define b_qsize         b_resid         /* queue size per drive, in tmsutab */
        !           270: #define b_ubinfo        b_resid         /* Unibus mapping info, per buffer */
        !           271: 
        !           272: 
        !           273: /*************************************************************************/
        !           274: 
        !           275: #define DELAYTEN 1000
        !           276: 
        !           277: /*
        !           278:  * Unfortunately qbgetpri can't be used because the TK50 doesn't flip the
        !           279:  * TMSCP_STEP2 flag in the tmscpsa register until after the pending interrupt
        !           280:  * has been acknowledged by the cpu. If you are at spl6(), the TMSCP_STEP2
        !           281:  * flag never gets set and you return (0).
        !           282:  */
        !           283: tmscpprobe(reg, ctlr)
        !           284:        caddr_t reg;            /* address of the IP register */
        !           285:        int ctlr;               /* index of controller in the tmscp_softc array */
        !           286: {
        !           287:        register int br, cvec;  /* MUST be 1st (r11 & r10): IPL and intr vec */
        !           288:        register struct tmscp_softc *sc = &tmscp_softc[ctlr];
        !           289:                                /* ptr to software controller structure */
        !           290:        struct tmscpdevice *tmscpaddr; /* ptr to tmscpdevice struct (IP & SA) */
        !           291:        int count;              /* for probe delay time out */
        !           292: 
        !           293: #      ifdef lint
        !           294:        br = 0; cvec = br; br = cvec; reg = reg;
        !           295:        tmscpreset(0); tmscpintr(0);
        !           296: #      endif
        !           297: 
        !           298:        tmscpaddr = (struct tmscpdevice *) reg;
        !           299:        /* 
        !           300:         * Set host-settable interrupt vector.
        !           301:         * Assign 0 to the ip register to start the tmscp-device initialization.
        !           302:         * The device is not really initialized at this point, this is just to
        !           303:         * find out if the device exists.
        !           304:         */
        !           305:        sc->sc_ivec = (uba_hd[numuba].uh_lastiv -= 4);
        !           306:        tmscpaddr->tmscpip = 0;
        !           307: 
        !           308:        count=0;
        !           309:        while(count < DELAYTEN)
        !           310:                {       /* wait for at most 10 secs */
        !           311:                if((tmscpaddr->tmscpsa & TMSCP_STEP1) != 0)
        !           312:                        break;
        !           313:                DELAY(10000);
        !           314:                count=count+1;
        !           315:                }
        !           316:        if (count == DELAYTEN)
        !           317:                return(0);              
        !           318: 
        !           319:        tmscpaddr->tmscpsa = TMSCP_ERR|(NCMDL2<<11)|(NRSPL2<<8)|TMSCP_IE|(sc->sc_ivec/4);
        !           320: 
        !           321:        count=0;
        !           322:        while(count < DELAYTEN)
        !           323:                {
        !           324:                if((tmscpaddr->tmscpsa & TMSCP_STEP2) != 0)
        !           325:                        break;
        !           326:                DELAY(10000);
        !           327:                count = count+1;
        !           328:                }
        !           329:        if (count == DELAYTEN)
        !           330:                return(0);
        !           331: 
        !           332: #ifdef QBA
        !           333:        sc->sc_ipl = br = 0x15;
        !           334: #endif
        !           335:        return(sizeof (struct tmscpdevice));
        !           336: }
        !           337: 
        !           338: /*
        !           339:  * Try to find a slave (a drive) on the controller.
        !           340:  * If the controller is not in the run state, call init to initialize it.
        !           341:  */
        !           342: tmscpslave (ui, reg)
        !           343:        struct uba_device *ui;  /* ptr to the uba device structure */
        !           344:        caddr_t reg;            /* addr of the device controller */
        !           345: {
        !           346:        register struct uba_ctlr *um = tmscpminfo[ui->ui_ctlr];
        !           347:        register struct tmscp_softc *sc = &tmscp_softc[ui->ui_ctlr];
        !           348:        register struct tms_info *tms = &tms_info[ui->ui_unit];
        !           349:        struct   tmscpdevice *tmscpaddr;        /* ptr to IP & SA */
        !           350:        struct   mscp *mp;
        !           351:        int      i;                     /* Something to write into to start */
        !           352:                                        /* the tmscp polling */
        !           353: 
        !           354: #      ifdef lint
        !           355:        reg = reg;
        !           356: #      endif
        !           357:        tmscpaddr = (struct tmscpdevice *)um->um_addr;
        !           358:        /* 
        !           359:         * If its not in the run state, start the initialization process
        !           360:         * (tmscpintr will complete it);  if the initialization doesn't start;
        !           361:         * then return.
        !           362:         */
        !           363:        if(sc->sc_state != S_RUN)
        !           364:                {
        !           365: #              ifdef DEBUG
        !           366:                printd("tmscpslave: ctlr not running: calling init \n");
        !           367: #              endif   
        !           368:                if(!tmscpinit(ui->ui_ctlr))
        !           369:                        return(0);
        !           370:                }
        !           371:        /*
        !           372:         * Wait for the controller to come into the run state or go idle.
        !           373:         * If it goes idle return.
        !           374:         */
        !           375: #      ifdef DEBUG
        !           376:        i=1;
        !           377: #      endif   
        !           378:        while(sc->sc_state != S_RUN && sc->sc_state != S_IDLE)
        !           379: #              ifdef DEBUG
        !           380:                if (tmscpaddr->tmscpsa & TMSCP_ERR && i)
        !           381:                        {
        !           382:                        printd("tmscp-device: fatal error (%o)\n", tmscpaddr->tmscpsa&0xffff);
        !           383:                        i=0;
        !           384:                         }
        !           385: #              endif   
        !           386:                ;       /* wait */
        !           387:        if(sc->sc_state == S_IDLE)
        !           388:                {       /* The tmscp device failed to initialize */
        !           389:                printf("tmscp controller failed to init\n");
        !           390:                return(0);
        !           391:                }
        !           392:        /* The controller is up so see if the drive is there */
        !           393:        if(0 == (mp = tmscpgetcp(um)))
        !           394:                {
        !           395:                printf("tmscp can't get command packet\n");
        !           396:                return(0);
        !           397:                }
        !           398:        /* Need to determine the drive type for generic driver */
        !           399:        mp->mscp_opcode = M_OP_GTUNT;   /* This should give us the device type */
        !           400:        mp->mscp_unit = ui->ui_slave;
        !           401:        mp->mscp_cmdref = (long) ui->ui_slave;
        !           402:        tms->tms_status = 0;    /* set to zero */
        !           403:        tmscpip[ui->ui_ctlr][ui->ui_slave] = ui;
        !           404:        *((long *) mp->mscp_dscptr ) |= TMSCP_OWN | TMSCP_INT;/* maybe we should poll*/
        !           405:        i = tmscpaddr->tmscpip;
        !           406: #ifdef lint
        !           407:        i = i;
        !           408: #endif
        !           409:        while(!tms->tms_status)
        !           410:                ;                               /* Wait for some status */
        !           411: #      ifdef DEBUG
        !           412:        printd("tmscpslave: status = %o\n",tms->tms_status & M_ST_MASK);
        !           413: #      endif
        !           414:        tmscpip[ui->ui_ctlr][ui->ui_slave] = 0;
        !           415:        if(!tms->tms_type)                      /* packet from a GTUNT */
        !           416:                return(0);                      /* Failed No such drive */
        !           417:        else
        !           418:                return(1);                      /* Got it and it is there */
        !           419: }
        !           420: 
        !           421: 
        !           422: /* 
        !           423:  * Set ui flags to zero to show device is not online & set tmscpip.
        !           424:  * Unit to Controller mapping is set up here.
        !           425:  * Open routine will issue the online command, later.
        !           426:  */
        !           427: tmscpattach (ui)
        !           428:        register struct uba_device *ui;         /* ptr to unibus dev struct */
        !           429: {
        !           430: 
        !           431:        ui->ui_flags = 0;
        !           432:        tmscpip[ui->ui_ctlr][ui->ui_slave] = ui;
        !           433: #      ifdef DEBUG
        !           434:        /* 
        !           435:         * Check to see if the drive is available.
        !           436:         * If not then just print debug.
        !           437:         */
        !           438:        if(tms_info[ui->ui_unit].tms_status != M_ST_AVLBL)
        !           439:                printd("tmscpattach: unavailable \n");
        !           440: #      endif   
        !           441:        utoctlr[ui->ui_unit] = ui->ui_ctlr;
        !           442: }
        !           443: 
        !           444: 
        !           445: /*
        !           446:  * TMSCP interrupt routine.
        !           447:  */
        !           448: tmscpintr (d)
        !           449:        int d;          /* index to the controller */
        !           450: {
        !           451:        register struct uba_ctlr *um = tmscpminfo[d];
        !           452:        register struct tmscpdevice *tmscpaddr = (struct tmscpdevice *)um->um_addr;
        !           453:        struct buf *bp;
        !           454:        register int i;
        !           455:        register struct tmscp_softc *sc = &tmscp_softc[d];
        !           456:        register struct tmscp *tm = &tmscp[d];
        !           457:        struct tmscp *ttm;
        !           458:        struct mscp *mp;
        !           459: 
        !           460: #      ifdef DEBUG
        !           461:        printd10("tmscpintr: state %d, tmscpsa %o\n", sc->sc_state, tmscpaddr->tmscpsa);
        !           462: #      endif   
        !           463: 
        !           464: #ifdef QBA
        !           465:        splx(sc->sc_ipl);
        !           466: #endif
        !           467:        /*
        !           468:         * How the interrupt is handled depends on the state of the controller.
        !           469:         */
        !           470:        switch (sc->sc_state) {
        !           471: 
        !           472:        case S_IDLE:
        !           473:                printf("tmscp%d: random interrupt ignored\n", d);
        !           474:                return;
        !           475: 
        !           476:        /* Controller was in step 1 last, see if its gone to step 2 */
        !           477:        case S_STEP1:
        !           478: #              define STEP1MASK 0174377
        !           479: #              define STEP1GOOD (TMSCP_STEP2|TMSCP_IE|(NCMDL2<<3)|NRSPL2)
        !           480:                for (i = 0; i < 150; i++)
        !           481:                        {
        !           482:                        if ((tmscpaddr->tmscpsa&STEP1MASK) != STEP1GOOD)
        !           483:                                { /* still in step 1 (wait 1/100 sec) */
        !           484:                                DELAY(10000);
        !           485: #                              ifdef DEBUG
        !           486:                                printd("still in step 1, delaying\n");
        !           487: #                              endif DEBUG
        !           488:                                }
        !           489:                        else
        !           490:                                break;
        !           491:                        }
        !           492:                if (i > 149)
        !           493:                        {
        !           494:                        sc->sc_state = S_IDLE;
        !           495:                        printf("failed to initialize, in step1: sa 0x%x", tmscpaddr->tmscpsa);
        !           496:                        wakeup((caddr_t)um);
        !           497:                        return;
        !           498:                        }
        !           499:                tmscpaddr->tmscpsa = ((int)&sc->sc_tmscp->tmscp_ca.ca_ringbase)
        !           500:                        | ((cpu == VAX_780 || cpu == VAX_8600) ? TMSCP_PI : 0);
        !           501:                sc->sc_state = S_STEP2;
        !           502:                return;
        !           503: 
        !           504:        /* Controller was in step 2 last, see if its gone to step 3 */
        !           505:        case S_STEP2:
        !           506: #              define STEP2MASK 0174377
        !           507: #              define STEP2GOOD (TMSCP_STEP3|TMSCP_IE|(sc->sc_ivec/4))
        !           508:                for (i = 0; i < 150; i++)
        !           509:                        {
        !           510:                        if ((tmscpaddr->tmscpsa&STEP2MASK) != STEP2GOOD)
        !           511:                                { /* still in step 2 (wait 1/100 sec) */
        !           512:                                DELAY(10000);
        !           513: #                              ifdef DEBUG
        !           514:                                printd("still in step 2, delaying\n");
        !           515: #                              endif DEBUG
        !           516:                                }
        !           517:                        else
        !           518:                                break;
        !           519:                        }
        !           520:                if (i > 149)
        !           521:                        {
        !           522:                        sc->sc_state = S_IDLE;
        !           523:                        printf("failed to initialize, in step2: sa 0x%x", tmscpaddr->tmscpsa);
        !           524:                        wakeup((caddr_t)um);
        !           525:                        return;
        !           526:                        }
        !           527:                tmscpaddr->tmscpsa = ((int)&sc->sc_tmscp->tmscp_ca.ca_ringbase)>>16;
        !           528:                sc->sc_state = S_STEP3;
        !           529:                return;
        !           530: 
        !           531:        /* Controller was in step 3 last, see if its gone to step 4 */
        !           532:        case S_STEP3:
        !           533: #              define STEP3MASK 0174000
        !           534: #              define STEP3GOOD TMSCP_STEP4
        !           535:                for (i = 0; i < 150; i++)
        !           536:                        {
        !           537:                        if ((tmscpaddr->tmscpsa&STEP3MASK) != STEP3GOOD)
        !           538:                                { /* still in step 3 (wait 1/100 sec) */
        !           539:                                DELAY(10000);
        !           540: #                              ifdef DEBUG
        !           541:                                printd("still in step 3, delaying\n");
        !           542: #                              endif DEBUG
        !           543:                                }
        !           544:                        else
        !           545:                                break;
        !           546:                        }
        !           547:                if (i > 149)
        !           548:                        {
        !           549:                        sc->sc_state = S_IDLE;
        !           550:                        printf("failed to initialize, in step3: sa 0x%x", tmscpaddr->tmscpsa);
        !           551:                        wakeup((caddr_t)um);
        !           552:                        return;
        !           553:                        }
        !           554:                /*
        !           555:                 * Get microcode version and model number of controller;
        !           556:                 * Signal initialization complete (_GO) (to the controller);
        !           557:                 *    ask for Last Fail response if tmscperror is set;
        !           558:                 * Set state to "set controller characteristics".
        !           559:                 */
        !           560:                tmscpmicro[d] = tmscpaddr->tmscpsa;
        !           561:                tmscpaddr->tmscpsa = TMSCP_GO | (tmscperror? TMSCP_LF : 0);
        !           562:                sc->sc_state = S_SCHAR;
        !           563: #              ifdef DEBUG
        !           564:                printd("tmscpintr: completed state %d \n", sc->sc_state);
        !           565:                printd("tmscp%d Version %d model %d\n",d,tmscpmicro[d]&0xF,
        !           566:                        (tmscpmicro[d]>>4) & 0xF);
        !           567: #              endif
        !           568: 
        !           569:            /*
        !           570:             * Initialize the data structures (response and command queues).
        !           571:             */
        !           572:            ttm = sc->sc_tmscp;
        !           573:            for (i = 0; i < NRSP; i++)
        !           574:                    {
        !           575:                    tm->tmscp_ca.ca_rspdsc[i] = TMSCP_OWN | TMSCP_INT | 
        !           576:                           (long)&ttm->tmscp_rsp[i].mscp_cmdref;
        !           577:                    tm->tmscp_rsp[i].mscp_dscptr = &tm->tmscp_ca.ca_rspdsc[i];
        !           578:                    tm->tmscp_rsp[i].mscp_header.tmscp_msglen = mscp_msglen;
        !           579:                    }
        !           580:            for (i = 0; i < NCMD; i++)
        !           581:                    {
        !           582:                    tm->tmscp_ca.ca_cmddsc[i] = TMSCP_INT | 
        !           583:                                        (long)&ttm->tmscp_cmd[i].mscp_cmdref;
        !           584:                    tm->tmscp_cmd[i].mscp_dscptr = &tm->tmscp_ca.ca_cmddsc[i];
        !           585:                    tm->tmscp_cmd[i].mscp_header.tmscp_msglen = mscp_msglen;
        !           586:                    tm->tmscp_cmd[i].mscp_header.tmscp_vcid = 1;
        !           587:                    }
        !           588:            bp = &tmscpwtab[d];
        !           589:            bp->av_forw = bp->av_back = bp;
        !           590:            sc->sc_lastcmd = 1;
        !           591:            sc->sc_lastrsp = 0;
        !           592:            mp = &tmscp[um->um_ctlr].tmscp_cmd[0];
        !           593:            mp->mscp_unit = mp->mscp_modifier = 0;
        !           594:            mp->mscp_flags = 0;
        !           595:            mp->mscp_version = 0;
        !           596:            mp->mscp_cntflgs = M_CF_ATTN|M_CF_MISC|M_CF_THIS;
        !           597:            /*
        !           598:             * A host time out value of 0 means that the controller will not
        !           599:             * time out.  This is ok for the TK50.
        !           600:             */
        !           601:            mp->mscp_hsttmo = 0;
        !           602:            mp->mscp_time.val[0] = 0;
        !           603:            mp->mscp_time.val[1] = 0;
        !           604:            mp->mscp_cntdep = 0;
        !           605:            mp->mscp_opcode = M_OP_STCON;
        !           606:            *((long *)mp->mscp_dscptr) |= TMSCP_OWN|TMSCP_INT;
        !           607:            i = tmscpaddr->tmscpip;      /* initiate polling */
        !           608:            return;
        !           609: 
        !           610:        case S_SCHAR:
        !           611:        case S_RUN:
        !           612:                break;
        !           613: 
        !           614:        default:
        !           615:            printf("tmscp%d: interrupt in unknown state %d ignored\n",d,sc->sc_state);
        !           616:            return;
        !           617:        }       /* end switch */
        !           618: 
        !           619:        /*
        !           620:         * The controller state is S_SCHAR or S_RUN
        !           621:         */
        !           622: 
        !           623:        /*
        !           624:         * If the error bit is set in the SA register then print an error
        !           625:         * message and reinitialize the controller.
        !           626:         */
        !           627:        if (tmscpaddr->tmscpsa&TMSCP_ERR)
        !           628:                {
        !           629:                printf("tmscp%d: fatal error (%o)\n", d, tmscpaddr->tmscpsa&0xffff);
        !           630:                tmscpaddr->tmscpip = 0;
        !           631:                wakeup((caddr_t)um);
        !           632:                }
        !           633:        /*
        !           634:         * Check for a buffer purge request. (Won't happen w/ TK50 on Q22 bus)
        !           635:         */
        !           636:        if (tm->tmscp_ca.ca_bdp)
        !           637:                {
        !           638:                UBAPURGE(um->um_hd->uh_uba, tm->tmscp_ca.ca_bdp);
        !           639:                tm->tmscp_ca.ca_bdp = 0;
        !           640:                tmscpaddr->tmscpsa = 0;      /* signal purge complete */
        !           641:                }
        !           642: 
        !           643:        /*
        !           644:         * Check for response ring transition.
        !           645:         */
        !           646:        if (tm->tmscp_ca.ca_rspint)
        !           647:                {
        !           648:                tm->tmscp_ca.ca_rspint = 0;
        !           649:                for (i = sc->sc_lastrsp;; i++)
        !           650:                        {
        !           651:                        i %= NRSP;
        !           652:                        if (tm->tmscp_ca.ca_rspdsc[i]&TMSCP_OWN)
        !           653:                            break;
        !           654:                        tmscprsp(um, tm, sc, i);
        !           655:                        tm->tmscp_ca.ca_rspdsc[i] |= TMSCP_OWN;
        !           656:                        }
        !           657:                sc->sc_lastrsp = i;
        !           658:                }
        !           659: 
        !           660:        /*
        !           661:         * Check for command ring transition.
        !           662:         */
        !           663:        if (tm->tmscp_ca.ca_cmdint)
        !           664:                {
        !           665: #              ifdef DEBUG
        !           666:                printd("tmscpintr: command ring transition\n");
        !           667: #              endif           
        !           668:                tm->tmscp_ca.ca_cmdint = 0;
        !           669:                }
        !           670:        if(tmscp_cp_wait)
        !           671:                wakeup((caddr_t)&tmscp_cp_wait);
        !           672:        (void) tmscpstart(um);
        !           673: }
        !           674: 
        !           675: 
        !           676: /*
        !           677:  * Open a tmscp device and set the unit online.  If the controller is not 
        !           678:  * in the run state, call init to initialize the tmscp controller first.
        !           679:  */
        !           680: 
        !           681: /* ARGSUSED */
        !           682: tmscpopen(dev, flag)
        !           683:        dev_t dev;
        !           684:        int flag;
        !           685: {
        !           686:        register int unit;
        !           687:        register struct uba_device *ui;
        !           688:        register struct tmscp_softc *sc;
        !           689:        register struct tms_info *tms;
        !           690:        register struct mscp *mp;
        !           691:        register struct uba_ctlr *um;
        !           692:        struct tmscpdevice *tmscpaddr;
        !           693:        int s,i;
        !           694:        
        !           695:        unit = TMSUNIT(dev);
        !           696: #      ifdef DEBUG
        !           697:        printd("tmscpopen unit %d\n",unit);
        !           698:        if(tmscpdebug)DELAY(10000);
        !           699: #      endif
        !           700:        if (unit >= NTMS || (ui = tmsdinfo[unit]) == 0 || ui->ui_alive == 0)
        !           701:                return (ENXIO);
        !           702:        tms = &tms_info[ui->ui_unit];
        !           703:        if (tms->tms_openf)
        !           704:                return (EBUSY);
        !           705:        sc = &tmscp_softc[ui->ui_ctlr];
        !           706:        tms->tms_openf = 1;
        !           707:        tms->tms_tpr = tprintf_open();
        !           708:        s = spl5();
        !           709:        if (sc->sc_state != S_RUN)
        !           710:                {
        !           711:                if (sc->sc_state == S_IDLE)
        !           712:                        if(!tmscpinit(ui->ui_ctlr))
        !           713:                                {
        !           714:                                printf("tmscp controller failed to init\n");
        !           715:                                (void) splx(s);
        !           716:                                tms->tms_openf = 0;
        !           717:                                return(ENXIO);
        !           718:                                }
        !           719:                /* 
        !           720:                 * Wait for initialization to complete
        !           721:                 */
        !           722:                timeout(wakeup,(caddr_t)ui->ui_mi,11*hz);       /* to be sure*/
        !           723:                sleep((caddr_t)ui->ui_mi, 0);
        !           724:                if (sc->sc_state != S_RUN)
        !           725:                        {
        !           726:                        (void) splx(s);
        !           727:                        tms->tms_openf = 0;
        !           728:                        return (EIO);
        !           729:                        }
        !           730:                }
        !           731:        /*
        !           732:         * Check to see if the device is really there.
        !           733:         * this code was taken from Fred Canters 11 driver
        !           734:         */
        !           735:        um = ui->ui_mi;
        !           736:        tmscpaddr = (struct tmscpdevice *) um->um_addr;
        !           737:        (void) splx(s);
        !           738:        if(ui->ui_flags == 0)
        !           739:                {
        !           740:                s = spl5();
        !           741:                while(0 ==(mp = tmscpgetcp(um)))
        !           742:                        {
        !           743:                        tmscp_cp_wait++;
        !           744:                        sleep((caddr_t)&tmscp_cp_wait,PSWP+1);
        !           745:                        tmscp_cp_wait--;
        !           746:                        }
        !           747:                (void) splx(s);
        !           748:                mp->mscp_opcode = M_OP_ONLIN;
        !           749:                mp->mscp_unit = ui->ui_slave;
        !           750:                mp->mscp_cmdref = (long) & tms->tms_type;
        !           751:                                            /* need to sleep on something */
        !           752: #              ifdef DEBUG
        !           753:                printd("tmscpopen: bring unit %d online\n",ui->ui_unit);
        !           754: #              endif   
        !           755:                *((long *) mp->mscp_dscptr ) |= TMSCP_OWN | TMSCP_INT;
        !           756:                i = tmscpaddr->tmscpip;
        !           757: #ifdef lint
        !           758:                i = i;
        !           759: #endif
        !           760:                /* 
        !           761:                 * To make sure we wake up, timeout in 240 seconds.
        !           762:                 * Wakeup in tmscprsp routine.
        !           763:                 * 240 seconds (4 minutes) is necessary since a rewind
        !           764:                 * can take a few minutes.
        !           765:                 */
        !           766:                timeout(wakeup,(caddr_t) mp->mscp_cmdref,240 * hz);
        !           767:                sleep((caddr_t) mp->mscp_cmdref,PSWP+1);
        !           768:                }
        !           769:        if(ui->ui_flags == 0) {
        !           770:                tms->tms_openf = 0;
        !           771:                return(ENXIO);  /* Didn't go online */
        !           772:        }
        !           773:        tms->tms_lastiow = 0;
        !           774:        /*
        !           775:         * If the high density device is not specified, set unit to low
        !           776:         * density.  This is done as an "internal" ioctl command so
        !           777:         * that the command setup and response handling
        !           778:         * is done thru "regular" command routines.
        !           779:         */
        !           780:        if ((minor(dev) & T_HIDENSITY) == 0)
        !           781:                tmscpcommand(dev, TMS_LOWDENSITY, 1);
        !           782:        else
        !           783:                tmscpcommand(dev, TMS_HIDENSITY, 1);
        !           784:        return (0);
        !           785: }
        !           786: 
        !           787: 
        !           788: /*
        !           789:  * Close tape device.
        !           790:  *
        !           791:  * If tape was open for writing or last operation was
        !           792:  * a write, then write two EOF's and backspace over the last one.
        !           793:  * Unless this is a non-rewinding special file, rewind the tape.
        !           794:  *
        !           795:  * NOTE:
        !           796:  *     We want to be sure that any serious exception is cleared on the
        !           797:  *     close. A Clear Serious Exception (CSE) modifier is always done on
        !           798:  *     the rewind command.  For the non-rewind case we check to see if the
        !           799:  *     "serex" field is set in the softc struct; if it is then issue a noop
        !           800:  *     command with the CSE modifier.
        !           801:  * Make the tape available to others, by clearing openf flag.
        !           802:  */
        !           803: tmscpclose(dev, flag)
        !           804:        register dev_t dev;
        !           805:        register flag;
        !           806: {
        !           807:        register struct tms_info *tms;
        !           808:        register struct uba_device *ui;
        !           809: 
        !           810:        ui = tmsdinfo[TMSUNIT(dev)];
        !           811: #      ifdef DEBUG
        !           812:        printd("tmscpclose: ctlr =  %d\n",TMSCPCTLR(dev));
        !           813:        printd("tmscpclose: unit = %d\n",TMSUNIT(dev));
        !           814:        if(tmscpdebug)DELAY(10000);
        !           815: #      endif
        !           816:        tms = &tms_info[ui->ui_unit];
        !           817:        if (flag == FWRITE || (flag&FWRITE) && tms->tms_lastiow)
        !           818:                {
        !           819:                /*         device, command, count */
        !           820:                tmscpcommand (dev, TMS_WRITM, 1);
        !           821:                tmscpcommand (dev, TMS_WRITM, 1);
        !           822:                tmscpcommand (dev, TMS_BSR, 1);
        !           823:                }
        !           824:        if ((minor(dev)&T_NOREWIND) == 0)
        !           825:                /*
        !           826:                 * Don't hang waiting for rewind complete.
        !           827:                 */
        !           828:                tmscpcommand(dev, TMS_REW, 0);
        !           829:        else
        !           830:                if (tms->tms_serex)
        !           831:                        {
        !           832: #                      ifdef DEBUG
        !           833:                        printd("tmscpclose: clearing serex\n");
        !           834:                        if(tmscpdebug)DELAY(10000);
        !           835: #                      endif
        !           836:                        tmscpcommand(dev, TMS_CSE, 1);
        !           837:                        }
        !           838:        tprintf_close(tms->tms_tpr);
        !           839:        tms->tms_openf = 0;
        !           840:        return (0);
        !           841: }
        !           842: 
        !           843: 
        !           844: /*
        !           845:  * Execute a command on the tape drive a specified number of times.
        !           846:  * This routine sets up a buffer and calls the strategy routine which
        !           847:  * links the buffer onto the drive's buffer queue.
        !           848:  * The start routine will take care of creating a tmscp command packet
        !           849:  * with the command.  The start routine is called by the strategy or the
        !           850:  * interrupt routine.
        !           851:  */
        !           852: 
        !           853: tmscpcommand (dev, com, count)
        !           854:        register dev_t dev;
        !           855:        int com, count;
        !           856: {
        !           857:        register struct uba_device *ui;
        !           858:        register struct buf *bp;
        !           859:        register int s;
        !           860:        int unit = TMSUNIT(dev);
        !           861: 
        !           862:        ui = tmsdinfo[unit];
        !           863:        bp = &ctmscpbuf[ui->ui_ctlr];
        !           864: 
        !           865:        s = spl5();
        !           866:        while (bp->b_flags&B_BUSY)
        !           867:                {
        !           868:                /*
        !           869:                 * This special check is because B_BUSY never
        !           870:                 * gets cleared in the non-waiting rewind case.
        !           871:                 */
        !           872:                if (bp->b_bcount == 0 && (bp->b_flags&B_DONE))
        !           873:                        break;
        !           874:                bp->b_flags |= B_WANTED;
        !           875:                sleep((caddr_t)bp, PRIBIO);
        !           876:                }
        !           877:        bp->b_flags = B_BUSY|B_READ;
        !           878:        splx(s);
        !           879:        /*
        !           880:         * Load the buffer.  The b_count field gets used to hold the command
        !           881:         * count.  the b_resid field gets used to hold the command mneumonic.
        !           882:         * These 2 fields are "known" to be "safe" to use for this purpose.
        !           883:         * (Most other drivers also use these fields in this way.)
        !           884:         */
        !           885:        bp->b_dev = dev;
        !           886:        bp->b_bcount = count;
        !           887:        bp->b_resid = com;
        !           888:        bp->b_blkno = 0;
        !           889:        tmscpstrategy(bp);
        !           890:        /*
        !           891:         * In case of rewind from close, don't wait.
        !           892:         * This is the only case where count can be 0.
        !           893:         */
        !           894:        if (count == 0)
        !           895:                return;
        !           896:        iowait(bp);
        !           897:        if (bp->b_flags&B_WANTED)
        !           898:                wakeup((caddr_t)bp);
        !           899:        bp->b_flags &= B_ERROR;
        !           900: }
        !           901: 
        !           902: /*
        !           903:  * Find an unused command packet
        !           904:  */
        !           905: struct mscp *
        !           906: tmscpgetcp(um)
        !           907:        struct uba_ctlr *um;
        !           908: {
        !           909:        register struct mscp *mp;
        !           910:        register struct tmscpca *cp;
        !           911:        register struct tmscp_softc *sc;
        !           912:        register int i;
        !           913:        int     s;
        !           914: 
        !           915:        s = spl5();
        !           916:        cp = &tmscp[um->um_ctlr].tmscp_ca;
        !           917:        sc = &tmscp_softc[um->um_ctlr];
        !           918:        /*
        !           919:         * If no credits, can't issue any commands
        !           920:         * until some outstanding commands complete.
        !           921:         */
        !           922:        i = sc->sc_lastcmd;
        !           923: #      ifdef DEBUG
        !           924:        printd10("tmscpgetcp: %d credits remain\n", sc->sc_credits);
        !           925: #      endif
        !           926:        if(((cp->ca_cmddsc[i]&(TMSCP_OWN|TMSCP_INT))==TMSCP_INT) &&
        !           927:          (sc->sc_credits >= 2))
        !           928:                {
        !           929:                sc->sc_credits--;       /* This commits to issuing a command */
        !           930:                cp->ca_cmddsc[i] &= ~TMSCP_INT;
        !           931:                mp = &tmscp[um->um_ctlr].tmscp_cmd[i];
        !           932:                mp->mscp_unit = mp->mscp_modifier = 0;
        !           933:                mp->mscp_opcode = mp->mscp_flags = 0;
        !           934:                mp->mscp_bytecnt = mp->mscp_buffer = 0;
        !           935:                sc->sc_lastcmd = (i + 1) % NCMD;
        !           936:                (void) splx(s);
        !           937:                return(mp);
        !           938:                }
        !           939:        (void) splx(s);
        !           940:        return(NULL);
        !           941: }
        !           942: 
        !           943: 
        !           944: /*
        !           945:  * Initialize a TMSCP device.  Set up UBA mapping registers,
        !           946:  * initialize data structures, and start hardware
        !           947:  * initialization sequence.
        !           948:  */
        !           949: tmscpinit (d)
        !           950:        int d;                  /* index to the controller */
        !           951: {
        !           952:        register struct tmscp_softc *sc;
        !           953:        register struct tmscp *t;  /* communications area; cmd & resp packets */
        !           954:        struct tmscpdevice *tmscpaddr;
        !           955:        struct uba_ctlr *um;
        !           956: 
        !           957:        sc = &tmscp_softc[d];
        !           958:        um = tmscpminfo[d];
        !           959:        um->um_tab.b_active++;
        !           960:        t = &tmscp[d];
        !           961:        tmscpaddr = (struct tmscpdevice *)um->um_addr;
        !           962:        if (sc->sc_mapped == 0)
        !           963:                {
        !           964:                /*
        !           965:                 * Map the communications area and command
        !           966:                 * and response packets into Unibus address
        !           967:                 * space.
        !           968:                 */
        !           969:                sc->sc_ubainfo = uballoc(um->um_ubanum, (caddr_t)t, sizeof (struct tmscp), 0);
        !           970:                sc->sc_tmscp = (struct tmscp *)(UBAI_ADDR(sc->sc_ubainfo));
        !           971:                sc->sc_mapped = 1;
        !           972:                }
        !           973: 
        !           974:        /*
        !           975:         * Start the hardware initialization sequence.
        !           976:         */
        !           977:        tmscpaddr->tmscpip = 0;              /* start initialization */
        !           978: 
        !           979:        while((tmscpaddr->tmscpsa & TMSCP_STEP1) == 0)
        !           980:                {
        !           981: #              ifdef DEBUG
        !           982:                printd("tmscpinit: tmscpsa = 0%o\n",tmscpaddr->tmscpsa);
        !           983:                DELAY(100000);
        !           984: #              endif
        !           985:                if(tmscpaddr->tmscpsa & TMSCP_ERR)
        !           986:                        return(0);      /* CHECK */
        !           987:                }
        !           988:        tmscpaddr->tmscpsa=TMSCP_ERR|(NCMDL2<<11)|(NRSPL2<<8)|TMSCP_IE|(sc->sc_ivec/4);
        !           989:        /*
        !           990:         * Initialization continues in the interrupt routine.
        !           991:         */
        !           992:        sc->sc_state = S_STEP1;
        !           993:        sc->sc_credits = 0;
        !           994:        return(1);
        !           995: }
        !           996: 
        !           997: 
        !           998: /*
        !           999:  * Start I/O operation
        !          1000:  * This code is convoluted.  The majority of it was copied from the uda driver.
        !          1001:  */
        !          1002: 
        !          1003: tmscpstart(um)
        !          1004:        register struct uba_ctlr *um;
        !          1005: {
        !          1006:        register struct buf *bp, *dp;
        !          1007:        register struct mscp *mp;
        !          1008:        register struct tmscp_softc *sc;
        !          1009:        register struct tms_info *tms;
        !          1010:        register struct uba_device *ui;
        !          1011:        struct   tmscpdevice *tmscpaddr;
        !          1012:        struct   tmscp *tm = &tmscp[um->um_ctlr];
        !          1013:        int i,tempi;
        !          1014:        char ioctl;             /* flag: set true if its an IOCTL command */
        !          1015: 
        !          1016:        sc = &tmscp_softc[um->um_ctlr];
        !          1017:        
        !          1018:     for(;;)
        !          1019:        {
        !          1020:        if ((dp = um->um_tab.b_actf) == NULL)
        !          1021:                {
        !          1022:                /*
        !          1023:                 * Release unneeded UBA resources and return
        !          1024:                 * (drive was inactive)
        !          1025:                 */
        !          1026:                um->um_tab.b_active = 0;
        !          1027:                break;
        !          1028:                }
        !          1029:        if ((bp = dp->b_actf) == NULL)
        !          1030:                {
        !          1031:                /*
        !          1032:                 * No more requests for this drive, remove
        !          1033:                 * from controller queue and look at next drive.
        !          1034:                 * We know we're at the head of the controller queue.
        !          1035:                 */
        !          1036:                dp->b_active = 0;
        !          1037:                um->um_tab.b_actf = dp->b_forw;
        !          1038:                continue;               /* Need to check for loop */
        !          1039:                }
        !          1040:        um->um_tab.b_active++;
        !          1041:        tmscpaddr = (struct tmscpdevice *)um->um_addr;
        !          1042:        ui = tmsdinfo[(TMSUNIT(bp->b_dev))];
        !          1043:        tms = &tms_info[ui->ui_unit];
        !          1044:        if ((tmscpaddr->tmscpsa&TMSCP_ERR) || sc->sc_state != S_RUN)
        !          1045:                {
        !          1046:                tprintf(tms->tms_tpr,
        !          1047:                    "tms%d: hard error bn%d\n",
        !          1048:                    minor(bp->b_dev)&03, bp->b_blkno);
        !          1049:                log(TMS_PRI, "tmscp%d: sa 0%o, state %d\n",um->um_ctlr,
        !          1050:                                tmscpaddr->tmscpsa&0xffff, sc->sc_state);
        !          1051:                (void)tmscpinit(um->um_ctlr);
        !          1052:                /* SHOULD REQUEUE OUTSTANDING REQUESTS, LIKE TMSCPRESET */
        !          1053:                break;
        !          1054:                }
        !          1055:        /*
        !          1056:         * Default is that last command was NOT a write command;
        !          1057:         * if a write command is done it will be detected in tmscprsp.
        !          1058:         */
        !          1059:        tms->tms_lastiow = 0;
        !          1060:        if (ui->ui_flags == 0)
        !          1061:                {        /* not online */
        !          1062:                if ((mp = tmscpgetcp(um)) == NULL)
        !          1063:                        break;
        !          1064:                mp->mscp_opcode = M_OP_ONLIN;
        !          1065:                mp->mscp_unit = ui->ui_slave;
        !          1066:                dp->b_active = 2;
        !          1067:                um->um_tab.b_actf = dp->b_forw; /* remove from controller q */
        !          1068:                *((long *)mp->mscp_dscptr) |= TMSCP_OWN|TMSCP_INT;
        !          1069:                if (tmscpaddr->tmscpsa&TMSCP_ERR)
        !          1070:                        printf("tmscp%d fatal error (0%o)\n",um->um_ctlr,
        !          1071:                                        tmscpaddr->tmscpsa&0xffff);
        !          1072:                i = tmscpaddr->tmscpip;
        !          1073:                continue;
        !          1074:                }
        !          1075:        switch (cpu) {
        !          1076: 
        !          1077:        case VAX_8600:
        !          1078:        case VAX_780:
        !          1079:                i = UBA_NEEDBDP|UBA_CANTWAIT;
        !          1080:                break;
        !          1081:        case VAX_750:
        !          1082:                i = um->um_ubinfo|UBA_HAVEBDP|UBA_CANTWAIT;
        !          1083:                break;
        !          1084:        case VAX_730:
        !          1085:        case VAX_630:
        !          1086:                i = UBA_CANTWAIT;
        !          1087:                break;
        !          1088:        }   /* end switch (cpu) */
        !          1089:        /*
        !          1090:         * If command is an ioctl command then set the ioctl flag for later use.
        !          1091:         * If not (i.e. it is a read or write) then attempt
        !          1092:         * to set up a buffer pointer.
        !          1093:         */
        !          1094:        ioctl = 0;
        !          1095:        if (bp == &ctmscpbuf[um->um_ctlr])
        !          1096:                ioctl = 1;
        !          1097:        else
        !          1098:                if ((i = ubasetup(um->um_ubanum, bp, i)) == 0)
        !          1099:                        {
        !          1100:                        if(dp->b_qsize != 0)
        !          1101:                                break; /* When a command completes and */
        !          1102:                                     /* frees a bdp tmscpstart will be called */
        !          1103:                        if ((mp = tmscpgetcp(um)) == NULL)
        !          1104:                                break;
        !          1105: #                      ifdef DEBUG
        !          1106:                        printd("tmscpstart: GTUNT %d ubasetup = %d\n",ui->ui_unit, i);
        !          1107:                        if(tmscpdebug)DELAY(10000);
        !          1108: #                      endif
        !          1109:                        mp->mscp_opcode = M_OP_GTUNT;
        !          1110:                        mp->mscp_unit = ui->ui_slave;
        !          1111:                        *((long *)mp->mscp_dscptr) |= TMSCP_OWN|TMSCP_INT;
        !          1112:                        if (tmscpaddr->tmscpsa&TMSCP_ERR)
        !          1113:                                printf("tmscp%d: fatal error (0%o)\n",um->um_ctlr,
        !          1114:                                            tmscpaddr->tmscpsa&0xffff);
        !          1115:                        i = tmscpaddr->tmscpip; /* initiate polling */
        !          1116:                        break;
        !          1117:                        }
        !          1118: #      if defined(VAX750)
        !          1119:        if (cpu == VAX_750)
        !          1120:                tempi = i & 0xfffffff;                  /* mask off bdp */
        !          1121:        else
        !          1122: #      endif
        !          1123:                tempi = i;
        !          1124:        if ((mp = tmscpgetcp(um)) == NULL)
        !          1125:                {
        !          1126:                if (!ioctl)             /* only need to release if NOT ioctl */
        !          1127:                        ubarelse(um->um_ubanum,&tempi);
        !          1128:                break;
        !          1129:                }
        !          1130:        mp->mscp_cmdref = (long)bp;             /* pointer to get back */
        !          1131:        mp->mscp_unit = ui->ui_slave;
        !          1132:        /*
        !          1133:         * If its an ioctl-type command then set up the appropriate
        !          1134:         * tmscp command;  by doing a switch on the "b_resid" field where
        !          1135:         * the command mneumonic is stored.
        !          1136:         */
        !          1137:        if (ioctl)
        !          1138:                {
        !          1139: #              ifdef DEBUG
        !          1140:                printd("tmscpstart: doing ioctl cmd %d\n", bp->b_resid);
        !          1141: #              endif
        !          1142:                /*
        !          1143:                 * The reccnt and tmkcnt fields are set to zero by the getcp
        !          1144:                 * routine (as bytecnt and buffer fields).  Thus reccnt and
        !          1145:                 * tmkcnt are only modified here if they need to be set to
        !          1146:                 * a non-zero value.
        !          1147:                 */
        !          1148:                switch ((int)bp->b_resid) {
        !          1149: 
        !          1150:                case TMS_WRITM:
        !          1151:                        mp->mscp_opcode = M_OP_WRITM;
        !          1152:                        break;
        !          1153:                case TMS_FSF:
        !          1154:                        mp->mscp_opcode = M_OP_REPOS;
        !          1155:                        mp->mscp_tmkcnt = bp->b_bcount;
        !          1156:                        break;
        !          1157:                case TMS_BSF:
        !          1158:                        mp->mscp_opcode = M_OP_REPOS;
        !          1159:                        mp->mscp_modifier = M_MD_REVRS;
        !          1160:                        mp->mscp_tmkcnt = bp->b_bcount;
        !          1161:                        break;
        !          1162:                case TMS_FSR:
        !          1163:                        mp->mscp_opcode = M_OP_REPOS;
        !          1164:                        mp->mscp_modifier = M_MD_OBJCT;
        !          1165:                        mp->mscp_reccnt = bp->b_bcount;
        !          1166:                        break;
        !          1167:                case TMS_BSR:
        !          1168:                        mp->mscp_opcode = M_OP_REPOS;
        !          1169:                        mp->mscp_modifier = M_MD_REVRS | M_MD_OBJCT;
        !          1170:                        mp->mscp_reccnt = bp->b_bcount;
        !          1171:                        break;
        !          1172:                /*
        !          1173:                 * Clear serious exception is done for Rewind & Available cmds
        !          1174:                 */
        !          1175:                case TMS_REW:
        !          1176:                        mp->mscp_opcode = M_OP_REPOS;
        !          1177:                        mp->mscp_modifier = M_MD_REWND | M_MD_CLSEX;
        !          1178:                        if (bp->b_bcount == 0)
        !          1179:                                mp->mscp_modifier |= M_MD_IMMED;
        !          1180:                        tms->tms_serex = 0;
        !          1181:                        break;
        !          1182:                case TMS_OFFL:
        !          1183:                        mp->mscp_opcode = M_OP_AVAIL;
        !          1184:                        mp->mscp_modifier = M_MD_UNLOD | M_MD_CLSEX;
        !          1185:                        tms->tms_serex = 0;
        !          1186:                        break;
        !          1187:                case TMS_SENSE:
        !          1188:                        mp->mscp_opcode = M_OP_GTUNT;
        !          1189:                        break;
        !          1190:                case TMS_CACHE:
        !          1191:                        mp->mscp_opcode = M_OP_STUNT;
        !          1192:                        tms->tms_unitflgs |= M_UF_WBKNV;
        !          1193:                        mp->mscp_unitflgs = tms->tms_unitflgs;
        !          1194:                        mp->mscp_format = tms->tms_format;
        !          1195:                        /* default device dependant parameters */
        !          1196:                        mp->mscp_mediaid = 0;
        !          1197:                        break;
        !          1198:                case TMS_NOCACHE:
        !          1199:                        mp->mscp_opcode = M_OP_STUNT;
        !          1200:                        tms->tms_unitflgs &= ~(M_UF_WBKNV);
        !          1201:                        mp->mscp_unitflgs = tms->tms_unitflgs;
        !          1202:                        mp->mscp_format = tms->tms_format;
        !          1203:                        /* default device dependant parameters */
        !          1204:                        mp->mscp_mediaid = 0;
        !          1205:                        break;
        !          1206:                case TMS_CSE:
        !          1207:                        /*
        !          1208:                         * This is a no-op command. It performs a 
        !          1209:                         * clear serious exception only.  (Done on a
        !          1210:                         * non-rewinding close after a serious exception.)
        !          1211:                         */
        !          1212:                        mp->mscp_opcode = M_OP_REPOS;
        !          1213:                        mp->mscp_modifier = M_MD_CLSEX;
        !          1214:                        tms->tms_serex = 0;
        !          1215:                        tms->tms_clserex = 1;
        !          1216:                        break;
        !          1217:                case TMS_LOWDENSITY:
        !          1218:                        /*
        !          1219:                         * Set the unit to low density
        !          1220:                         */
        !          1221:                        mp->mscp_opcode = M_OP_STUNT;
        !          1222:                        mp->mscp_unitflgs = tms->tms_unitflgs;
        !          1223:                        mp->mscp_mediaid = 0;   /* default device dependant parameters */
        !          1224:                        if ((tms->tms_fmtmenu & M_TF_800) != 0)
        !          1225:                                mp->mscp_format = M_TF_800;
        !          1226:                        else
        !          1227:                                mp->mscp_format = M_TF_PE & tms->tms_fmtmenu;
        !          1228:                        tms->tms_format = mp->mscp_format;
        !          1229:                        break;
        !          1230:                case TMS_HIDENSITY:
        !          1231:                        /*
        !          1232:                         * Set the unit to high density (format == 0)
        !          1233:                         */
        !          1234:                        mp->mscp_opcode = M_OP_STUNT;
        !          1235:                        mp->mscp_unitflgs = tms->tms_unitflgs;
        !          1236:                        mp->mscp_mediaid = 0;   /* default device dependant parameters */
        !          1237:                        mp->mscp_format = 0;
        !          1238:                        tms->tms_format = 0;
        !          1239:                        break;
        !          1240:                default:
        !          1241:                        printf("Bad ioctl on tms unit %d\n", ui->ui_unit);
        !          1242:                        /* Need a no-op. Reposition no amount */
        !          1243:                        mp->mscp_opcode = M_OP_REPOS;
        !          1244:                        break;
        !          1245:                }   /* end switch (bp->b_resid) */
        !          1246:                }
        !          1247:        else    /* Its a read/write command (not an ioctl) */
        !          1248:                {
        !          1249:                mp->mscp_opcode = bp->b_flags&B_READ ? M_OP_READ : M_OP_WRITE;
        !          1250:                mp->mscp_bytecnt = bp->b_bcount;
        !          1251:                mp->mscp_buffer = UBAI_ADDR(i) | (UBAI_BDP(i) << 24);
        !          1252: 
        !          1253:                bp->b_ubinfo = tempi;                   /* save mapping info */
        !          1254:                }
        !          1255:        if (tms->tms_serex == 2)                        /* if tape mark read */
        !          1256:                {
        !          1257:                mp->mscp_modifier |= M_MD_CLSEX;        /*  clear serious exc */
        !          1258:                tms->tms_serex = 0;
        !          1259:                }
        !          1260:        *((long *)mp->mscp_dscptr) |= TMSCP_OWN|TMSCP_INT;
        !          1261: #      ifdef DEBUG
        !          1262:        printd("tmscpstart: opcode 0%o mod %o unit %d cnt %d\n",mp->mscp_opcode,mp->mscp_modifier,mp->mscp_unit,mp->mscp_bytecnt);
        !          1263:        if(tmscpdebug)DELAY(100000);
        !          1264: #      endif
        !          1265:        i = tmscpaddr->tmscpip;              /* initiate polling */
        !          1266:        dp->b_qsize++;
        !          1267:        /*
        !          1268:         * Move drive to the end of the controller queue
        !          1269:         */
        !          1270:        if (dp->b_forw != NULL)
        !          1271:                {
        !          1272:                um->um_tab.b_actf = dp->b_forw;
        !          1273:                um->um_tab.b_actl->b_forw = dp;
        !          1274:                um->um_tab.b_actl = dp;
        !          1275:                dp->b_forw = NULL;
        !          1276:                }
        !          1277:        /*
        !          1278:         * Move buffer to I/O wait queue
        !          1279:         */
        !          1280:        dp->b_actf = bp->av_forw;
        !          1281:        dp = &tmscpwtab[um->um_ctlr];
        !          1282:        bp->av_forw = dp;
        !          1283:        bp->av_back = dp->av_back;
        !          1284:        dp->av_back->av_forw = bp;
        !          1285:        dp->av_back = bp;
        !          1286:        if (tmscpaddr->tmscpsa&TMSCP_ERR)
        !          1287:                {
        !          1288:                printf("tmscp%d: fatal error (0%o)\n", um->um_ctlr, tmscpaddr->tmscpsa&0xffff);
        !          1289:                (void)tmscpinit(um->um_ctlr);
        !          1290:                break;
        !          1291:                }
        !          1292:     }   /* end for */
        !          1293:     /*
        !          1294:      * Check for response ring transitions lost in the
        !          1295:      * Race condition
        !          1296:      */
        !          1297:     for (i = sc->sc_lastrsp;; i++)
        !          1298:            {
        !          1299:            i %= NRSP;
        !          1300:            if (tm->tmscp_ca.ca_rspdsc[i]&TMSCP_OWN)
        !          1301:                    break;
        !          1302:            tmscprsp(um, tm, sc, i);
        !          1303:            tm->tmscp_ca.ca_rspdsc[i] |= TMSCP_OWN;
        !          1304:            }
        !          1305:     sc->sc_lastrsp = i;
        !          1306: }
        !          1307: 
        !          1308: 
        !          1309: /*
        !          1310:  * Process a response packet
        !          1311:  */
        !          1312: tmscprsp(um, tm, sc, i)
        !          1313:        register struct uba_ctlr *um;
        !          1314:        register struct tmscp *tm;
        !          1315:        register struct tmscp_softc *sc;
        !          1316:        int i;
        !          1317: {
        !          1318:        register struct mscp *mp;
        !          1319:        register struct tms_info *tms;
        !          1320:        struct uba_device *ui;
        !          1321:        struct buf *dp, *bp;
        !          1322:        int st;
        !          1323: 
        !          1324:        mp = &tm->tmscp_rsp[i];
        !          1325:        mp->mscp_header.tmscp_msglen = mscp_msglen;
        !          1326:        sc->sc_credits += mp->mscp_header.tmscp_credits & 0xf;  /* low 4 bits */
        !          1327:        if ((mp->mscp_header.tmscp_credits & 0xf0) > 0x10)      /* Check */
        !          1328:                return;
        !          1329: #      ifdef DEBUG
        !          1330:        printd("tmscprsp, opcode 0%o status 0%o\n",mp->mscp_opcode,mp->mscp_status&M_ST_MASK);
        !          1331: #      endif   
        !          1332:        /*
        !          1333:         * If it's an error log message (datagram),
        !          1334:         * pass it on for more extensive processing.
        !          1335:         */
        !          1336:        if ((mp->mscp_header.tmscp_credits & 0xf0) == 0x10)
        !          1337:                {       /* check */
        !          1338:                tmserror(um, (struct mslg *)mp);
        !          1339:                return;
        !          1340:                }
        !          1341:        st = mp->mscp_status&M_ST_MASK;
        !          1342:        /*
        !          1343:         * The controller interrupts as drive 0.
        !          1344:         * This means that you must check for controller interrupts
        !          1345:         * before you check to see if there is a drive 0.
        !          1346:         */
        !          1347:        if((M_OP_STCON|M_OP_END) == mp->mscp_opcode)
        !          1348:                {
        !          1349:                if (st == M_ST_SUCC)
        !          1350:                        {
        !          1351: #                      ifdef DEBUG
        !          1352:                        printd("ctlr has %d credits\n", mp->mscp_header.tmscp_credits & 0xf);
        !          1353:                        printd("ctlr timeout = %d\n", mp->mscp_cnttmo);
        !          1354: #                      endif
        !          1355:                        sc->sc_state = S_RUN;
        !          1356:                        }
        !          1357:                else
        !          1358:                        sc->sc_state = S_IDLE;
        !          1359:                um->um_tab.b_active = 0;
        !          1360:                wakeup((caddr_t)um);
        !          1361:                return;
        !          1362:                }
        !          1363:        if (mp->mscp_unit >= NTMS)
        !          1364:                return;
        !          1365:        if ((ui = tmscpip[um->um_ctlr][mp->mscp_unit]) == 0)
        !          1366:                return;
        !          1367:        tms = &tms_info[ui->ui_unit];
        !          1368:        /*
        !          1369:         * Save endcode, endflags, and status for mtioctl get unit status.
        !          1370:         * NOTE: Don't do this on Clear serious exception (reposition no-op);
        !          1371:         *    which is done on close since this would
        !          1372:         *    overwrite the real status we want.
        !          1373:         */
        !          1374:        if (tms->tms_clserex != 1)
        !          1375:                {
        !          1376:                tms->tms_endcode = mp->mscp_opcode;
        !          1377:                tms->tms_flags = mp->mscp_flags;
        !          1378:                tms->tms_status = st;
        !          1379:                }
        !          1380:        else tms->tms_clserex = 0;
        !          1381: 
        !          1382:        switch (mp->mscp_opcode) {
        !          1383:        case M_OP_ONLIN|M_OP_END:
        !          1384:                tms->tms_type = mp->mscp_mediaid;
        !          1385:                dp = &tmsutab[ui->ui_unit];
        !          1386:                if (st == M_ST_SUCC)
        !          1387:                        {
        !          1388:                        /*
        !          1389:                         * Link the drive onto the controller queue
        !          1390:                         */
        !          1391:                        dp->b_forw = NULL;
        !          1392:                        if (um->um_tab.b_actf == NULL)
        !          1393:                                um->um_tab.b_actf = dp;
        !          1394:                        else
        !          1395:                                um->um_tab.b_actl->b_forw = dp;
        !          1396:                        um->um_tab.b_actl = dp;
        !          1397:                        ui->ui_flags = 1;       /* mark it online */
        !          1398:                        tms->tms_dsize=(daddr_t)mp->mscp_maxwrt;
        !          1399: #                      ifdef DEBUG
        !          1400:                        printd("tmscprsp: unit %d online\n", mp->mscp_unit);
        !          1401: #                      endif                   
        !          1402:                        /* 
        !          1403:                         * This define decodes the Media type identifier
        !          1404:                         */
        !          1405: #                      define F_to_C(x,i)     ( ((x)->mscp_mediaid) >> (i*5+7) & 0x1f ? ( ( (((x)->mscp_mediaid) >>( i*5 + 7)) & 0x1f) + 'A' - 1): ' ')
        !          1406: #                      ifdef DEBUG
        !          1407:                        printd("tmscprsp: unit %d online %x %c%c %c%c%c%d\n"
        !          1408:                                ,mp->mscp_unit, mp->mscp_mediaid ,F_to_C(mp,4)
        !          1409:                                ,F_to_C(mp,3), F_to_C(mp,2)
        !          1410:                                ,F_to_C(mp,1), F_to_C(mp,0), mp->mscp_mediaid & 0x7f);
        !          1411: #                      endif                           
        !          1412:                        dp->b_active = 1;
        !          1413:                        }       /* end if st == M_ST_SUCC */
        !          1414:                else 
        !          1415:                        {
        !          1416:                        if (bp = dp->b_actf)
        !          1417:                                tprintf(tms->tms_tpr,
        !          1418:                                    "tms%d: hard error bn%d: OFFLINE\n",
        !          1419:                                    minor(bp->b_dev)&03, bp->b_blkno);
        !          1420:                        else
        !          1421:                                tprintf(tms->tms_tpr,
        !          1422:                                    "tms%d: hard error: OFFLINE\n",
        !          1423:                                    ui->ui_unit);
        !          1424:                        while (bp = dp->b_actf)
        !          1425:                                {
        !          1426:                                dp->b_actf = bp->av_forw;
        !          1427:                                bp->b_flags |= B_ERROR;
        !          1428:                                iodone(bp);
        !          1429:                                }
        !          1430:                        }
        !          1431:                if(mp->mscp_cmdref!=NULL)
        !          1432:                        /* Seems to get lost sometimes in uda */
        !          1433:                        wakeup((caddr_t)mp->mscp_cmdref);
        !          1434:                break;
        !          1435:        /*
        !          1436:         * The AVAILABLE ATTENTION message occurs when the
        !          1437:         * unit becomes available after loading,
        !          1438:         * marking the unit offline (ui_flags = 0) will force an
        !          1439:         * online command prior to using the unit.
        !          1440:         */
        !          1441:        case M_OP_AVATN:
        !          1442:                ui->ui_flags = 0;
        !          1443:                tms->tms_type = mp->mscp_mediaid;
        !          1444:                break;
        !          1445:        case M_OP_END:
        !          1446:                /*
        !          1447:                 * An endcode without an opcode (0200) is an invalid command.
        !          1448:                 * The mscp specification states that this would be a protocol
        !          1449:                 * type error, such as illegal opcodes. The mscp spec. also
        !          1450:                 * states that parameter error type of invalid commands should
        !          1451:                 * return the normal end message for the command. This does not appear
        !          1452:                 * to be the case. An invalid logical block number returned an endcode
        !          1453:                 * of 0200 instead of the 0241 (read) that was expected.
        !          1454:                 */
        !          1455:                        
        !          1456:                printf("tmscp%d: invalid cmd, endcode = %o, status=%o\n",
        !          1457:                        um->um_ctlr, mp->mscp_opcode, st);
        !          1458:                bp = (struct buf *)mp->mscp_cmdref;
        !          1459:                /*
        !          1460:                 * Unlink buffer from I/O wait queue.
        !          1461:                 * And signal iodone, so the higher level command can exit!
        !          1462:                 *
        !          1463:                 */
        !          1464:                bp->av_back->av_forw = bp->av_forw;
        !          1465:                bp->av_forw->av_back = bp->av_back;
        !          1466:                dp = &tmsutab[ui->ui_unit];
        !          1467:                dp->b_qsize--;
        !          1468:                iodone(bp);
        !          1469:                break;
        !          1470:        case M_OP_WRITE|M_OP_END:
        !          1471:                /* mark the last io op as a write */
        !          1472:                tms->tms_lastiow = 1;
        !          1473:        case M_OP_READ|M_OP_END:
        !          1474:        case M_OP_WRITM|M_OP_END:
        !          1475:        case M_OP_REPOS|M_OP_END:
        !          1476:        case M_OP_STUNT|M_OP_END:
        !          1477:        /*
        !          1478:         * The AVAILABLE message occurs when the mt ioctl "rewoffl" is
        !          1479:         * issued.  For the ioctl, "rewoffl", a tmscp AVAILABLE command is
        !          1480:         * done with the UNLOAD modifier.  This performs a rewind, followed
        !          1481:         * by marking the unit offline.  So mark the unit offline
        !          1482:         * software wise as well (ui_flags = 0 and 
        !          1483:         * tms->tms_openf = 0).
        !          1484:         */
        !          1485:        case M_OP_AVAIL|M_OP_END:
        !          1486: #              ifdef DEBUG
        !          1487:                printd("tmscprsp: position = %d\n", mp->mscp_lbn);
        !          1488: #              endif
        !          1489:                bp = (struct buf *)mp->mscp_cmdref;
        !          1490:                /*
        !          1491:                 * Only need to release buffer if the command was read or write.
        !          1492:                 * No ubasetup was done in "tmscpstart" if it was an ioctl cmd.
        !          1493:                 */
        !          1494:                if (mp->mscp_opcode == (M_OP_READ|M_OP_END) || 
        !          1495:                    mp->mscp_opcode == (M_OP_WRITE|M_OP_END))
        !          1496:                        ubarelse(um->um_ubanum, (int *)&bp->b_ubinfo);
        !          1497:                /*
        !          1498:                 * Unlink buffer from I/O wait queue.
        !          1499:                 */
        !          1500:                bp->av_back->av_forw = bp->av_forw;
        !          1501:                bp->av_forw->av_back = bp->av_back;
        !          1502: #              if defined(VAX750)
        !          1503:                if (cpu == VAX_750) { 
        !          1504:                    if ((tmscpwtab[um->um_ctlr].av_forw == &tmscpwtab[um->um_ctlr]) &&
        !          1505:                                        (um->um_ubinfo != 0)) {
        !          1506:                        ubarelse(um->um_ubanum, &um->um_ubinfo);
        !          1507:                    }
        !          1508:                    else {
        !          1509:                        if (mp->mscp_opcode == (M_OP_READ|M_OP_END) ||
        !          1510:                            mp->mscp_opcode == (M_OP_WRITE|M_OP_END))
        !          1511:                                UBAPURGE(uba_hd[um->um_ubanum].uh_uba,(um->um_ubinfo >>28) & 0x0f);
        !          1512:                    }
        !          1513:                }
        !          1514: #              endif
        !          1515:                dp = &tmsutab[ui->ui_unit];
        !          1516:                dp->b_qsize--;
        !          1517:                if (st == M_ST_OFFLN || st == M_ST_AVLBL)
        !          1518:                        {
        !          1519:                        ui->ui_flags = 0;       /* mark unit offline */
        !          1520:                        tms->tms_openf = 0;
        !          1521:                        tms->tms_type = mp->mscp_mediaid;
        !          1522:                        /*
        !          1523:                         * Link the buffer onto the front of the drive queue
        !          1524:                         */
        !          1525:                        if ((bp->av_forw = dp->b_actf) == 0)
        !          1526:                                dp->b_actl = bp;
        !          1527:                        dp->b_actf = bp;
        !          1528:                        /*
        !          1529:                         * Link the drive onto the controller queue
        !          1530:                         */
        !          1531:                        if (dp->b_active == 0)
        !          1532:                                {
        !          1533:                                dp->b_forw = NULL;
        !          1534:                                if (um->um_tab.b_actf == NULL)
        !          1535:                                        um->um_tab.b_actf = dp;
        !          1536:                                else
        !          1537:                                        um->um_tab.b_actl->b_forw = dp;
        !          1538:                                um->um_tab.b_actl = dp;
        !          1539:                                dp->b_active = 1;
        !          1540:                                }
        !          1541: #                      if defined(VAX750)
        !          1542:                        if (cpu == VAX_750 && um->um_ubinfo == 0)
        !          1543:                                um->um_ubinfo = uballoc(um->um_ubanum, (caddr_t)0, 0, UBA_NEEDBDP);
        !          1544: #                      endif
        !          1545:                        return;
        !          1546:                        }
        !          1547:                if (st != M_ST_SUCC)
        !          1548:                        {
        !          1549:                        if (mp->mscp_flags & M_EF_SEREX)
        !          1550:                                tms->tms_serex = 1;
        !          1551:                        if (st != M_ST_TAPEM)
        !          1552:                                {
        !          1553:                                tprintf(tms->tms_tpr,
        !          1554:                                    "tms%d: hard error bn%d\n",
        !          1555:                                    minor(bp->b_dev)&03, bp->b_blkno);
        !          1556:                                errinfo(st);            /* produces more info */
        !          1557: #                              ifdef DEBUG
        !          1558:                                printd("tmscprsp: error; status sub-code = 0%o, flags = 0%o\n",
        !          1559:                                        (mp->mscp_status & 177740)>>5, mp->mscp_flags);
        !          1560: #                              endif
        !          1561:                                bp->b_flags |= B_ERROR;
        !          1562:                                }
        !          1563:                        else
        !          1564:                                /* Hit a tape mark - Set serex flag to
        !          1565:                                 * a special value so we can clear the
        !          1566:                                 * serious exception on the next command.
        !          1567:                                 */
        !          1568:                                tms->tms_serex = 2;
        !          1569:                        }
        !          1570:                /*
        !          1571:                 * The tmscp spec states that controllers do not have to
        !          1572:                 * report the number of records or files skipped.  So on
        !          1573:                 * reposition commands we go strictly by cmd status.
        !          1574:                 */
        !          1575:                if (mp->mscp_opcode != (M_OP_REPOS|M_OP_END))
        !          1576:                        bp->b_resid = bp->b_bcount - mp->mscp_bytecnt;
        !          1577:                else
        !          1578:                        bp->b_resid = 0;
        !          1579:                tms->tms_resid = bp->b_resid;
        !          1580:                iodone(bp);
        !          1581:                break;
        !          1582: 
        !          1583:        case M_OP_GTUNT|M_OP_END:
        !          1584: #              ifdef DEBUG
        !          1585:                printd("tmscprsp: GTUNT end packet status = 0%o\n",st);
        !          1586:                printd("tmscprsp: unit %d mediaid %x %c%c %c%c%c%d %x %x t=%d\n"
        !          1587:                    ,mp->mscp_unit, mp->mscp_mediaid
        !          1588:                    ,F_to_C(mp,4),F_to_C(mp,3),F_to_C(mp,2)
        !          1589:                    ,F_to_C(mp,1),F_to_C(mp,0)
        !          1590:                    ,mp->mscp_mediaid & 0x7f
        !          1591:                    ,mp->mscp_unitid.val[0]
        !          1592:                    ,mp->mscp_unitid.val[1]
        !          1593:                    ,mp->mscp_format);
        !          1594: #              endif           
        !          1595:                tms->tms_type = mp->mscp_mediaid;
        !          1596:                tms->tms_fmtmenu = mp->mscp_fmtmenu;
        !          1597:                tms->tms_unitflgs = mp->mscp_unitflgs;
        !          1598:                break;
        !          1599: 
        !          1600:        default:
        !          1601:                printf("tmscp unknown packet\n");
        !          1602:                tmserror(um, (struct mslg *)mp);
        !          1603:        }       /* end switch mp->mscp_opcode */
        !          1604: }
        !          1605: 
        !          1606: 
        !          1607: /* 
        !          1608:  * Give a meaningful error when the mscp_status field returns an error code.
        !          1609:  */
        !          1610: 
        !          1611: errinfo(st)
        !          1612:        int st;                 /* the status code */
        !          1613: {
        !          1614:        switch(st) {
        !          1615:        case M_ST_ICMD:
        !          1616:                printf("invalid command\n");
        !          1617:                break;
        !          1618:        case M_ST_ABRTD:
        !          1619:                printf("command aborted\n");
        !          1620:                break;
        !          1621:        case M_ST_OFFLN:
        !          1622:                printf("unit offline\n");
        !          1623:                break;
        !          1624:        case M_ST_WRTPR:
        !          1625:                printf("unit write protected\n");
        !          1626:                break;
        !          1627:        case M_ST_COMP:
        !          1628:                printf("compare error\n");
        !          1629:                break;
        !          1630:        case M_ST_DATA:
        !          1631:                printf("data error\n");
        !          1632:                break;
        !          1633:        case M_ST_HSTBF:
        !          1634:                printf("host buffer access error\n");
        !          1635:                break;
        !          1636:        case M_ST_CNTLR:
        !          1637:                printf("controller error\n");
        !          1638:                break;
        !          1639:        case M_ST_DRIVE:
        !          1640:                printf("drive error\n");
        !          1641:                break;
        !          1642:        case M_ST_FMTER:
        !          1643:                printf("formatter error\n");
        !          1644:                break;
        !          1645:        case M_ST_BOT:
        !          1646:                printf("BOT encountered\n");
        !          1647:                break;
        !          1648:        case M_ST_TAPEM:
        !          1649:                printf("tape mark encountered\n");
        !          1650:                break;
        !          1651:        case M_ST_RDTRN:
        !          1652:                printf("record data truncated\n");
        !          1653:                break;
        !          1654:        case M_ST_PLOST:
        !          1655:                printf("position lost\n");
        !          1656:                break;
        !          1657:        case M_ST_SEX:
        !          1658:                printf("serious exception\n");
        !          1659:                break;
        !          1660:        case M_ST_LED:
        !          1661:                printf("LEOT detected\n");
        !          1662:                break;
        !          1663:        }
        !          1664: }
        !          1665: 
        !          1666: 
        !          1667: /*
        !          1668:  * Manage buffers and perform block mode read and write operations.
        !          1669:  */
        !          1670: 
        !          1671: tmscpstrategy (bp)
        !          1672:        register struct buf *bp;
        !          1673: {
        !          1674:        register struct uba_device *ui;
        !          1675:        register struct uba_ctlr *um;
        !          1676:        register struct buf *dp;
        !          1677:        register int unit = TMSUNIT(bp->b_dev);
        !          1678:        int s;
        !          1679: 
        !          1680:        if (unit >= NTMS)
        !          1681:                {
        !          1682: #              ifdef DEBUG
        !          1683:                printd ("tmscpstrategy: bad unit # %d\n",unit);
        !          1684: #              endif
        !          1685:                bp->b_flags |= B_ERROR;
        !          1686:                iodone(bp);
        !          1687:                return;
        !          1688:                }
        !          1689:        ui = tmsdinfo[unit];
        !          1690:        um = ui->ui_mi;
        !          1691:        if (ui == 0 || ui->ui_alive == 0)
        !          1692:                {
        !          1693:                bp->b_flags |= B_ERROR;
        !          1694:                iodone(bp);
        !          1695:                return;
        !          1696:                }
        !          1697:        s = spl5();
        !          1698:        /*
        !          1699:         * Link the buffer onto the drive queue
        !          1700:         */
        !          1701:        dp = &tmsutab[ui->ui_unit];
        !          1702:        if (dp->b_actf == 0)
        !          1703:                dp->b_actf = bp;
        !          1704:        else
        !          1705:                dp->b_actl->av_forw = bp;
        !          1706:        dp->b_actl = bp;
        !          1707:        bp->av_forw = 0;
        !          1708:        /*
        !          1709:         * Link the drive onto the controller queue
        !          1710:         */
        !          1711:        if (dp->b_active == 0)
        !          1712:                {
        !          1713:                dp->b_forw = NULL;
        !          1714:                if (um->um_tab.b_actf == NULL)
        !          1715:                        um->um_tab.b_actf = dp;
        !          1716:                else
        !          1717:                        um->um_tab.b_actl->b_forw = dp;
        !          1718:                um->um_tab.b_actl = dp;
        !          1719:                dp->b_active = 1;
        !          1720:                }
        !          1721:        /*
        !          1722:         * If the controller is not active, start it.
        !          1723:         */
        !          1724:        if (um->um_tab.b_active == 0)
        !          1725:                {
        !          1726: #              if defined(VAX750)
        !          1727:                if (cpu == VAX_750
        !          1728:                                 && tmscpwtab[um->um_ctlr].av_forw == &tmscpwtab[um->um_ctlr])
        !          1729:                        {
        !          1730:                        if (um->um_ubinfo != 0)
        !          1731:                                log(TMS_PRI, "tmscpstrategy: ubinfo 0x%x\n",
        !          1732:                                    um->um_ubinfo);
        !          1733:                        else
        !          1734:                                um->um_ubinfo = uballoc(um->um_ubanum, (caddr_t)0, 0, UBA_NEEDBDP);
        !          1735:                        }
        !          1736: #              endif
        !          1737: #              ifdef DEBUG
        !          1738:                printd10("tmscpstrategy: Controller not active, starting it\n");
        !          1739: #              endif
        !          1740:                (void) tmscpstart(um);
        !          1741:                }
        !          1742:        splx(s);
        !          1743:        return;
        !          1744: }
        !          1745: 
        !          1746: #define DBSIZE 32
        !          1747: 
        !          1748: #define ca_Rspdsc       ca_rspdsc[0]
        !          1749: #define ca_Cmddsc       ca_rspdsc[1]
        !          1750: #define tmscp_Rsp       tmscp_rsp[0]
        !          1751: #define tmscp_Cmd       tmscp_cmd[0]
        !          1752: 
        !          1753: struct  tmscp     tmscpd[NTMSCP];
        !          1754: 
        !          1755: tmscpdump(dev)
        !          1756:        dev_t dev;
        !          1757: {
        !          1758:        struct tmscpdevice *tmscpaddr;
        !          1759:        struct tmscp *tmscp_ubaddr;
        !          1760:        char *start;
        !          1761:        int num, blk, unit;
        !          1762:        register struct uba_regs *uba;
        !          1763:        register struct uba_device *ui;
        !          1764:        register struct tmscp *tmscpp;
        !          1765:        register struct pte *io;
        !          1766:        register int i;
        !          1767: 
        !          1768:        unit = minor(dev) & 03;
        !          1769:        if (unit >= NTMS)
        !          1770:                return (ENXIO);
        !          1771: #      define phys(cast, addr) ((cast)((int)addr & 0x7fffffff))
        !          1772:        ui = phys(struct uba_device *, tmsdinfo[unit]);
        !          1773:        if (ui->ui_alive == 0)
        !          1774:                return (ENXIO);
        !          1775:        uba = phys(struct uba_hd *, ui->ui_hd)->uh_physuba;
        !          1776:        ubainit(uba);
        !          1777:        tmscpaddr = (struct tmscpdevice *)ui->ui_physaddr;
        !          1778:        DELAY(2000000);
        !          1779:        tmscpp = phys(struct tmscp *, &tmscpd[ui->ui_ctlr]);
        !          1780: 
        !          1781:        num = btoc(sizeof(struct tmscp)) + 1;
        !          1782:        io = &uba->uba_map[NUBMREG-num];
        !          1783:        for(i = 0; i<num; i++)
        !          1784:                *(int *)io++ = UBAMR_MRV|(btop(tmscpp)+i);
        !          1785:        tmscp_ubaddr = (struct tmscp *)(((int)tmscpp & PGOFSET)|((NUBMREG-num)<<9));
        !          1786: 
        !          1787:        tmscpaddr->tmscpip = 0;
        !          1788:        while ((tmscpaddr->tmscpsa & TMSCP_STEP1) == 0)
        !          1789:                if(tmscpaddr->tmscpsa & TMSCP_ERR) return(EFAULT);
        !          1790:        tmscpaddr->tmscpsa = TMSCP_ERR;
        !          1791:        while ((tmscpaddr->tmscpsa & TMSCP_STEP2) == 0)
        !          1792:                if(tmscpaddr->tmscpsa & TMSCP_ERR) return(EFAULT);
        !          1793:        tmscpaddr->tmscpsa = (short)&tmscp_ubaddr->tmscp_ca.ca_ringbase;
        !          1794:        while ((tmscpaddr->tmscpsa & TMSCP_STEP3) == 0)
        !          1795:                if(tmscpaddr->tmscpsa & TMSCP_ERR) return(EFAULT);
        !          1796:        tmscpaddr->tmscpsa = (short)(((int)&tmscp_ubaddr->tmscp_ca.ca_ringbase) >> 16);
        !          1797:        while ((tmscpaddr->tmscpsa & TMSCP_STEP4) == 0)
        !          1798:                if(tmscpaddr->tmscpsa & TMSCP_ERR) return(EFAULT);
        !          1799:        tmscpaddr->tmscpsa = TMSCP_GO;
        !          1800:        tmscpp->tmscp_ca.ca_Rspdsc = (long)&tmscp_ubaddr->tmscp_Rsp.mscp_cmdref;
        !          1801:        tmscpp->tmscp_ca.ca_Cmddsc = (long)&tmscp_ubaddr->tmscp_Cmd.mscp_cmdref;
        !          1802:        tmscpp->tmscp_Cmd.mscp_header.tmscp_vcid = 1;   /* for tape */
        !          1803:        tmscpp->tmscp_Cmd.mscp_cntflgs = 0;
        !          1804:        tmscpp->tmscp_Cmd.mscp_version = 0;
        !          1805:        if (tmscpcmd(M_OP_STCON, tmscpp, tmscpaddr) == 0) {
        !          1806:                return(EFAULT);
        !          1807:        }
        !          1808:        tmscpp->tmscp_Cmd.mscp_unit = ui->ui_slave;
        !          1809:        if (tmscpcmd(M_OP_ONLIN, tmscpp, tmscpaddr) == 0) {
        !          1810:                return(EFAULT);
        !          1811:        }
        !          1812: 
        !          1813:        num = maxfree;
        !          1814:        start = 0;
        !          1815:        while (num > 0)
        !          1816:                {
        !          1817:                blk = num > DBSIZE ? DBSIZE : num;
        !          1818:                io = uba->uba_map;
        !          1819:                for (i = 0; i < blk; i++)
        !          1820:                        *(int *)io++ = (btop(start)+i) | UBAMR_MRV;
        !          1821:                *(int *)io = 0;
        !          1822:                tmscpp->tmscp_Cmd.mscp_lbn = btop(start);
        !          1823:                tmscpp->tmscp_Cmd.mscp_unit = ui->ui_slave;
        !          1824:                tmscpp->tmscp_Cmd.mscp_bytecnt = blk*NBPG;
        !          1825: #              ifdef   MVAX
        !          1826:                if( cpu == MVAX_I )
        !          1827:                        tmscpp->tmscp_Cmd.mscp_buffer = (long) start;
        !          1828:                else
        !          1829: #              endif   MVAX
        !          1830:                        tmscpp->tmscp_Cmd.mscp_buffer = 0;
        !          1831:                if (tmscpcmd(M_OP_WRITE, tmscpp, tmscpaddr) == 0)
        !          1832:                        return(EIO);
        !          1833:                start += blk*NBPG;
        !          1834:                num -= blk;
        !          1835:                }
        !          1836:        return (0);
        !          1837: }
        !          1838: 
        !          1839: 
        !          1840: /*
        !          1841:  * Perform a standalone tmscp command.  This routine is only used by tmscpdump.
        !          1842:  */
        !          1843: 
        !          1844: tmscpcmd(op, tmscpp, tmscpaddr)
        !          1845:        int op;
        !          1846:        register struct tmscp *tmscpp;
        !          1847:        struct tmscpdevice *tmscpaddr;
        !          1848: {
        !          1849:        int i;
        !          1850: 
        !          1851: 
        !          1852:        tmscpp->tmscp_Cmd.mscp_opcode = op;
        !          1853:        tmscpp->tmscp_Rsp.mscp_header.tmscp_msglen = mscp_msglen;
        !          1854:        tmscpp->tmscp_Cmd.mscp_header.tmscp_msglen = mscp_msglen;
        !          1855:        tmscpp->tmscp_ca.ca_Rspdsc |= TMSCP_OWN|TMSCP_INT;
        !          1856:        tmscpp->tmscp_ca.ca_Cmddsc |= TMSCP_OWN|TMSCP_INT;
        !          1857:        if (tmscpaddr->tmscpsa&TMSCP_ERR)
        !          1858:                printf("tmscp fatal error (0%o)\n", tmscpaddr->tmscpsa&0xffff);
        !          1859:        i = tmscpaddr->tmscpip;
        !          1860: #ifdef lint
        !          1861:        i = i;
        !          1862: #endif
        !          1863:        for (;;)
        !          1864:                {
        !          1865:                if (tmscpp->tmscp_ca.ca_cmdint)
        !          1866:                        tmscpp->tmscp_ca.ca_cmdint = 0;
        !          1867:                if (tmscpp->tmscp_ca.ca_rspint)
        !          1868:                        break;
        !          1869:                }
        !          1870:        tmscpp->tmscp_ca.ca_rspint = 0;
        !          1871:        if (tmscpp->tmscp_Rsp.mscp_opcode != (op|M_OP_END) ||
        !          1872:            (tmscpp->tmscp_Rsp.mscp_status&M_ST_MASK) != M_ST_SUCC)
        !          1873:                {
        !          1874:                printf("error: com %d opc 0x%x stat 0x%x\ndump ", op,
        !          1875:                        tmscpp->tmscp_Rsp.mscp_opcode, tmscpp->tmscp_Rsp.mscp_status);
        !          1876:                return(0);
        !          1877:                }
        !          1878:        return(1);
        !          1879: }
        !          1880: 
        !          1881: /*
        !          1882:  * Catch ioctl commands, and call the "command" routine to do them.
        !          1883:  */
        !          1884: 
        !          1885: /* ARGSUSED */
        !          1886: tmscpioctl(dev, cmd, data, flag)
        !          1887:        dev_t dev;
        !          1888:        int cmd;
        !          1889:        caddr_t data;
        !          1890:        int flag;
        !          1891: {
        !          1892:        register struct buf *bp = &ctmscpbuf[TMSCPCTLR(dev)];
        !          1893:        register callcount;     /* number of times to call cmd routine */
        !          1894:        register struct uba_device *ui;
        !          1895:        register struct tms_info *tms;
        !          1896:        int fcount;             /* number of files (or records) to space */
        !          1897:        int error = 0;
        !          1898:        register struct mtop *mtop;     /* mag tape cmd op to perform */
        !          1899:        register struct mtget *mtget;   /* mag tape struct to get info in */
        !          1900: 
        !          1901:        /* we depend of the values and order of the TMS ioctl codes here */
        !          1902:        static tmsops[] =
        !          1903:         {TMS_WRITM,TMS_FSF,TMS_BSF,TMS_FSR,TMS_BSR,TMS_REW,TMS_OFFL,TMS_SENSE,
        !          1904:          TMS_CACHE,TMS_NOCACHE};
        !          1905: 
        !          1906:        switch (cmd) {
        !          1907:        case MTIOCTOP:  /* tape operation */
        !          1908:                mtop = (struct mtop *)data;
        !          1909:                switch (mtop->mt_op) {
        !          1910: 
        !          1911:                case MTWEOF:
        !          1912:                        callcount = mtop->mt_count;
        !          1913:                        fcount = 1;
        !          1914:                        break;
        !          1915:                case MTFSF: case MTBSF:
        !          1916:                case MTFSR: case MTBSR:
        !          1917:                        callcount = 1;
        !          1918:                        fcount = mtop->mt_count;
        !          1919:                        break;
        !          1920:                case MTREW: case MTOFFL: case MTNOP:
        !          1921:                case MTCACHE: case MTNOCACHE:
        !          1922:                        callcount = 1;
        !          1923:                        fcount = 1;             /* wait for this rewind */
        !          1924:                        break;
        !          1925:                default:
        !          1926:                        return (ENXIO);
        !          1927:                }       /* end switch mtop->mt_op */
        !          1928: 
        !          1929:                if (callcount <= 0 || fcount <= 0)
        !          1930:                        return (EINVAL);
        !          1931:                while (--callcount >= 0)
        !          1932:                        {
        !          1933:                        tmscpcommand(dev, tmsops[mtop->mt_op], fcount);
        !          1934:                        if ((mtop->mt_op == MTFSR || mtop->mt_op == MTBSR) &&
        !          1935:                            bp->b_resid)
        !          1936:                                return (EIO);
        !          1937:                        if (bp->b_flags & B_ERROR)      /* like hitting BOT */
        !          1938:                                break;
        !          1939:                        }
        !          1940:                if (bp->b_flags&B_ERROR)
        !          1941:                        if ((error = bp->b_error)==0)
        !          1942:                                return (EIO);
        !          1943:                return (error);
        !          1944: 
        !          1945:        case MTIOCGET:
        !          1946:                /*
        !          1947:                 * Return status info associated with the particular UNIT.
        !          1948:                 */
        !          1949:                ui = tmsdinfo[TMSUNIT(dev)];
        !          1950:                tms = &tms_info[ui->ui_unit];
        !          1951:                mtget = (struct mtget *)data;
        !          1952:                mtget->mt_type = MT_ISTMSCP;
        !          1953:                mtget->mt_dsreg = tms->tms_flags << 8;
        !          1954:                mtget->mt_dsreg |= tms->tms_endcode;
        !          1955:                mtget->mt_erreg = tms->tms_status;
        !          1956:                mtget->mt_resid = tms->tms_resid;
        !          1957:                break;
        !          1958: 
        !          1959:        default:
        !          1960:                return (ENXIO);
        !          1961:        }
        !          1962:        return (0);
        !          1963: }
        !          1964: 
        !          1965: 
        !          1966: /*
        !          1967:  * Reset (for raw mode use only).
        !          1968:  */
        !          1969: 
        !          1970: tmscpreset (uban)
        !          1971:        int uban;
        !          1972: {
        !          1973:        register struct uba_ctlr *um;
        !          1974:        register struct uba_device *ui;
        !          1975:        register struct buf *bp, *dp;
        !          1976:        register int unit;
        !          1977:        struct buf *nbp;
        !          1978:        int d;
        !          1979: 
        !          1980:        for (d = 0; d < NTMSCP; d++)
        !          1981:                {
        !          1982:                if ((um = tmscpminfo[d]) == 0 || um->um_ubanum != uban ||
        !          1983:                    um->um_alive == 0)
        !          1984:                        continue;
        !          1985:                printf(" tmscp%d", d);
        !          1986:                um->um_tab.b_active = 0;
        !          1987:                um->um_tab.b_actf = um->um_tab.b_actl = 0;
        !          1988:                tmscp_softc[d].sc_state = S_IDLE;
        !          1989:                tmscp_softc[d].sc_mapped = 0;
        !          1990:                for (unit = 0; unit < NTMS; unit++)
        !          1991:                        {
        !          1992:                        if ((ui = tmsdinfo[unit]) == 0)
        !          1993:                                continue;
        !          1994:                        if (ui->ui_alive == 0 || ui->ui_mi != um)
        !          1995:                                continue;
        !          1996:                        tmsutab[unit].b_active = 0;
        !          1997:                        tmsutab[unit].b_qsize = 0;
        !          1998:                        }
        !          1999:                for (bp = tmscpwtab[d].av_forw; bp != &tmscpwtab[d]; bp = nbp)
        !          2000:                        {
        !          2001:                        nbp = bp->av_forw;
        !          2002:                        bp->b_ubinfo = 0;
        !          2003:                        /*
        !          2004:                         * Link the buffer onto the drive queue
        !          2005:                         */
        !          2006:                        dp = &tmsutab[TMSUNIT(bp->b_dev)];
        !          2007:                        if (dp->b_actf == 0)
        !          2008:                                dp->b_actf = bp;
        !          2009:                        else
        !          2010:                                dp->b_actl->av_forw = bp;
        !          2011:                        dp->b_actl = bp;
        !          2012:                        bp->av_forw = 0;
        !          2013:                        /*
        !          2014:                         * Link the drive onto the controller queue
        !          2015:                         */
        !          2016:                        if (dp->b_active == 0)
        !          2017:                                {
        !          2018:                                dp->b_forw = NULL;
        !          2019:                                if (um->um_tab.b_actf == NULL)
        !          2020:                                        um->um_tab.b_actf = dp;
        !          2021:                                else
        !          2022:                                        um->um_tab.b_actl->b_forw = dp;
        !          2023:                                um->um_tab.b_actl = dp;
        !          2024:                                dp->b_active = 1;
        !          2025:                                }
        !          2026:                        }
        !          2027:                (void)tmscpinit(d);
        !          2028:                }
        !          2029: }
        !          2030: 
        !          2031: 
        !          2032: /*
        !          2033:  * Process an error log message
        !          2034:  *
        !          2035:  * Only minimal decoding is done, only "useful"
        !          2036:  * information is printed.  Eventually should
        !          2037:  * send message to an error logger.
        !          2038:  */
        !          2039: 
        !          2040: tmserror(um, mp)
        !          2041:        register struct uba_ctlr *um;
        !          2042:        register struct mslg *mp;
        !          2043: {
        !          2044:        register i;
        !          2045: 
        !          2046: #      ifdef DEBUG
        !          2047:        printd("tmserror:\n");
        !          2048: #      endif
        !          2049:        if(!(mp->mslg_flags & (M_LF_SUCC | M_LF_CONT)))
        !          2050:                log(TMS_PRI, "tmscp%d: %s error, ", um->um_ctlr,
        !          2051:                mp->mslg_flags & ( M_LF_SUCC | M_LF_CONT ) ? "soft" : "hard");
        !          2052: 
        !          2053:        switch (mp->mslg_format) {
        !          2054: 
        !          2055:        case M_FM_CNTERR:
        !          2056:                log(TMS_PRI, "controller error, event 0%o\n", mp->mslg_event);
        !          2057:                break;
        !          2058:        case M_FM_BUSADDR:
        !          2059:                log(TMS_PRI, "host memory access error, event 0%o, addr 0%o\n",
        !          2060:                        mp->mslg_event, mp->mslg_busaddr);
        !          2061:                break;
        !          2062:        case M_FM_TAPETRN:
        !          2063:                log(TMS_PRI, "tape transfer error, unit %d, grp 0x%x, event 0%o\n",
        !          2064:                        mp->mslg_unit, mp->mslg_group, mp->mslg_event);
        !          2065:                break;
        !          2066:        case M_FM_STIERR:
        !          2067:                log(TMS_PRI, "STI error, unit %d, event 0%o\n",
        !          2068:                        mp->mslg_unit, mp->mslg_event);
        !          2069: #ifdef notdef
        !          2070:                /* too painful to do with log() */
        !          2071:                for(i = 0; i < 62;i++)
        !          2072:                        mprintf("\t0x%x",mp->mslg_stiunsucc[i] & 0xff);
        !          2073:                mprintf("\n");
        !          2074: #endif
        !          2075:                break;
        !          2076:        case M_FM_STIDEL:
        !          2077:                log(TMS_PRI, "STI Drive Error Log, unit %d, event 0%o\n",
        !          2078:                        mp->mslg_unit, mp->mslg_event);
        !          2079:                break;
        !          2080:        case M_FM_STIFEL:
        !          2081:                log(TMS_PRI, "STI Formatter Error Log, unit %d, event 0%o\n",
        !          2082:                        mp->mslg_unit, mp->mslg_event);
        !          2083:                break;
        !          2084:        default:
        !          2085:                log(TMS_PRI, "unknown error, unit %d, format 0%o, event 0%o\n",
        !          2086:                        mp->mslg_unit, mp->mslg_format, mp->mslg_event);
        !          2087:        }
        !          2088: 
        !          2089:        if (tmscperror)
        !          2090:                {
        !          2091:                register long *p = (long *)mp;
        !          2092: 
        !          2093:                for (i = 0; i < mp->mslg_header.tmscp_msglen; i += sizeof(*p))
        !          2094:                        printf("%x ", *p++);
        !          2095:                printf("\n");
        !          2096:                }
        !          2097: }
        !          2098: #endif

unix.superglobalmegacorp.com

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