Annotation of researchv10no/sys/io/ttyld.c, revision 1.1

1.1     ! root        1: #include "sys/param.h"
        !             2: #include "sys/stream.h"
        !             3: #include "sys/ttyio.h"
        !             4: #include "sys/ttyld.h"
        !             5: #include "sys/conf.h"
        !             6: 
        !             7: extern char    partab[];
        !             8: 
        !             9: #define        CANBSIZ 256             /* size of largest input line */
        !            10: 
        !            11: extern struct  ttyld   ttyld[];
        !            12: extern int ttycnt;
        !            13: 
        !            14: char   maptab[] = {
        !            15:        000,000,000,000,000,000,000,000,
        !            16:        000,000,000,000,000,000,000,000,
        !            17:        000,000,000,000,000,000,000,000,
        !            18:        000,000,000,000,000,000,000,000,
        !            19:        000,'|',000,000,000,000,000,'`',
        !            20:        '{','}',000,000,000,000,000,000,
        !            21:        000,000,000,000,000,000,000,000,
        !            22:        000,000,000,000,000,000,000,000,
        !            23:        000,000,000,000,000,000,000,000,
        !            24:        000,000,000,000,000,000,000,000,
        !            25:        000,000,000,000,000,000,000,000,
        !            26:        000,000,000,000,'\\',000,'~',000,
        !            27:        000,'A','B','C','D','E','F','G',
        !            28:        'H','I','J','K','L','M','N','O',
        !            29:        'P','Q','R','S','T','U','V','W',
        !            30:        'X','Y','Z',000,000,000,000,000,
        !            31: };
        !            32: 
        !            33: struct block   *canonblock();
        !            34: 
        !            35: long   ttyopen();
        !            36: int    ttyclose(), ttyldin(), ttyinsrv(), ttyosrv();
        !            37: static struct qinit ttrinit = { ttyldin, ttyinsrv, ttyopen, ttyclose, 3*1024, 60};
        !            38: static struct qinit ttwinit = { putq, ttyosrv, ttyopen, ttyclose, 3*1024, 200};
        !            39: struct streamtab ttystream = { &ttrinit, &ttwinit};
        !            40: 
        !            41: /*
        !            42:  * TTY open
        !            43:  */
        !            44: long
        !            45: ttyopen(qp, dev)
        !            46: register struct queue *qp;
        !            47: {
        !            48:        register struct ttyld *tp;
        !            49:        static struct tchars tchars = {CINTR,CQUIT,CSTART,CSTOP,CEOT,0377};
        !            50: 
        !            51:        if (qp->ptr)                            /* already attached */
        !            52:                return(1);
        !            53:        for (tp = ttyld; tp->t_state&TTUSE; tp++)
        !            54:                if (tp >= &ttyld[ttycnt-1])
        !            55:                        return(0);
        !            56:        tp->t_state = TTUSE;
        !            57:        tp->t_flags = ECHO|CRMOD;
        !            58:        tp->t_delct = 0;
        !            59:        tp->t_col = 0;
        !            60:        tp->t_erase = CERASE;
        !            61:        tp->t_kill = CKILL;
        !            62:        tp->t_chr = tchars;
        !            63:        qp->ptr = (caddr_t)tp;
        !            64:        qp->flag |= QDELIM|QNOENB;
        !            65:        WR(qp)->ptr = (caddr_t)tp;
        !            66:        return(1);
        !            67: }
        !            68: 
        !            69: ttyclose(qp)
        !            70: struct queue *qp;
        !            71: {
        !            72:        struct ttyld *tp = (struct ttyld *)qp->ptr;
        !            73: 
        !            74:        if (tp->t_state & TTSTOP)
        !            75:                putctl(WR(qp)->next, M_START);  /* what else can we do? */
        !            76:        tp->t_state = 0;
        !            77: }
        !            78: 
        !            79: /*
        !            80:  * Queue put procedure for tty input
        !            81:  */
        !            82: ttyldin(q, bp)
        !            83: struct queue *q;
        !            84: register struct block *bp;
        !            85: {
        !            86:        register struct ttyld *tp;
        !            87:        register c;
        !            88:        register struct queue *wrq = WR(q);     /* writer side */
        !            89:        int escape, flags;
        !            90: 
        !            91:        tp = (struct ttyld *)q->ptr;
        !            92:        flags = tp->t_flags;
        !            93:        bp->class &= ~S_DELIM;
        !            94:        if (bp->type!=M_DATA) {
        !            95:                switch(bp->type) {
        !            96: 
        !            97:                case M_BREAK:
        !            98:                        if (tp->t_flags&RAW) {          /* speed-change hack*/
        !            99:                                bp->type = M_DATA;
        !           100:                                bp->class &= ~S_DELIM;
        !           101:                                if (bp->wptr<bp->lim)
        !           102:                                        *bp->wptr++ = '\0';
        !           103:                                break;
        !           104:                        }
        !           105:                        ttysig(q, SIGINT);
        !           106:                        freeb(bp);
        !           107:                        return;
        !           108: 
        !           109:                case M_HANGUP:
        !           110:                case M_IOCACK:
        !           111:                case M_IOCNAK:
        !           112:                        (*q->next->qinfo->putp)(q->next, bp);
        !           113:                        return;
        !           114: 
        !           115:                case M_IOCTL:
        !           116:                        ttldioc(WR(q), bp, q, 1);
        !           117:                        return;
        !           118:                }
        !           119:                flags |= RAW;
        !           120:        }
        !           121:        if (tp->t_flags&TANDEM && tp->t_state&TTBLOCK
        !           122:         && q->count <= q->qinfo->lolimit) {
        !           123:                tp->t_state &= ~TTBLOCK;
        !           124:                putd(putq, WR(q), tp->t_chr.t_startc);
        !           125:        }
        !           126:        if (flags&RAW) {
        !           127:                if ((q->next->flag&QFULL)==0 && q->count==0)
        !           128:                        (*q->next->qinfo->putp)(q->next, bp);
        !           129:                else
        !           130:                        putq(q, bp);
        !           131:                return;
        !           132:        }
        !           133:        while (bp->rptr<bp->wptr) {
        !           134:                c = *bp->rptr++ & 0177;
        !           135:                if (tp->t_state&TTSTOP) {
        !           136:                        if (c!=tp->t_chr.t_stopc
        !           137:                         || tp->t_chr.t_stopc==tp->t_chr.t_startc) {
        !           138:                                tp->t_state &= ~TTSTOP;
        !           139:                                putctl(wrq->next, M_START);
        !           140:                        }
        !           141:                } else {
        !           142:                        if (c==tp->t_chr.t_stopc) {
        !           143:                                tp->t_state |= TTSTOP;
        !           144:                                putctl(wrq->next, M_STOP);
        !           145:                        }
        !           146:                }
        !           147:                if (c==tp->t_chr.t_stopc || c==tp->t_chr.t_startc)
        !           148:                        continue;
        !           149:                if (c==tp->t_chr.t_intrc) {
        !           150:                        ttysig(q, SIGINT);
        !           151:                        continue;
        !           152:                }
        !           153:                if (c==tp->t_chr.t_quitc) {
        !           154:                        ttysig(q, SIGQUIT);
        !           155:                        continue;
        !           156:                }
        !           157:                if (c=='\r' && tp->t_flags&CRMOD)
        !           158:                        c = '\n';
        !           159:                if (tp->t_flags&LCASE && c>='A' && c<='Z')
        !           160:                        c += 'a'-'A';
        !           161:                escape = 0;
        !           162:                if (tp->t_flags & CBREAK) {
        !           163:                        if ((q->next->flag&QFULL)==0 && q->count==0)
        !           164:                                putd(q->next->qinfo->putp, q->next, c);
        !           165:                        else
        !           166:                                putd(putq, q, c);
        !           167:                } else {
        !           168:                        if (tp->t_state&TTESC) {
        !           169:                                escape = 1;
        !           170:                                c |= 0200;
        !           171:                        }
        !           172:                        if (c == '\\')
        !           173:                                tp->t_state |= TTESC;
        !           174:                        else {
        !           175:                                tp->t_state &= ~TTESC;
        !           176:                                if (c == ('\\'|0200)) {
        !           177:                                        c &= 0177;
        !           178:                                        tp->t_state |= TTESC;
        !           179:                                }
        !           180:                                /* ttyhog? */
        !           181:                                if ((q->flag&QFULL) == 0 || (c=='\n' && tp->t_delct==0))
        !           182:                                        putd(putq, q, c);
        !           183:                                else
        !           184:                                        c = '\007';
        !           185:                        }
        !           186:                        if ((c&0177)=='\n'||c==tp->t_chr.t_eofc
        !           187:                         ||c==tp->t_chr.t_brkc) {
        !           188:                                register struct block *bp1;
        !           189:                                if (bp1 = allocb(1)) {
        !           190:                                        bp1->class |= S_DELIM;
        !           191:                                        tp->t_delct++;
        !           192:                                        putq(q, bp1);
        !           193:                                }
        !           194:                                qenable(q);
        !           195:                        }
        !           196:                }
        !           197:                if (tp->t_flags&TANDEM && (tp->t_state&TTBLOCK) == 0
        !           198:                 && q->count >= (q->qinfo->limit+q->qinfo->lolimit)/2 ) {
        !           199:                        q->next->flag |= QWANTW;
        !           200:                        tp->t_state |= TTBLOCK;
        !           201:                        putctl1d(wrq, M_DATA, tp->t_chr.t_stopc);
        !           202:                }
        !           203:                if (tp->t_flags&ECHO && (wrq->flag&QFULL)==0) {
        !           204:                        c &= 0177;
        !           205:                        if(bp->rptr == bp->wptr)
        !           206:                                putctl1d(wrq, M_DATA, c);
        !           207:                        else
        !           208:                                putd(wrq->qinfo->putp, wrq, c);
        !           209:                        if (c==tp->t_kill && (tp->t_flags&CBREAK)==0
        !           210:                         && !escape)
        !           211:                                putctl1d(wrq, M_DATA, '\n');
        !           212:                }
        !           213:        }
        !           214:        freeb(bp);
        !           215: }
        !           216: 
        !           217: /*
        !           218:  * tty input server processing.  Erase-kill and escape processing;
        !           219:  * gathering into lines.
        !           220:  */
        !           221: 
        !           222: ttyinsrv(q)
        !           223: register struct queue *q;
        !           224: {
        !           225:        register struct ttyld *tp;
        !           226:        register struct block *bp, *bp1;
        !           227: 
        !           228:        tp = (struct ttyld *)q->ptr;
        !           229:        if (q->next->flag&QFULL)
        !           230:                return;
        !           231:        if (tp->t_flags&(CBREAK|RAW)) {
        !           232:                while ((q->next->flag&QFULL)==0 && (bp = getq(q)))
        !           233:                        (*q->next->qinfo->putp)(q->next, bp);
        !           234:        } else {
        !           235:                while (tp->t_delct && q->first) {
        !           236:                        bp1 = allocb(CANBSIZ);
        !           237:                        if (bp1==NULL)
        !           238:                                return;
        !           239:                        while (bp = getq(q)) {
        !           240:                                bp1 = canonblock(q, bp, bp1, tp);
        !           241:                                if (bp1->class&S_DELIM) {
        !           242:                                        tp->t_delct--;
        !           243:                                        break;
        !           244:                                }
        !           245:                        }
        !           246:                        (*q->next->qinfo->putp)(q->next, bp1);
        !           247:                }
        !           248:        }
        !           249:        if (tp->t_flags&TANDEM && tp->t_state&TTBLOCK
        !           250:         && q->count <= q->qinfo->lolimit) {
        !           251:                tp->t_state &= ~TTBLOCK;
        !           252:                putd(putq, WR(q), tp->t_chr.t_startc);
        !           253:        }
        !           254: }
        !           255: 
        !           256: /*
        !           257:  * canonicalize bp into bp1, noticing delimiter.  bp is freed.
        !           258:  */
        !           259: struct block *
        !           260: canonblock(q, bp, bp1, tp)
        !           261: register struct queue *q;
        !           262: register struct block *bp, *bp1;
        !           263: register struct ttyld *tp;
        !           264: {
        !           265:        register c;
        !           266: 
        !           267:        while (bp->rptr<bp->wptr) {
        !           268:                if (bp1->wptr >= bp1->lim-1) {
        !           269:                        (*q->next->qinfo->putp)(q->next, bp1);
        !           270:                        bp1 = allocb(CANBSIZ);
        !           271:                }
        !           272:                c = *bp->rptr++;
        !           273:                if ((c&0200) == 0) {    /* not escaped */
        !           274:                        if (c == tp->t_erase) {
        !           275:                                if (bp1->wptr > bp1->rptr)
        !           276:                                        bp1->wptr--;
        !           277:                                continue;
        !           278:                        }
        !           279:                        if (c == tp->t_kill) {
        !           280:                                bp1->wptr = bp1->rptr;
        !           281:                                continue;
        !           282:                        }
        !           283:                        if (c == tp->t_chr.t_eofc)
        !           284:                                continue;
        !           285:                } else {
        !           286:                        c &= 0177;
        !           287:                        if (tp->t_flags&LCASE && maptab[c])
        !           288:                                c = maptab[c];
        !           289:                        else if (c==tp->t_erase || c==tp->t_kill
        !           290:                              || c==tp->t_chr.t_eofc)
        !           291:                                ;
        !           292:                        else
        !           293:                                *bp1->wptr++ = '\\';
        !           294:                }
        !           295:                *bp1->wptr++ = c;
        !           296:        }
        !           297:        if (bp->class&S_DELIM)
        !           298:                bp1->class |= S_DELIM;
        !           299:        freeb(bp);
        !           300:        return(bp1);
        !           301: }
        !           302: 
        !           303: /*
        !           304:  * TTY write processing: delays, tabs, CR/NL and the like.
        !           305:  */
        !           306: ttyosrv(q)
        !           307: register struct queue *q;
        !           308: {
        !           309:        register struct ttyld *tp;
        !           310:        register struct block *bp;
        !           311: 
        !           312:        tp = (struct ttyld *)q->ptr;
        !           313:        while (bp = getq(q)) {
        !           314:                switch(bp->type) {
        !           315: 
        !           316:                default:
        !           317:                        freeb(bp);
        !           318:                        continue;
        !           319: 
        !           320:                case M_IOCTL:
        !           321:                        if (q->next->flag & QFULL) {
        !           322:                                putbq(q, bp);
        !           323:                                return;
        !           324:                        }
        !           325:                        ttldioc(q, bp, RD(q), 0);
        !           326:                        continue;
        !           327: 
        !           328:                case M_FLUSH:
        !           329:                        flushq(q, 0);
        !           330:                case M_IOCNAK:          /* flow through */
        !           331:                case M_IOCACK:
        !           332:                        (*q->next->qinfo->putp)(q->next, bp);
        !           333:                        continue;
        !           334: 
        !           335:                case M_DATA:
        !           336:                case M_BREAK:
        !           337:                        if (q->next->flag & QFULL) {
        !           338:                                putbq(q, bp);
        !           339:                                return;
        !           340:                        }
        !           341:                        if (tp->t_flags&RAW || bp->type==M_BREAK) {
        !           342:                                (*q->next->qinfo->putp)(q->next, bp);
        !           343:                        } else
        !           344:                                outconv(q, bp);
        !           345:                        continue;
        !           346:                }
        !           347:        }
        !           348: }
        !           349: 
        !           350: outconv(q, ibp)
        !           351: struct queue *q;
        !           352: register struct block *ibp;
        !           353: {
        !           354:        register struct ttyld *tp;
        !           355:        register struct block *obp = NULL;
        !           356:        register c;
        !           357:        register count, ctype;
        !           358: 
        !           359:        tp = (struct ttyld *)q->ptr;
        !           360:    more:
        !           361:        while (ibp->rptr < ibp->wptr) {
        !           362:                if (obp==NULL || obp->wptr >= obp->lim) {
        !           363:                        if (obp)
        !           364:                                (*q->next->qinfo->putp)(q->next, obp);
        !           365:                        if (q->next->flag&QFULL || (obp=allocb(QBSIZE))==NULL) {
        !           366:                                putbq(q, ibp);
        !           367:                                return;
        !           368:                        }
        !           369:                }
        !           370:                /*
        !           371:                 * The following dance is an inner loop
        !           372:                 */
        !           373:                count = ibp->wptr - ibp->rptr;
        !           374:                if ((c = obp->lim - obp->wptr) < count)
        !           375:                        count = c;
        !           376:                while ((ctype = partab[c = *ibp->rptr++ & 0177] & 077) == 0) {
        !           377:                        tp->t_col++;
        !           378:                        *obp->wptr++ = c;
        !           379:                        if (--count <= 0)
        !           380:                                goto more;
        !           381:                }
        !           382:                if (c=='\t' && (tp->t_flags&TBDELAY)==XTABS) {
        !           383:                        for (;;) {
        !           384:                                *obp->wptr++ = ' ';
        !           385:                                tp->t_col++;
        !           386:                                if ((tp->t_col & 07) == 0)      /* every 8 */
        !           387:                                        break;
        !           388:                                if (obp->wptr >= obp->lim) {
        !           389:                                        ibp->rptr--;
        !           390:                                        break;
        !           391:                                }
        !           392:                        }
        !           393:                        continue;
        !           394:                }
        !           395: 
        !           396:                /*
        !           397:                 * turn <nl> to <cr><lf> if desired.
        !           398:                 */
        !           399:                if (c=='\n' && tp->t_flags&CRMOD) {
        !           400:                        if ((tp->t_state&TTCR)==0) {
        !           401:                                tp->t_state |= TTCR;
        !           402:                                c = '\r';
        !           403:                                ctype = partab['\r'] & 077;
        !           404:                                --ibp->rptr;
        !           405:                        } else
        !           406:                                tp->t_state &= ~TTCR;
        !           407:                }
        !           408:                /*
        !           409:                 * store character
        !           410:                 */
        !           411:                *obp->wptr++ = c;
        !           412:                /*
        !           413:                 * Calculate delays and column movement
        !           414:                 */
        !           415:                count = 0;
        !           416:                switch (ctype) {
        !           417: 
        !           418:                /* ordinary */
        !           419:                case 0:
        !           420:                        tp->t_col++;
        !           421:                        break;
        !           422:        
        !           423:                /* non-printing */
        !           424:                case 1:
        !           425:                        break;
        !           426:        
        !           427:                /* backspace */
        !           428:                case 2:
        !           429:                        if (tp->t_col)
        !           430:                                tp->t_col--;
        !           431:                        break;
        !           432:        
        !           433:                /* newline */
        !           434:                case 3:
        !           435:                        ctype = (tp->t_flags >> 8) & 03;
        !           436:                        if(ctype == 1) { /* tty 37 */
        !           437:                                if (tp->t_col)
        !           438:                                        count = max(((unsigned)tp->t_col>>4) + 3, (unsigned)6);
        !           439:                        } else if (ctype == 2)  /* vt05 */
        !           440:                                count = 6;
        !           441:                        if ((tp->t_flags&CRMOD)==0)
        !           442:                                tp->t_col = 0;
        !           443:                        break;
        !           444:        
        !           445:                /* tab */
        !           446:                case 4:
        !           447:                        ctype = (tp->t_flags >> 10) & 03;
        !           448:                        if(ctype == 1) { /* tty 37 */
        !           449:                                count = 1 - (tp->t_col | ~07);
        !           450:                                if (count < 5)
        !           451:                                        count = 0;
        !           452:                        }
        !           453:                        tp->t_col |= 07;
        !           454:                        tp->t_col++;
        !           455:                        break;
        !           456:        
        !           457:                /* vertical motion */
        !           458:                case 5:
        !           459:                        if(tp->t_flags & VTDELAY)
        !           460:                                count = 127;
        !           461:                        break;
        !           462:        
        !           463:                /* carriage return */
        !           464:                case 6:
        !           465:                        ctype = (tp->t_flags >> 12) & 03;
        !           466:                        if (ctype == 1)          /* tn 300 */
        !           467:                                count = 5;
        !           468:                        else if (ctype == 2)    /* ti 700 */
        !           469:                                count = 10;
        !           470:                        else if (ctype == 3)
        !           471:                                count = 20;
        !           472:                        tp->t_col = 0;
        !           473:                        break;
        !           474:                }
        !           475:                if (count) {
        !           476:                        (*q->next->qinfo->putp)(q->next, obp);
        !           477:                        putctl1(q->next, M_DELAY, count);
        !           478:                        obp = NULL;
        !           479:                }
        !           480:        }
        !           481:        if (obp) {
        !           482:                obp->class |= ibp->class&S_DELIM;
        !           483:                (*q->next->qinfo->putp)(q->next, obp);
        !           484:        } else if (ibp->class&S_DELIM)
        !           485:                putctld(q->next, M_DATA);
        !           486:        freeb(ibp);
        !           487: }
        !           488: 
        !           489: /*
        !           490:  * Reader generates a signal and passes it up
        !           491:  */
        !           492: ttysig(q, sig)
        !           493: register struct queue *q;
        !           494: {
        !           495:        register struct ttyld *tp = (struct ttyld *)q->ptr;
        !           496: 
        !           497:        flushq(q, 0);                   /* flush reader */
        !           498:        flushq(WR(q), 0);
        !           499:        tp->t_state &= ~TTESC;
        !           500:        tp->t_delct = 0;
        !           501:        putctl(q->next, M_FLUSH);
        !           502:        putctl1(q->next, M_SIGNAL, sig);
        !           503:        putctl(WR(q)->next, M_FLUSH);
        !           504: }
        !           505: 
        !           506: ttldioc(q, bp, rdq, fromdev)
        !           507: register struct block *bp;
        !           508: struct queue *q, *rdq;
        !           509: {
        !           510:        register struct ttyld *tp;
        !           511:        register struct sgttyb *sp;
        !           512:        int s;
        !           513: 
        !           514:        sp = (struct sgttyb *)stiodata(bp);
        !           515:        tp = (struct ttyld *)q->ptr;
        !           516:        switch (stiocom(bp)) {
        !           517: 
        !           518:        /*
        !           519:         * Set new parameters
        !           520:         */
        !           521:        case TIOCSETP:
        !           522:        case TIOCSETN:
        !           523:                s = spl6();
        !           524:                if (sp->sg_flags & (RAW|CBREAK)
        !           525:                  && (rdq->next->flag&QFULL)==0) {
        !           526:                        register struct block *bp1;
        !           527:                        ttyinsrv(rdq);
        !           528:                        while (bp1 = getq(rdq))
        !           529:                                (*rdq->next->qinfo->putp)(rdq->next, bp1);
        !           530:                }
        !           531:                tp->t_erase = sp->sg_erase;
        !           532:                tp->t_kill = sp->sg_kill;
        !           533:                tp->t_flags = sp->sg_flags;
        !           534:                splx(s);
        !           535:                bp->type = M_IOCACK;
        !           536:                if (tp->t_flags & (RAW|CBREAK))
        !           537:                        rdq->flag &= ~(QDELIM|QNOENB);
        !           538:                else
        !           539:                        rdq->flag |= QDELIM|QNOENB;
        !           540:                if (tp->t_flags & RAW && tp->t_state & TTSTOP) {
        !           541:                        putctl(q->next, M_START);       /* else it is stuck forever */
        !           542:                        tp->t_state &=~ TTSTOP;
        !           543:                }
        !           544:                break;
        !           545: 
        !           546:        /*
        !           547:         * Send current parameters to user
        !           548:         */
        !           549:        case TIOCGETP:
        !           550:                sp->sg_erase = tp->t_erase;
        !           551:                sp->sg_kill = tp->t_kill;
        !           552:                sp->sg_flags = tp->t_flags;
        !           553:                sp->sg_ispeed = sp->sg_ospeed = B9600;
        !           554:                bp->wptr = bp->rptr+sizeof(struct sgttyb)+STIOCHDR;
        !           555:                bp->type = M_IOCACK;
        !           556:                break;
        !           557: 
        !           558:        /*
        !           559:         * Set and fetch special characters
        !           560:         */
        !           561:        case TIOCSETC:
        !           562:                tp->t_chr = *(struct tchars *)((struct stioctl *)bp->rptr)->data;
        !           563:                bp->wptr = bp->rptr;
        !           564:                bp->type = M_IOCACK;
        !           565:                break;
        !           566: 
        !           567:        case TIOCGETC:
        !           568:                *(struct tchars *)((struct stioctl *)bp->rptr)->data = tp->t_chr;
        !           569:                bp->wptr = bp->rptr+sizeof(struct tchars)+STIOCHDR;
        !           570:                bp->type = M_IOCACK;
        !           571:                break;
        !           572: 
        !           573:        default:
        !           574:                if (fromdev) {
        !           575:                        bp->type = M_IOCACK;
        !           576:                        qreply(rdq, bp); /* reply to device side */
        !           577:                } else
        !           578:                        (*q->next->qinfo->putp)(q->next, bp); /* pass to device */
        !           579:                return;
        !           580: 
        !           581:        }
        !           582:        if (fromdev)
        !           583:                qreply(rdq, bp);        /* to device side */
        !           584:        else
        !           585:                qreply(q, bp);          /* to process side */
        !           586: }

unix.superglobalmegacorp.com

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