Annotation of coherent/d/PS2_KERNEL/io.386/asy.c, revision 1.1

1.1     ! root        1: /*
        !             2:  * File:       asy.c
        !             3:  *
        !             4:  * Purpose:    8250-family async port device driver
        !             5:  *
        !             6:  *     Devices are named /dev/asy[00..31]{fpl}
        !             7:  *     Minor number bit assignments:
        !             8:  *             x... ....       1 for NO modem control, "l"
        !             9:  *             .x.. ....       1 for polled operation (no irq service), "p"
        !            10:  *             ..x. ....       1 for RTS/CTS flow control, "f"
        !            11:  *             ...x xxxx       channel number - 0..31
        !            12:  *
        !            13:  * $Log:       asy.c,v $
        !            14:  * Revision 1.7  92/07/16  16:34:44  hal
        !            15:  * Kernel #58
        !            16:  * 
        !            17:  * Revision 1.6  92/07/07  09:04:30  root
        !            18:  * Allow up to 16 slots per group.
        !            19:  * Support Digiboard as fourth group type.
        !            20:  * 
        !            21:  * Revision 1.5  92/06/11  17:28:05  root
        !            22:  * Temporary fling with condev & asy_putchar.
        !            23:  * 
        !            24:  * Revision 1.4  92/06/10  17:26:59  hal
        !            25:  * Conditional logging to screen of opens and closes.  Ker #55.
        !            26:  * 
        !            27:  * Revision 1.3  92/06/06  12:39:33  hal
        !            28:  * Last before adding termio field to tty struct.
        !            29:  * 
        !            30:  */
        !            31: 
        !            32: /*
        !            33:  * -----------------------------------------------------------------
        !            34:  * Includes.
        !            35:  */
        !            36: #include <sys/coherent.h>
        !            37: #include <sys/stat.h>
        !            38: #include <sys/proc.h>
        !            39: #include <sys/tty.h>
        !            40: #include <sys/con.h>
        !            41: #include <sys/devices.h>
        !            42: #include <errno.h>
        !            43: #include <poll.h>
        !            44: #include <sys/sched.h>         /* CVTTOUT, IVTTOUT, SVTTOUT */
        !            45: #include <sys/asy.h>
        !            46: #include <sys/ins8250.h>
        !            47: #include <sys/poll_clk.h>
        !            48: #ifdef _I386
        !            49: #include <termio.h>
        !            50: #endif
        !            51: 
        !            52: /*
        !            53:  * -----------------------------------------------------------------
        !            54:  * Definitions.
        !            55:  *     Constants.
        !            56:  *     Macros with argument lists.
        !            57:  *     Typedefs.
        !            58:  *     Enums.
        !            59:  */
        !            60: #define        IEN_USE_MSI     (IE_RxI | IE_TxI | IE_LSI | IE_MSI)
        !            61: #define        IEN_NO_MSI      (IE_RxI | IE_TxI | IE_LSI)
        !            62: 
        !            63: #define CTLQ   0021
        !            64: #define CTLS   0023
        !            65: 
        !            66: #define NUM_IRQ                16      /* PC allows irq numbers 0..15          */
        !            67: #define BPB            8       /* 8 bits per byte                      */
        !            68: #define DTRTMOUT       3       /* DTR seconds for close                */
        !            69: #define LOOP_LIMIT     100     /* safety valve on irq loops            */
        !            70: 
        !            71: /*
        !            72:  * For rawin silo (see poll_clk.h), use last element of si_buf to count
        !            73:  * the number of characters in the silo.
        !            74:  */
        !            75: #define SILO_CHAR_COUNT        si_buf[SI_BUFSIZ-1]
        !            76: #define SILO_HIGH_MARK (SI_BUFSIZ-SI_BUFSIZ/4)
        !            77: #define SILO_LOW_MARK  (SI_BUFSIZ/4)
        !            78: #define MAX_SILO_INDEX (SI_BUFSIZ-2)
        !            79: #define MAX_SILO_CHARS (SI_BUFSIZ-1)
        !            80: 
        !            81: #define RAWIN_FLUSH(in_silo)   { \
        !            82:   in_silo->si_ox = in_silo->si_ix; \
        !            83:   in_silo->SILO_CHAR_COUNT = 0; }
        !            84: #define RAWOUT_FLUSH(out_silo) { out_silo->si_ox = out_silo->si_ix; }
        !            85: #define channel(dev)   (dev & 0x1F)
        !            86: 
        !            87: #define IEN    ((a0->a_nms)?IEN_NO_MSI:IEN_USE_MSI)
        !            88: #ifdef _I386
        !            89: #define        EEBUSY  EBUSY
        !            90: #else
        !            91: #define        EEBUSY  EDBUSY
        !            92: #endif
        !            93: 
        !            94: #define NW_OUTSILO     1       /* bits in need_wake[] entries          */
        !            95: 
        !            96: typedef void (* VPTR)();       /* pointer to function returning void   */
        !            97: typedef void (* FPTR)();       /* pointer to function returning int    */
        !            98: 
        !            99: /*
        !           100:  * -----------------------------------------------------------------
        !           101:  * Functions.
        !           102:  *     Import Functions.
        !           103:  *     Export Functions.
        !           104:  *     Local Functions.
        !           105:  */
        !           106: int nulldev();
        !           107: 
        !           108: void asy_putchar();
        !           109: 
        !           110: /*
        !           111:  * Configuration functions (local).
        !           112:  */
        !           113: static void asyclose();
        !           114: static void asyioctl();
        !           115: static void asyioctl0();
        !           116: static void asyload();
        !           117: static void asyopen();
        !           118: static void asyread();
        !           119: static void asytimer();
        !           120: static void asyunload();
        !           121: static void asywrite();
        !           122: static int asypoll();
        !           123: static void cinit();
        !           124: 
        !           125: /*
        !           126:  * Support functions (local).
        !           127:  */
        !           128: static void add_irq();
        !           129: static void asy_irq();
        !           130: static int asy_send();
        !           131: static void asybreak();
        !           132: static void asyclock();
        !           133: static void asycycle();
        !           134: static void asydump();
        !           135: static int asyintr();
        !           136: static void asyparam();
        !           137: static void asysph();
        !           138: static void asyspr();
        !           139: static void asystart();
        !           140: static void irqdummy();
        !           141: static void upd_irq1();
        !           142: 
        !           143: static void i2(),i3(),i4(),i5(),i6(),i7(),i8(),i9();
        !           144: static void i10(),i11(),i12(),i13(),i14(),i15();
        !           145: static int p1(),p2(),p3(),p4();
        !           146: 
        !           147: /*
        !           148:  * -----------------------------------------------------------------
        !           149:  * Global Data.
        !           150:  *     Import Variables.
        !           151:  *     Export Variables.
        !           152:  *     Local Variables.
        !           153:  */
        !           154: extern int albaud[], alp_rate[];
        !           155: 
        !           156: /*
        !           157:  * When asypatch runs, it checks whether its internal value for
        !           158:  * ASY_VERSION matches this driver's value, so as to prevent the patch
        !           159:  * utility and the driver from getting out of phase.
        !           160:  */
        !           161: int ASY_VER = ASY_VERSION;
        !           162: int ASY_HPCL = 1;
        !           163: int ASY_NUM = 0;
        !           164: int ASYGP_NUM = 0;
        !           165: asy0_t asy0[MAX_ASY] = {
        !           166:        { 0 }
        !           167: };
        !           168: asy_gp_t asy_gp[MAX_ASYGP] = {
        !           169:        { 0 }
        !           170: };
        !           171: 
        !           172: static asy1_t * asy1;          /* unused entries have type US_NONE     */
        !           173: static short dummy_port;       /* used only during driver startup      */
        !           174: static int poll_divisor;       /* set by asyspr(), read by asyclk()    */
        !           175: static char pptbl[MAX_ASY];    /* channel numbers of polled ports      */
        !           176: static int ppnum;              /* number of channels in pptbl          */
        !           177: 
        !           178: /*
        !           179:  * itbl keeps function pointers for irq service routines, for ease of setting
        !           180:  *   and clearing vectors.
        !           181:  *
        !           182:  * irq0[x] and irq1[x] are lists for irq number x.
        !           183:  * irq0 has nodes that may possibly cause an irq.
        !           184:  * irq1 contains nodes for active devices.
        !           185:  * Whenever a device becomes active or inactive, irq1 is rebuilt from irq0.
        !           186:  *
        !           187:  * nodespace is an array of available nodes used in making the lists.
        !           188:  * nextnode points to the next free node.
        !           189:  * Nodes are taken from nodespace only during driver load.
        !           190:  */
        !           191: static VPTR itbl[NUM_IRQ] = {
        !           192:        0,0,i2,i3,i4,i5,i6,i7,i8,i9,i10,i11,i12,i13,i14,i15 };
        !           193: static FPTR ptbl[PT_MAX] = { asyintr,p1,p2,p3,p4 };
        !           194: static struct irqnode *irq0[NUM_IRQ], *irq1[NUM_IRQ];
        !           195: static struct irqnode nodespace[MAX_ASY];
        !           196: static char need_wake[MAX_ASY];
        !           197: static char nextnode;
        !           198: static int initialized;        /* for asy_putchar() */
        !           199: 
        !           200: /*
        !           201:  * Configuration table (export data).
        !           202:  */
        !           203: CON asycon ={
        !           204:        DFCHR|DFPOL,                    /* Flags */
        !           205:        ASY_MAJOR,                      /* Major index */
        !           206:        asyopen,                        /* Open */
        !           207:        asyclose,                       /* Close */
        !           208:        nulldev,                        /* Block */
        !           209:        asyread,                        /* Read */
        !           210:        asywrite,                       /* Write */
        !           211: #ifdef _I386
        !           212:        asyioctl0,                      /* Ioctl */
        !           213: #else
        !           214:        asyioctl,                       /* Ioctl */
        !           215: #endif
        !           216:        nulldev,                        /* Powerfail */
        !           217:        asytimer,                       /* Timeout */
        !           218:        asyload,                        /* Load */
        !           219:        asyunload,                      /* Unload */
        !           220:        asypoll                         /* Poll */
        !           221: };
        !           222: 
        !           223: /*
        !           224:  * -----------------------------------------------------------------
        !           225:  * Code.
        !           226:  */
        !           227: 
        !           228: /*
        !           229:  * asyload()
        !           230:  */
        !           231: static void
        !           232: asyload()
        !           233: {
        !           234:        int     s, chan;
        !           235:        asy0_t  *a0;
        !           236:        asy1_t  *a1;
        !           237:        TTY     *tp;
        !           238:        short   port;
        !           239:        char    irq;
        !           240:        char    speed;
        !           241:        char    g;
        !           242:        char    sense_ct = 0;
        !           243: 
        !           244:        /*
        !           245:         * Allocate space for asy structs.  Possible error return.
        !           246:         */
        !           247:        asy1 = (asy1_t *)kalloc(ASY_NUM * sizeof(asy1_t));
        !           248:        if (asy1 == 0) {
        !           249:                printf("asyload: can't allocate space for %d async devices\n",
        !           250:                  ASY_NUM);
        !           251:                return;
        !           252:        }
        !           253:        kclear(asy1, ASY_NUM*sizeof(asy1_t));
        !           254: 
        !           255:        /*
        !           256:         * For each non-null port:
        !           257:         *      if port is uses irq
        !           258:         *              set dummy routine in case uart_sense causes bogus irpts
        !           259:         *      sense chip type
        !           260:         *      write baud rate to sgtty/termio structs
        !           261:         *      disable port interrupts
        !           262:         *      hang up port
        !           263:         *      set default baud rate (also resets UART)
        !           264:         *      hook "start" function into line discipline module
        !           265:         *      hook "param" function into line discipline module
        !           266:         *      hook CS into line discipline module
        !           267:         *      if port is uses irq
        !           268:         *              release dummy routine
        !           269:         *              if not in a port group
        !           270:         *                      add to irq list
        !           271:         */
        !           272:        for (chan = 0; chan < ASY_NUM; chan++) {
        !           273:                a0 = asy0 + chan;
        !           274:                a1 = asy1 + chan;
        !           275:                tp = &a1->a_tty;
        !           276:                speed = a0->a_speed;
        !           277:                tp->t_sgttyb.sg_ispeed = tp->t_sgttyb.sg_ospeed = speed;
        !           278:                tp->t_dispeed = tp->t_dospeed = speed;
        !           279:                port = a0->a_port;
        !           280: 
        !           281:                /*
        !           282:                 * A port address of zero means a skipped entry in the table.
        !           283:                 * In this case a1->a_ut keeps its initial value of US_NONE.
        !           284:                 */
        !           285:                if (port) {
        !           286:                        dummy_port = port;
        !           287:                        if (a0->a_irqno)
        !           288:                                setivec(a0->a_irqno, irqdummy);
        !           289:                        /*
        !           290:                         * uart_sense() prints port info.
        !           291:                         * Do this four times per line.
        !           292:                         */
        !           293:                        a1->a_ut = uart_sense(port);
        !           294:                        sense_ct++;
        !           295:                        if ((sense_ct & 1) == 0)
        !           296:                                putchar('\n');
        !           297:                        else
        !           298:                                putchar('\t');
        !           299:                        s = sphi();
        !           300:                        outb(port+MCR, 0);
        !           301:                        outb(port+LCR, LC_DLAB);
        !           302:                        outb(port+DLL, albaud[speed]);
        !           303:                        outb(port+DLH, albaud[speed] >> 8);
        !           304:                        outb(port+LCR, LC_CS8);
        !           305:                        tp->t_start = asystart;
        !           306:                        /* leave tp->t_param at 0 */
        !           307:                        tp->t_cs_sel = cs_sel();
        !           308:                        tp->t_ddp = (int *)chan;
        !           309:                        spl(s);
        !           310:                        if (a0->a_irqno) {
        !           311:                                clrivec(a0->a_irqno);
        !           312:                                if (a0->a_asy_gp == NO_ASYGP)
        !           313:                                        add_irq(a0->a_irqno, asyintr, chan);
        !           314:                        }
        !           315:                }
        !           316:        }
        !           317:        if (sense_ct & 1)
        !           318:                putchar('\n');
        !           319: 
        !           320:        /*
        !           321:         * for each port group
        !           322:         *      add group to irq list
        !           323:         */
        !           324:        for (g = 0; g < ASYGP_NUM; g++) {
        !           325:                add_irq(asy_gp[g].irq, ptbl[asy_gp[g].gp_type], g);
        !           326:        }
        !           327: 
        !           328:        /*
        !           329:         * Attach irq routines.
        !           330:         */
        !           331:        for (irq = 0; irq < NUM_IRQ; irq++)
        !           332:                if (irq0[irq]) {
        !           333:                        setivec(irq, itbl[irq]);
        !           334:                }
        !           335: }
        !           336: 
        !           337: /*
        !           338:  * asyunload()
        !           339:  */
        !           340: static void
        !           341: asyunload()
        !           342: {
        !           343:        char chan, irq;
        !           344: 
        !           345:        /*
        !           346:         * for each channel
        !           347:         *      disable UART interrupts
        !           348:         *      hangup port
        !           349:         *      cancel timer
        !           350:         */
        !           351:        for (chan = 0; chan < ASY_NUM; chan++) {
        !           352:                asy0_t * a0 = asy0 + chan;
        !           353:                asy1_t * a1 = asy1 + chan;
        !           354:                short port = a0->a_port;
        !           355:                TTY *tp = &a1->a_tty;
        !           356: 
        !           357:                outb(port+IER, 0);
        !           358:                outb(port+MCR, 0);
        !           359:                timeout(tp->t_rawtim, 0, NULL, 0);
        !           360:        }
        !           361: 
        !           362:        /*
        !           363:         * for each irq
        !           364:         *      if irq routine was attached
        !           365:         *              detach it
        !           366:         */
        !           367:        for (irq = 0; irq < NUM_IRQ; irq++)
        !           368:                if (irq0[irq])
        !           369:                        clrivec(irq);
        !           370: 
        !           371:        /*
        !           372:         * Deallocate dynamic asy storage.
        !           373:         */
        !           374:        if (asy1)
        !           375:                kfree(asy1);
        !           376: }
        !           377: 
        !           378: /*
        !           379:  * asyopen()
        !           380:  */
        !           381: static void
        !           382: asyopen(dev, mode)
        !           383: dev_t dev;
        !           384: int mode;
        !           385: {
        !           386:        int     s;
        !           387:        char    msr, mcr;
        !           388:        char    chan = channel(dev);
        !           389:        asy0_t  *a0 = asy0 + chan;
        !           390:        asy1_t  *a1 = asy1 + chan;
        !           391:        TTY     *tp = &a1->a_tty;
        !           392:        short   port = a0->a_port;
        !           393: 
        !           394:        if (a1->a_ut == US_NONE) { /* chip not found */
        !           395:                T_HAL(4, devmsg(dev, "no UART"));
        !           396:                u.u_error = ENXIO;
        !           397:                goto bad_open;
        !           398:        }
        !           399: 
        !           400:        if ((tp->t_flags & T_EXCL) && !super()) {
        !           401:                T_HAL(4, devmsg(dev, "exclusive use"));
        !           402:                u.u_error = ENODEV;
        !           403:                goto bad_open;
        !           404:        }
        !           405: 
        !           406: #if 0
        !           407:        if (drvl[major(dev)].d_time != 0) {     /* Modem settling */
        !           408:                T_HAL(4, devmsg(dev, "modem settling"));
        !           409:                u.u_error = EEBUSY;
        !           410:                goto bad_open;
        !           411:        }
        !           412: #endif
        !           413: 
        !           414:        /*
        !           415:         * Can't open for hardware flow control if modem status
        !           416:         * interrupts are disallowed.
        !           417:         */
        !           418:        if (a0->a_nms && (dev & CFLOW)) {
        !           419:                T_HAL(4, devmsg(dev, "no modem status irq's"));
        !           420:                u.u_error = ENXIO;
        !           421:                goto bad_open;
        !           422:        }
        !           423: 
        !           424:        /*
        !           425:         * Can't open a polled port if another driver is using polling.
        !           426:         */
        !           427:        if (dev & CPOLL && poll_owner & ~ POLL_ASY) {
        !           428:                T_HAL(4, devmsg(dev, "polling unavailable"));
        !           429:                u.u_error = EEBUSY;
        !           430:                goto bad_open;
        !           431:        }
        !           432: 
        !           433:        /*
        !           434:         * Can't have both com[13] or both com[24] IRQ at once.
        !           435:         */
        !           436:        if (!(dev & CPOLL) && a0->a_ixc) {
        !           437:                struct irqnode *np = irq1[a0->a_irqno];
        !           438:                while (np) {
        !           439:                        if (np->func != ptbl[0] || np->arg != chan) {
        !           440:                                T_HAL(4, devmsg(dev, "irq conflict"));
        !           441:                                u.u_error = EEBUSY;
        !           442:                                goto bad_open;
        !           443:                        }
        !           444:                        np = np->next_actv;
        !           445:                }
        !           446:        }
        !           447: 
        !           448:        /*
        !           449:         * If port already in use, are new and old open modes compatible?
        !           450:         */
        !           451:        if (a1->a_in_use) {
        !           452:                int oldmode = 0, newmode = 0; /* mctl:1 irq:2 flow:4 */
        !           453: 
        !           454:                if (a1->a_modc)
        !           455:                        oldmode += 1;
        !           456:                if (a1->a_irq)
        !           457:                        oldmode += 2;
        !           458:                if (a1->a_flc)
        !           459:                        oldmode += 4;
        !           460:                if ((dev & NMODC) == 0)
        !           461:                        newmode += 1;
        !           462:                if ((dev & CPOLL) == 0)
        !           463:                        newmode += 2;
        !           464:                if (dev & CFLOW)
        !           465:                        newmode += 4;
        !           466:                if (oldmode != newmode) {
        !           467:                        T_HAL(4, devmsg(dev, "conflicting open modes"));
        !           468:                        u.u_error = EEBUSY;
        !           469:                        goto bad_open;
        !           470:                }
        !           471:        }
        !           472: 
        !           473:        /*
        !           474:         * Sleep here if another process is opening or closing the port.
        !           475:         * This can happen if:
        !           476:         *   another process is trying a first open and awaiting CD;
        !           477:         *   another process is closing the port after losing CD;
        !           478:         *   a remote process opened the port, spawned a daemon,
        !           479:         *     and disconnected, and the daemon ignored SIGHUP and is
        !           480:         *     improperly keeping the port open.
        !           481:         * Don't try to set tp->t_flags before this sleep!  During
        !           482:         *   the sleep, ttclose() may be called and clear the flags.
        !           483:         */
        !           484:        while (a1->a_in_use && (a1->a_hcls ||
        !           485:          ((dev & NMODC) == 0 && (inb(port+MSR) & MS_RLSD) == 0))) {
        !           486:                v_sleep((char *)(&tp->t_open), CVTTOUT, IVTTOUT, SVTTOUT,
        !           487:                  "asyblk");
        !           488:                if (SELF->p_ssig && nondsig()) {  /* signal? */
        !           489:                        u.u_error = EINTR;
        !           490:                        goto bad_open;
        !           491:                }
        !           492:        }
        !           493: 
        !           494:        /*
        !           495:         * If channel not in use, mark it as such.
        !           496:         */
        !           497:        if (a1->a_in_use == 0) {
        !           498:                /*
        !           499:                 * Save modes for this open attempt to avoid future conflicts.
        !           500:                 * Then start asycycle() for this port.
        !           501:                 */
        !           502:                if (dev & NMODC) {
        !           503:                        tp->t_flags &= ~T_MODC;
        !           504:                        a1->a_modc = 0;
        !           505:                } else {
        !           506:                        tp->t_flags |= T_MODC;
        !           507:                        a1->a_modc = 1;
        !           508:                }
        !           509:                if (dev & CPOLL)
        !           510:                        a1->a_irq = 0;
        !           511:                else
        !           512:                        a1->a_irq = 1;
        !           513:                if (dev & CFLOW) {
        !           514:                        tp->t_flags |= T_CFLOW;
        !           515:                        a1->a_flc = 1;
        !           516:                } else {
        !           517:                        tp->t_flags &= ~T_CFLOW;
        !           518:                        a1->a_flc = 0;
        !           519:                }
        !           520:        }
        !           521:        a1->a_in_use++;
        !           522: 
        !           523:        /*
        !           524:         * From here, error exit is bad_open_u.
        !           525:         */
        !           526: 
        !           527:        if (tp->t_open == 0) {        /* not already open */
        !           528:                silo_t  * in_silo = &a1->a_in;
        !           529: 
        !           530:                if (!(dev & CPOLL)) {
        !           531:                        upd_irq1(a0->a_irqno);
        !           532:                        a1->a_has_irq = 1;
        !           533:                }
        !           534: 
        !           535:                /*
        !           536:                 * Need to start cycling to scan for CD.
        !           537:                 */
        !           538:                asycycle(chan);
        !           539: 
        !           540:                s = sphi();
        !           541:                /*
        !           542:                 * Raise basic modem control lines even if modem
        !           543:                 * control hasn't been specified.
        !           544:                 * MC_OUT2 turns on NON-open-collector IRQ line from the UART.
        !           545:                 * since we can't have two UART's on same IRQ with MC_OUT2 on
        !           546:                 */
        !           547:                mcr = MC_RTS | MC_DTR;
        !           548:                if (dev & CPOLL) {
        !           549:                        outb(port+MCR, mcr);
        !           550:                } else {
        !           551:                        outb(port+MCR, mcr | a0->a_outs);
        !           552:                        outb(port+IER, IEN);
        !           553:                }
        !           554: 
        !           555:                if ((dev & NMODC) == 0) {       /* want modem control? */
        !           556:                        tp->t_flags |= T_HOPEN | T_STOP;
        !           557:                        for (;;) {      /* wait for carrier */
        !           558:                                msr = inb(port+MSR);
        !           559:                                /*
        !           560:                                 * If carrier detect present
        !           561:                                 *   if port not already open
        !           562:                                 *     break out of loop and finish first open
        !           563:                                 *   else
        !           564:                                 *     do second (or third, etc.) open
        !           565:                                 */
        !           566:                                if (msr & MS_RLSD)
        !           567:                                        break;
        !           568:                                sleep((char *)(&tp->t_open), CVTTOUT, IVTTOUT,
        !           569:                                  SVTTOUT, "need CD");  /* wait for carrier */
        !           570:                                if (SELF->p_ssig && nondsig()) {  /* signal? */
        !           571:                                        outb(port+MCR, 0);
        !           572:                                        outb(port+IER, 0);
        !           573:                                        u.u_error = EINTR;
        !           574:                                        tp->t_flags &= ~(T_HOPEN | T_STOP);
        !           575:                                        spl(s);
        !           576:                                        goto bad_open_u;
        !           577:                                }
        !           578:                        }
        !           579: 
        !           580:                        /*
        !           581:                         * Mark that we are no longer hanging in open.
        !           582:                         * Allow output over the port unless hardware flow
        !           583:                         * control says not to.
        !           584:                         */
        !           585:                        tp->t_flags &= ~T_HOPEN;
        !           586:                        tp->t_flags &= ~T_STOP;
        !           587:                        if (!(tp->t_flags & T_CFLOW) || (msr & MS_CTS))
        !           588:                                a1->a_ohlt = 0;
        !           589:                        else
        !           590:                                a1->a_ohlt = 1;
        !           591: 
        !           592:                        /*
        !           593:                         * Awaken any other opens on same device.
        !           594:                         */
        !           595:                        wakeup((char *)(&tp->t_open));
        !           596:                }
        !           597:                ttopen(tp);                             /* stty inits */
        !           598:                tp->t_flags |= T_CARR;
        !           599:                if (ASY_HPCL)
        !           600:                        tp->t_flags |= T_HPCL;
        !           601: 
        !           602:                asyparam(tp);   /* gimmick: do this while t_open is zero */
        !           603: 
        !           604:                /*
        !           605:                 * TO DO: flush UART input register(s).
        !           606:                 */
        !           607: 
        !           608:                spl(s);
        !           609: 
        !           610:                /*
        !           611:                 * Turn on polling for the port.
        !           612:                 */
        !           613:                if (dev & CPOLL) {
        !           614:                        a1->a_poll = 1;
        !           615:                        asyspr();
        !           616:                }
        !           617:        } /* end of first-open case */
        !           618: 
        !           619:        tp->t_open++;
        !           620:        T_HAL(0x400, printf("ch%d open + %d\n", chan, tp->t_open));
        !           621:        ttsetgrp(tp, dev, mode);
        !           622: 
        !           623:        return;
        !           624: 
        !           625: bad_open_u:
        !           626:        a1->a_in_use--;
        !           627:        wakeup((char *)(&tp->t_open));
        !           628: bad_open:
        !           629:        return;
        !           630: }
        !           631: 
        !           632: /*
        !           633:  * asyclose()
        !           634:  */
        !           635: static void
        !           636: asyclose(dev, mode)
        !           637: dev_t dev;
        !           638: int mode;
        !           639: {
        !           640:        int     chan = channel(dev);
        !           641:        asy0_t  *a0 = asy0 + chan;
        !           642:        asy1_t  *a1 = asy1 + chan;
        !           643:        TTY     *tp = &a1->a_tty;
        !           644:        silo_t  * out_silo = &a1->a_out;
        !           645:        silo_t  * in_silo = &a1->a_in;
        !           646:        int     flags, maj;
        !           647:        int     s;
        !           648:        short   port = a0->a_port;
        !           649:        char    lsr;
        !           650: 
        !           651:        if (--tp->t_open)
        !           652:                goto not_last_close;
        !           653:        T_HAL(0x400, printf("ch%d open - %d\n", chan, tp->t_open));
        !           654:        s = sphi();
        !           655: 
        !           656:        a1->a_hcls = 1;                 /* disallow reopen til done closing */
        !           657:        flags = tp->t_flags;            /* save flags - ttclose zeroes them */
        !           658: #if 1
        !           659:        ttclose(tp);
        !           660: #endif
        !           661: 
        !           662:        /*
        !           663:         * Wait for output silo and UART xmit buffer to empty.
        !           664:         * Allow signal to break the sleep.
        !           665:         */
        !           666:        for (;;) {
        !           667:                lsr = inb(port + LSR);
        !           668:                if ((lsr & LS_TxIDLE)
        !           669:                  && (out_silo->si_ix == out_silo->si_ox))
        !           670:                        break;
        !           671:                need_wake[chan] |= NW_OUTSILO;
        !           672:                v_sleep((char *)out_silo, CVTTOUT, IVTTOUT, SVTTOUT,
        !           673:                  "asyclose");
        !           674:                if (SELF->p_ssig && nondsig()) {  /* signal? */
        !           675:                        RAWOUT_FLUSH(out_silo);
        !           676:                        break;
        !           677:                }
        !           678:        }
        !           679:        need_wake[chan] &= ~NW_OUTSILO;
        !           680: 
        !           681:        /*
        !           682:         * If not hanging in open
        !           683:         */
        !           684:        if ((flags & T_HOPEN) == 0) {
        !           685:                /*
        !           686:                 * Disable interrupts.
        !           687:                 */
        !           688:                outb(port+IER, 0);
        !           689:                outb(port+MCR, inb(port+MCR) & ~MC_OUTS);
        !           690:        }
        !           691: 
        !           692:        /*
        !           693:         * If hupcls
        !           694:         */
        !           695:        if (flags & T_HPCL) {
        !           696:                T_HAL(0x400, printf("ch%d drop DTR\n", chan));
        !           697:                /*
        !           698:                 * Hangup port - drop DTR and RTS.
        !           699:                 */
        !           700:                outb(port+MCR, inb(port+MCR) & MC_OUTS);
        !           701: 
        !           702:                /*
        !           703:                 * Hold dtr low for timeout
        !           704:                 */
        !           705:                maj = major(dev);
        !           706:                drvl[maj].d_time = 1;
        !           707:                v_sleep((char *)&drvl[maj].d_time, CVTTOUT, IVTTOUT, SVTTOUT,
        !           708:                  "drop DTR");
        !           709:                drvl[maj].d_time = 0;
        !           710:        }
        !           711: 
        !           712:        a1->a_poll = 0;
        !           713:        asyspr();
        !           714:        RAWIN_FLUSH(in_silo);
        !           715:        a1->a_hcls = 0;         /* allow reopen - done closing */
        !           716:        wakeup((char *)(&tp->t_open));
        !           717:        spl(s);
        !           718:        a1->a_in_use--;
        !           719: 
        !           720:        if (!(dev & CPOLL))
        !           721:                upd_irq1(a0->a_irqno);
        !           722:        return;
        !           723: 
        !           724: not_last_close:
        !           725:        T_HAL(0x400, printf("ch%d open - %d\n", chan, tp->t_open));
        !           726:        a1->a_in_use--;
        !           727:        wakeup((char *)(&tp->t_open));
        !           728:        return;
        !           729: }
        !           730: 
        !           731: /*
        !           732:  * asyread()
        !           733:  */
        !           734: static void
        !           735: asyread(dev, iop)
        !           736: dev_t dev;
        !           737: register IO * iop;
        !           738: {
        !           739:        int     chan = channel(dev);
        !           740:        asy1_t  *a1 = asy1 + chan;
        !           741:        TTY     *tp = &a1->a_tty;
        !           742: 
        !           743:        ttread(tp, iop);
        !           744: }
        !           745: 
        !           746: /*
        !           747:  * asytimer()
        !           748:  */
        !           749: static void
        !           750: asytimer(dev)
        !           751: dev_t dev;
        !           752: {
        !           753:        if (++drvl[major(dev)].d_time > DTRTMOUT)
        !           754:                wakeup((char *)&drvl[major(dev)].d_time);
        !           755: }
        !           756: 
        !           757: /*
        !           758:  * asywrite()
        !           759:  */
        !           760: static void
        !           761: asywrite(dev, iop)
        !           762: dev_t dev;
        !           763: register IO * iop;
        !           764: {
        !           765:        int     chan = channel(dev);
        !           766:        asy0_t  *a0 = asy0 + chan;
        !           767:        asy1_t  *a1 = asy1 + chan;
        !           768:        TTY     *tp = &a1->a_tty;
        !           769:        short   port = a0->a_port;
        !           770:        register int c;
        !           771: 
        !           772:        /*
        !           773:         * Treat user writes through tty driver.
        !           774:         */
        !           775:        if (iop->io_seg != IOSYS) {
        !           776:                ttwrite(tp, iop);
        !           777:                return;
        !           778:        }
        !           779: 
        !           780:        /*
        !           781:         * Treat kernel writes by blocking on transmit buffer.
        !           782:         */
        !           783:        while ((c = iogetc(iop)) >= 0) {
        !           784:                /*
        !           785:                 * Wait until transmit buffer is empty.
        !           786:                 * Check twice to prevent critical race with interrupt handler.
        !           787:                 */
        !           788:                for (;;) {
        !           789:                        if (inb(port+LSR) & LS_TxRDY)
        !           790:                                if (inb(port+LSR) & LS_TxRDY)
        !           791:                                        break;
        !           792:                }
        !           793: 
        !           794:                /*
        !           795:                 * Output the next character.
        !           796:                 */
        !           797:                outb(port+DREG, c);
        !           798:        }
        !           799: }
        !           800: 
        !           801: /*
        !           802:  * asyioctl()
        !           803:  */
        !           804: #ifdef _I386
        !           805: static void
        !           806: asyioctl0(dev, com, vec)
        !           807: dev_t dev;
        !           808: int com;
        !           809: struct sgttyb *vec;
        !           810: {
        !           811:        tioc286(dev, com, vec, asyioctl);
        !           812: }
        !           813: #endif
        !           814: 
        !           815: static void
        !           816: asyioctl(dev, com, vec)
        !           817: dev_t  dev;
        !           818: int    com; struct sgttyb *vec;
        !           819: {
        !           820:        int     chan = channel(dev);
        !           821:        asy0_t  *a0 = asy0 + chan;
        !           822:        asy1_t  *a1 = asy1 + chan;
        !           823:        TTY     *tp = &a1->a_tty;
        !           824:        int     s;
        !           825:        int     stat1, stat2;
        !           826:        silo_t  *out_silo = &a1->a_out;
        !           827:        silo_t  *in_silo = &a1->a_in;
        !           828:        short   port = a0->a_port;
        !           829:        char    msr;
        !           830:        char    ier_save;
        !           831:        char    do_ttioctl = 0;
        !           832:        char    do_asyparam = 0;
        !           833: 
        !           834:        s = sphi();
        !           835:        ier_save = inb(port+IER);
        !           836:        stat1 = inb(port+MCR);          /* get current MCR register status */
        !           837:        stat2 = inb(port+LCR);          /* get current LCR register status */
        !           838: 
        !           839: #if 0
        !           840:        /*
        !           841:         * If command will drain input, do the drain now
        !           842:         * before calling ttioctl().
        !           843:         */
        !           844:        switch(com) {
        !           845:        case TCSETAW:
        !           846:        case TCSETAF:
        !           847:        case TIOCSETP:
        !           848:                /*
        !           849:                 * Wait for output silo and UART xmit buffer to empty.
        !           850:                 * Allow signal to break the sleep.
        !           851:                 */
        !           852:                for (;;) {
        !           853:                        if (!ttoutp(tp)
        !           854:                          && (out_silo->si_ix == out_silo->si_ox)
        !           855:                          && (inb(port + LSR) & LS_TxIDLE))
        !           856:                                break;
        !           857:                        need_wake[chan] |= NW_OUTSILO;
        !           858:                        v_sleep((char *)out_silo, CVTTOUT, IVTTOUT, SVTTOUT,
        !           859:                          "asydrain");
        !           860:                        if (SELF->p_ssig && nondsig()) {  /* signal? */
        !           861:                                break;
        !           862:                        }
        !           863:                }
        !           864:                need_wake[chan] &= ~NW_OUTSILO;
        !           865:        }
        !           866: #endif
        !           867: 
        !           868:        switch(com) {
        !           869:        case TIOCSBRK:                  /* set BREAK */
        !           870:                outb(port+LCR, stat2|LC_SBRK);
        !           871:                break;
        !           872:        case TIOCCBRK:                  /* clear BREAK */
        !           873:                outb(port+LCR, stat2 & ~LC_SBRK);
        !           874:                break;
        !           875:        case TIOCSDTR:                  /* set DTR */
        !           876:                outb(port+MCR, stat1|MC_DTR);
        !           877:                break;
        !           878:        case TIOCCDTR:                  /* clear DTR */
        !           879:                outb(port+MCR, stat1 & ~MC_DTR);
        !           880:                break;
        !           881:        case TIOCSRTS:                  /* set RTS */
        !           882:                outb(port+MCR, stat1|MC_RTS);
        !           883:                break;
        !           884:        case TIOCCRTS:                  /* clear RTS */
        !           885:                outb(port+MCR, stat1 & ~MC_RTS);
        !           886:                break;
        !           887:        case TIOCRSPEED:                /* set "raw" I/O speed divisor */
        !           888:                outb(port+LCR, stat2|LC_DLAB);  /* set speed latch bit */
        !           889:                outb(port+DLL, (unsigned) vec);
        !           890:                outb(port+DLH, (unsigned) vec >> 8);
        !           891:                outb(port+LCR, stat2);       /* reset latch bit */
        !           892:                break;
        !           893:        case TIOCWORDL:         /* set word length and stop bits */
        !           894:                outb(port+LCR, ((stat2&~0x7) | ((unsigned) vec & 0x7)));
        !           895:                break;
        !           896:        case TIOCRMSR:          /* get CTS/DSR/RI/RLSD (MSR) */
        !           897:                msr = inb(port+MSR);
        !           898:                stat1 = msr >> 4;
        !           899:                kucopy(&stat1, (unsigned *) vec, sizeof(unsigned));
        !           900:                break;
        !           901:        case TIOCFLUSH:         /* Flush silos here, queues in tty.c */
        !           902:                RAWIN_FLUSH(in_silo);
        !           903:                RAWOUT_FLUSH(out_silo);
        !           904:                do_ttioctl = 1;
        !           905:                break;
        !           906: 
        !           907:                /*
        !           908:                 * If port parameters change, plan to call asyparam().
        !           909:                 * Need to check now before structs are updated.
        !           910:                 */
        !           911: #ifdef _I386
        !           912:        case TCSETA:
        !           913:        case TCSETAW:
        !           914:        case TCSETAF:
        !           915:                {
        !           916:                        struct  termio  trm;
        !           917: 
        !           918:                        ukcopy(vec, &trm, sizeof(struct termio));
        !           919:                        if (trm.c_cflag != tp->t_termio.c_cflag)
        !           920:                                do_asyparam = 1;
        !           921:                }
        !           922:                do_ttioctl = 1;
        !           923:                break;
        !           924: #endif
        !           925:        case TIOCSETP:
        !           926:        case TIOCSETN:
        !           927:                {
        !           928:                        struct  sgttyb  sg;
        !           929: 
        !           930:                        ukcopy(vec, &sg, sizeof(struct sgttyb));
        !           931:                        if (sg.sg_ispeed != tp->t_sgttyb.sg_ispeed
        !           932:                          || ((sg.sg_flags ^ tp->t_sgttyb.sg_flags) & ANYP))
        !           933:                                do_asyparam = 1;
        !           934:                }
        !           935:                do_ttioctl = 1;
        !           936:                break;
        !           937:        default:
        !           938:                do_ttioctl = 1;
        !           939:        }
        !           940:        outb(port+IER, ier_save);
        !           941:        if (do_ttioctl)
        !           942:                ttioctl(tp, com, vec);
        !           943:        spl(s);
        !           944:        if (do_asyparam)
        !           945:                asyparam(tp);
        !           946: }
        !           947: 
        !           948: /*
        !           949:  * asyparam()
        !           950:  */
        !           951: static void
        !           952: asyparam(tp)
        !           953: TTY * tp;
        !           954: {
        !           955:        int     chan = (int)tp->t_ddp;
        !           956:        asy0_t  *a0 = asy0 + chan;
        !           957:        asy1_t  *a1 = asy1 + chan;
        !           958:        short   port = a0->a_port;
        !           959:        int     s;
        !           960:        int     write_baud=1, write_lcr=1;
        !           961:        char    newlcr, speed;
        !           962: 
        !           963: #ifdef _I386
        !           964:        unsigned short cflag = tp->t_termio.c_cflag;
        !           965: 
        !           966:        T_HAL(4, printf("ch%d asyparam cflag=%x\n", chan, cflag));
        !           967:        speed = cflag & CBAUD;
        !           968:        switch (cflag & CSIZE) {
        !           969:        case CS5:  newlcr = LC_CS5;  break;
        !           970:        case CS6:  newlcr = LC_CS6;  break;
        !           971:        case CS7:  newlcr = LC_CS7;  break;
        !           972:        case CS8:  newlcr = LC_CS8;  break;
        !           973:        }
        !           974:        if (cflag & CSTOPB)
        !           975:                newlcr |= LC_STOPB;
        !           976:        if (cflag & PARENB) {
        !           977:                newlcr |= LC_PARENB;
        !           978:                if ((cflag & PARODD) == 0)
        !           979:                        newlcr |= LC_PAREVEN;
        !           980:        }
        !           981: #else
        !           982:        speed = tp->t_sgttyb.sg_ispeed;
        !           983:        switch (tp->t_sgttyb.sg_flags & (EVENP|ODDP|RAW)) {
        !           984:        case ODDP:
        !           985:                newlcr = LC_CS7|LC_PARENB;
        !           986:                break;
        !           987:        case EVENP:
        !           988:                newlcr = LC_CS7|LC_PARENB|LC_PAREVEN;
        !           989:                break;
        !           990:        default:
        !           991:                newlcr = LC_CS8;
        !           992:                break;
        !           993:        }
        !           994: #endif
        !           995: 
        !           996:        /*
        !           997:         * Don't bang on the UART needlessly.
        !           998:         * Writing baud rate resets the port, which loses characters.
        !           999:         * You want this on first open, NOT on later opens.
        !          1000:         */
        !          1001:        if (speed == a0->a_speed && tp->t_open) {
        !          1002:                write_baud = 0;
        !          1003:                if (newlcr == a1->a_lcr) {
        !          1004:                        write_lcr = 0;
        !          1005:                }
        !          1006:        }
        !          1007:        a0->a_speed = speed;
        !          1008:        a1->a_lcr = newlcr;
        !          1009: 
        !          1010:        if (write_lcr) {
        !          1011:                char ier_save;
        !          1012:                s = sphi();
        !          1013:                ier_save = inb(port+IER);
        !          1014:                if (write_baud) {
        !          1015:                        short divisor = albaud[speed];
        !          1016: 
        !          1017:                        T_HAL(4, printf("CH%d speed=%x\n", chan, speed));
        !          1018:                        outb(port+LCR, LC_DLAB);
        !          1019:                        outb(port+DLL, divisor);
        !          1020:                        outb(port+DLH, divisor >> 8);
        !          1021:                }
        !          1022:                T_HAL(4, printf("CH%d newlcr=%x\n", chan, newlcr));
        !          1023:                outb(port+LCR, newlcr);
        !          1024:                if (a1->a_ut == US_16550A)
        !          1025:                        outb(port+FCR, FC_ENABLE | FC_Rx_RST | FC_Rx_08);
        !          1026:                outb(port+IER, ier_save);
        !          1027:                spl(s);
        !          1028:        }
        !          1029:        if (write_baud)
        !          1030:                asyspr();
        !          1031: }
        !          1032: 
        !          1033: /*
        !          1034:  * asystart()
        !          1035:  */
        !          1036: static void
        !          1037: asystart(tp)
        !          1038: TTY * tp;
        !          1039: {
        !          1040:        int     chan = (int)tp->t_ddp;
        !          1041:        asy0_t  *a0 = asy0 + chan;
        !          1042:        asy1_t  *a1 = asy1 + chan;
        !          1043:        short   port = a0->a_port;
        !          1044:        int     s;
        !          1045:        int     need_xmit = 1;  /* True if should start sending data now. */
        !          1046:        silo_t  *out_silo = &a1->a_out;
        !          1047:        char    lsr;
        !          1048: 
        !          1049:        /*
        !          1050:         * Read line status register AFTER disabling interrupts.
        !          1051:         */
        !          1052:        s = sphi();
        !          1053:        lsr = inb(port + LSR);
        !          1054: 
        !          1055:        /*
        !          1056:         * Process break indication.
        !          1057:         * NOTE: Break indication cleared when line status register was read.
        !          1058:         */
        !          1059:        if (lsr & LS_BREAK)
        !          1060:                defer(asybreak, chan);
        !          1061: 
        !          1062:        /*
        !          1063:         * If no output data, it may be time to finish closing the port;
        !          1064:         * but won't need another xmit interrupt.
        !          1065:         */
        !          1066:        if (out_silo->si_ix == out_silo->si_ox) {
        !          1067:                if (need_wake[chan] & NW_OUTSILO) {
        !          1068:                        need_wake[chan] &= ~NW_OUTSILO;
        !          1069:                        wakeup((char *)out_silo);
        !          1070:                }
        !          1071:                need_xmit = 0;
        !          1072:        }
        !          1073: 
        !          1074:        /*
        !          1075:         * Do nothing if output is stopped.
        !          1076:         */
        !          1077:        if (tp->t_flags & T_STOP)
        !          1078:                need_xmit = 0;
        !          1079:        if (a1->a_ohlt)
        !          1080:                need_xmit = 0;
        !          1081: 
        !          1082:        /*
        !          1083:         * Start data transmission by writing to UART xmit reg.
        !          1084:         */
        !          1085:        if ((lsr & LS_TxRDY) && need_xmit) {
        !          1086:                int xmit_count;
        !          1087:                xmit_count = (a1->a_ut == US_16550A)?16:1;
        !          1088:                asy_send(out_silo, port+DREG, xmit_count);
        !          1089:        }
        !          1090:        spl(s);
        !          1091: }
        !          1092: 
        !          1093: /*
        !          1094:  * asypoll()
        !          1095:  */
        !          1096: static int
        !          1097: asypoll(dev, ev, msec)
        !          1098: dev_t dev;
        !          1099: int ev;
        !          1100: int msec;
        !          1101: {
        !          1102:        int     chan = channel(dev);
        !          1103:        asy1_t  *a1 = asy1 + chan;
        !          1104:        TTY     *tp = &a1->a_tty;
        !          1105: 
        !          1106:        return ttpoll(tp, ev, msec);
        !          1107: }
        !          1108: 
        !          1109: /*
        !          1110:  * asycycle()
        !          1111:  *
        !          1112:  * Do a wakeup of any sleeping asy's at regular intervals.
        !          1113:  */
        !          1114: static void
        !          1115: asycycle(chan)
        !          1116: int chan;
        !          1117: {
        !          1118:        asy0_t  *a0 = asy0 + chan;
        !          1119:        asy1_t  *a1 = asy1 + chan;
        !          1120:        TTY     *tp = &a1->a_tty;
        !          1121:        short   port = a0->a_port;
        !          1122:        int     s;
        !          1123:        char    msr, mcr;
        !          1124:        silo_t  *out_silo = &a1->a_out;
        !          1125:        silo_t  *in_silo = &a1->a_in;
        !          1126:        int     n, ch;
        !          1127:        int     do_start = 1;
        !          1128:        unsigned char   iir;
        !          1129: 
        !          1130:        /*
        !          1131:         * Check Carrier Detect (RLSD).
        !          1132:         *
        !          1133:         * Modem status interrupts were not enabled due to 8250 hardware bug.
        !          1134:         * Enabling modem status and receive interrupts may cause lockup
        !          1135:         * on older parts.
        !          1136:         */
        !          1137:        if (tp->t_flags & T_MODC) {
        !          1138: 
        !          1139:                /*
        !          1140:                 * Get status
        !          1141:                 */
        !          1142:                msr = inb(port+MSR);
        !          1143: 
        !          1144:                /*
        !          1145:                 * Carrier changed.
        !          1146:                 */
        !          1147:                if ((msr & MS_RLSD) && !(tp->t_flags & T_CARR)) {
        !          1148:                        /*
        !          1149:                         * Carrier is on - wakeup open.
        !          1150:                         */
        !          1151:                        s = sphi();
        !          1152:                        tp->t_flags |= T_CARR;
        !          1153:                        spl(s);
        !          1154:                        wakeup((char *)(&tp->t_open));
        !          1155:                }
        !          1156: 
        !          1157:                if (!(msr & MS_RLSD) && (tp->t_flags & T_CARR)) {
        !          1158:                        s = sphi();
        !          1159:                        RAWIN_FLUSH(in_silo);
        !          1160:                        RAWOUT_FLUSH(out_silo);
        !          1161:                        tp->t_flags &= ~T_CARR;
        !          1162:                        spl(s);
        !          1163:                        tthup(tp);
        !          1164:                }
        !          1165:        }
        !          1166: 
        !          1167:        /*
        !          1168:         * Empty raw input buffer.
        !          1169:         *
        !          1170:         * The line discipline module (tty.c) will set T_ISTOP true when the
        !          1171:         * tt input queue is nearly full (tp->t_iq.cq_cc >= IHILIM), and make
        !          1172:         * T_ISTOP false when it's ready for more input.
        !          1173:         *
        !          1174:         * When T_ISTOP is true, ttin() simply discards the character passed.
        !          1175:         */
        !          1176:        if (!(tp->t_flags & T_ISTOP)) {
        !          1177:                while (in_silo->SILO_CHAR_COUNT > 0) {
        !          1178:                        s = sphi();
        !          1179:                        ttin(tp, in_silo->si_buf[in_silo->si_ox]);
        !          1180:                        if (in_silo->si_ox < MAX_SILO_INDEX)
        !          1181:                                in_silo->si_ox++;
        !          1182:                        else
        !          1183:                                in_silo->si_ox = 0;
        !          1184:                        in_silo->SILO_CHAR_COUNT--;
        !          1185:                        spl(s);
        !          1186:                }
        !          1187:        }
        !          1188: 
        !          1189:        /*
        !          1190:         * Hardware flow control.
        !          1191:         *      Check CTS to see if we need to halt output.
        !          1192:         *      (MS_INTR should have done this - repeat code here to be sure)
        !          1193:         *      Check input silo to see if we need to raise RTS.
        !          1194:         */
        !          1195:        if (tp->t_flags & T_CFLOW) {
        !          1196: 
        !          1197:                /*
        !          1198:                 * Get status
        !          1199:                 */
        !          1200:                msr = inb(port+MSR);
        !          1201:                s = sphi();
        !          1202:                if (msr & MS_CTS)
        !          1203:                        a1->a_ohlt = 0;
        !          1204:                else
        !          1205:                        a1->a_ohlt = 1;
        !          1206:                spl(s);
        !          1207: T_HAL(4, {static cts = 0; if (!cts && (msr & MS_CTS)) { cts = 1; putchar('[');\
        !          1208: } else if (cts && !(msr & MS_CTS)) { cts = 0; putchar(']'); }});
        !          1209: 
        !          1210:                /*
        !          1211:                 * If using hardware flow control, see if we need to drop RTS.
        !          1212:                 */
        !          1213:                if ((tp->t_flags & T_CFLOW)
        !          1214:                && (in_silo->SILO_CHAR_COUNT > SILO_HIGH_MARK)) {
        !          1215:                        s = sphi();
        !          1216:                        mcr = inb(port+MCR);
        !          1217:                        if (mcr & MC_RTS) {
        !          1218:                                outb(port+MCR, mcr & ~MC_RTS);
        !          1219:                                T_HAL(4, putchar('-'));
        !          1220:                        }
        !          1221:                        spl(s);
        !          1222:                }
        !          1223: 
        !          1224:                /*
        !          1225:                 * If input silo below low mark, assert RTS.
        !          1226:                 */
        !          1227:                if (in_silo->SILO_CHAR_COUNT <= SILO_LOW_MARK) {
        !          1228:                        s = sphi();
        !          1229:                        mcr = inb(port+MCR);
        !          1230:                        if ((mcr & MC_RTS) == 0) {
        !          1231:                                outb(port+MCR, mcr | MC_RTS);
        !          1232:                                T_HAL(4, putchar('+'));
        !          1233:                        }
        !          1234:                        spl(s);
        !          1235:                }
        !          1236:        }
        !          1237: 
        !          1238:        /*
        !          1239:         * Calculate free output slot count.
        !          1240:         */
        !          1241:        n  = sizeof(out_silo->si_buf) - 1;
        !          1242:        n += out_silo->si_ox - out_silo->si_ix;
        !          1243:        n %= sizeof(out_silo->si_buf);
        !          1244: 
        !          1245:        /*
        !          1246:         * Fill raw output buffer.
        !          1247:         */
        !          1248:        for (;;) {
        !          1249:                if (--n < 0)
        !          1250:                        break;
        !          1251:                s = sphi();
        !          1252:                ch = ttout(tp);
        !          1253:                spl(s);
        !          1254:                if (ch < 0)
        !          1255:                        break;
        !          1256: 
        !          1257:                s = sphi();
        !          1258:                out_silo->si_buf[out_silo->si_ix] = ch;
        !          1259:                if (out_silo->si_ix >= sizeof(out_silo->si_buf) - 1)
        !          1260:                        out_silo->si_ix = 0;
        !          1261:                else
        !          1262:                        out_silo->si_ix++;
        !          1263:                spl(s);
        !          1264:        }
        !          1265: 
        !          1266: #ifdef _I386
        !          1267:        /*
        !          1268:         * if port has an interrupt pending (probably missed an irq)
        !          1269:         *      the following two loops should not be merged
        !          1270:         *      - need ALL port irq's inactive at once
        !          1271:         *      for each port on this irq line (use irq1 for this)
        !          1272:         *              disable interrupts (clear IER)
        !          1273:         *      for each port on this irq line
        !          1274:         *              restore interrupts
        !          1275:         */
        !          1276:        if (a1->a_has_irq && ((iir=inb(port+IIR)) & 1) == 0) {
        !          1277:                struct irqnode  *ip;
        !          1278:                asy_gp_t        *gp;
        !          1279:                int     s;
        !          1280:                short   p;
        !          1281:                char    c, slot;
        !          1282: 
        !          1283:                T_HAL(4, printf("CH%d missed iir:x\n", chan, iir));
        !          1284: 
        !          1285:                do_start = 0;
        !          1286:                s = sphi();
        !          1287:                ip = irq1[a0->a_irqno];
        !          1288:                while(ip) {
        !          1289:                        if (ip->func == asyintr) {
        !          1290:                                p = ip->arg;
        !          1291:                                outb(p + IER, 0);
        !          1292:                        } else {
        !          1293:                                gp = asy_gp + ip->arg;
        !          1294:                                for (slot = 0; slot < MAX_SLOTS; slot++) {
        !          1295:                                        if ((c=gp->chan_list[slot]) < MAX_ASY){
        !          1296:                                                p = asy0[c].a_port;
        !          1297:                                                outb(p + IER, 0);
        !          1298:                                        }
        !          1299:                                }
        !          1300:                        }
        !          1301:                        ip = ip->next_actv;
        !          1302:                }
        !          1303:                /*
        !          1304:                 * Now, all ports on the offending irq line have irq off.
        !          1305:                 */
        !          1306:                ip = irq1[a0->a_irqno];
        !          1307:                while(ip) {
        !          1308:                        if (ip->func == asyintr) {
        !          1309:                                p = ip->arg;
        !          1310:                                outb(p + IER, IEN);
        !          1311:                        } else {
        !          1312:                                gp = asy_gp + ip->arg;
        !          1313:                                for (slot = 0; slot < MAX_SLOTS; slot++) {
        !          1314:                                        if ((c=gp->chan_list[slot]) < MAX_ASY){
        !          1315:                                                p = asy0[c].a_port;
        !          1316:                                                outb(p + IER, IEN);
        !          1317:                                        }
        !          1318:                                }
        !          1319:                        }
        !          1320:                        ip = ip->next_actv;
        !          1321:                }
        !          1322:                spl(s);
        !          1323:        }
        !          1324: #endif
        !          1325: 
        !          1326:        if(do_start)
        !          1327:                ttstart(tp);
        !          1328: 
        !          1329:        /*
        !          1330:         * Schedule next cycle.
        !          1331:         */
        !          1332:        if (a1->a_in_use) {
        !          1333:                timeout(&tp->t_rawtim, HZ/10, asycycle, chan);
        !          1334:        }
        !          1335: }
        !          1336: 
        !          1337: /*
        !          1338:  * irqdummy()
        !          1339:  *
        !          1340:  * Suppress interrupts that may occur during chip sensing.
        !          1341:  */
        !          1342: static void
        !          1343: irqdummy()
        !          1344: {
        !          1345:        /*
        !          1346:         * Try to clear all pending interrupts.
        !          1347:         */
        !          1348:        inb(dummy_port+IIR);
        !          1349:        inb(dummy_port+LSR);
        !          1350:        inb(dummy_port+MSR);
        !          1351:        inb(dummy_port+DREG);
        !          1352: }
        !          1353: 
        !          1354: /*
        !          1355:  * add_irq()
        !          1356:  *
        !          1357:  * Given channel number, add port info to irq0 list.
        !          1358:  */
        !          1359: static void
        !          1360: add_irq(irq, func, arg)
        !          1361: int irq;
        !          1362: void (*func)();
        !          1363: int arg;
        !          1364: {
        !          1365:        struct irqnode * np;
        !          1366: 
        !          1367:        /*
        !          1368:         * Sanity check.
        !          1369:         */
        !          1370:        if (irq <=0 || irq >= NUM_IRQ || itbl[irq] == 0)
        !          1371:                return;
        !          1372: 
        !          1373:        if (nextnode < MAX_ASY) {
        !          1374:                np = nodespace + nextnode++;
        !          1375:                np->func = func;
        !          1376:                np->arg = arg;
        !          1377:                np->next = irq0[irq];
        !          1378:                irq0[irq] = np;
        !          1379:        } else {
        !          1380:                printf("asy: too many irq nodes (%d)\n", nextnode);
        !          1381:        }
        !          1382: }
        !          1383: 
        !          1384: /*
        !          1385:  * Interrupt handlers.
        !          1386:  */
        !          1387: static void i2() { asy_irq(irq1[2]); }
        !          1388: static void i3() { asy_irq(irq1[3]); }
        !          1389: static void i4() { asy_irq(irq1[4]); }
        !          1390: static void i5() { asy_irq(irq1[5]); }
        !          1391: static void i6() { asy_irq(irq1[6]); }
        !          1392: static void i7() { asy_irq(irq1[7]); }
        !          1393: static void i8() { asy_irq(irq1[8]); }
        !          1394: static void i9() { asy_irq(irq1[9]); }
        !          1395: static void i10() { asy_irq(irq1[10]); }
        !          1396: static void i11() { asy_irq(irq1[11]); }
        !          1397: static void i12() { asy_irq(irq1[12]); }
        !          1398: static void i13() { asy_irq(irq1[13]); }
        !          1399: static void i14() { asy_irq(irq1[14]); }
        !          1400: static void i15() { asy_irq(irq1[15]); }
        !          1401: 
        !          1402: /*
        !          1403:  * asy_irq()
        !          1404:  *
        !          1405:  * Given pointer to node list, service async interrupt.
        !          1406:  */
        !          1407: static void
        !          1408: asy_irq(ip)
        !          1409: struct irqnode * ip;
        !          1410: {
        !          1411:        struct irqnode  *here;
        !          1412:        int             doit;
        !          1413: 
        !          1414:        do {
        !          1415:                doit = 0;
        !          1416:                here = ip;
        !          1417:                while(here) {
        !          1418:                        doit |= (*(here->func))(here->arg);
        !          1419:                        here = here->next_actv;
        !          1420:                }
        !          1421:        } while(doit);
        !          1422: }
        !          1423: 
        !          1424: /*
        !          1425:  * upd_irq1()
        !          1426:  *
        !          1427:  * Given an irq number, rebuild the links for active devices.
        !          1428:  */
        !          1429: static void
        !          1430: upd_irq1(irq)
        !          1431: int irq;
        !          1432: {
        !          1433:        struct irqnode  *np;
        !          1434:        asy1_t  *a1;
        !          1435:        int     chan;
        !          1436:        int     s;
        !          1437: 
        !          1438:        /*
        !          1439:         * Sanity check.
        !          1440:         */
        !          1441:        if (irq <=0 || irq >= NUM_IRQ || itbl[irq] == 0)
        !          1442:                return;
        !          1443: 
        !          1444:        /*
        !          1445:         * For each node in the irq0 list
        !          1446:         *      if node is for irq status port
        !          1447:         *              for each channel using the status port
        !          1448:         *                      if channel in use, in irq mode
        !          1449:         *                              add node to irq1 list
        !          1450:         *                              skip rest of channels for this node
        !          1451:         *      else - node is for simple UART
        !          1452:         *              if channel in use, in irq mode
        !          1453:         *                      add node to irq1 list
        !          1454:         */
        !          1455:        s = sphi();
        !          1456:        np = irq0[irq];
        !          1457:        irq1[irq] = 0;
        !          1458:        while (np) {
        !          1459:                if (np->func != asyintr) {
        !          1460:                        char ix, loop = 1;
        !          1461:                        asy_gp_t *gp = asy_gp + np->arg;
        !          1462: 
        !          1463:                        for (ix = 0; ix < MAX_SLOTS && loop; ix++) {
        !          1464:                                if ((chan = gp->chan_list[ix]) < MAX_ASY) {
        !          1465:                                        a1 = asy1 + chan;
        !          1466:                                        if (a1->a_in_use && a1->a_irq) {
        !          1467:                                                np->next_actv = irq1[irq];
        !          1468:                                                irq1[irq] = np;
        !          1469:                                                loop = 0;
        !          1470:                                        }
        !          1471:                                }
        !          1472:                        }
        !          1473:                } else {
        !          1474:                        a1 = asy1 + np->arg;
        !          1475:                        if (a1->a_in_use && a1->a_irq) {
        !          1476:                                np->next_actv = irq1[irq];
        !          1477:                                irq1[irq] = np;
        !          1478:                        }
        !          1479:                }
        !          1480:                np = np->next;
        !          1481:        }
        !          1482:        spl(s);
        !          1483: }
        !          1484: 
        !          1485: /*
        !          1486:  * asybreak()
        !          1487:  */
        !          1488: static void
        !          1489: asybreak(chan)
        !          1490: int chan;
        !          1491: {
        !          1492:        int     s;
        !          1493:        asy1_t  *a1 = asy1 + chan;
        !          1494:        silo_t  *out_silo = &a1->a_out;
        !          1495:        silo_t  *in_silo = &a1->a_in;
        !          1496:        TTY     *tp = &a1->a_tty;
        !          1497: 
        !          1498:        s = sphi();
        !          1499:        RAWIN_FLUSH(in_silo);
        !          1500:        RAWOUT_FLUSH(out_silo);
        !          1501:        spl(s);
        !          1502:        ttsignal(tp, SIGINT);
        !          1503: }
        !          1504: 
        !          1505: /*
        !          1506:  * asyintr()
        !          1507:  *
        !          1508:  * Handle interrupt for a single channel.
        !          1509:  */
        !          1510: static int
        !          1511: asyintr(chan)
        !          1512: int chan;
        !          1513: {
        !          1514:        asy0_t  *a0 = asy0 + chan;
        !          1515:        asy1_t  *a1 = asy1 + chan;
        !          1516:        TTY     *tp = &a1->a_tty;
        !          1517:        silo_t  *out_silo = &a1->a_out;
        !          1518:        silo_t  *in_silo = &a1->a_in;
        !          1519:        int     c, xmit_count;
        !          1520:        int     ret = 0;
        !          1521:        short   port = a0->a_port;
        !          1522:        unsigned char   msr, lsr;
        !          1523: 
        !          1524:        if (chan >= MAX_ASY) {
        !          1525:                printf("asy: irq on channel %d\n", chan);
        !          1526:                return 0;
        !          1527:        }
        !          1528: 
        !          1529: rescan:
        !          1530:        switch (inb(port+IIR) & 0x07) {
        !          1531: 
        !          1532:        case LS_INTR:
        !          1533:                ret = 1;
        !          1534:                lsr = inb(port + LSR);
        !          1535:                T_HAL(0x800, printf("[%d:L%x]", chan, lsr));
        !          1536:                if (lsr & LS_BREAK)
        !          1537:                        defer(asybreak, chan);
        !          1538:                goto rescan;
        !          1539: 
        !          1540:        case Rx_INTR:
        !          1541:                T_HAL(0x800, printf("[%d:R]", chan));
        !          1542:                ret = 1;
        !          1543:                c = inb(port+DREG);
        !          1544:                if (tp->t_open == 0)
        !          1545:                        goto rescan;
        !          1546:                /*
        !          1547:                 * Must recognize XOFF quickly to avoid transmit overrun.
        !          1548:                 * Recognize XON here as well to avoid race conditions.
        !          1549:                 */
        !          1550:                if (!ISRIN) {
        !          1551:                        /*
        !          1552:                         * XOFF.
        !          1553:                         */
        !          1554:                        if (ISSTOP) {
        !          1555:                                tp->t_flags |= T_STOP;
        !          1556:                                goto rescan;
        !          1557:                        }
        !          1558: 
        !          1559:                        /*
        !          1560:                         * XON.
        !          1561:                         */
        !          1562:                        if (ISSTART) {
        !          1563:                                tp->t_flags &= ~T_STOP;
        !          1564:                                goto rescan;
        !          1565:                        }
        !          1566:                }
        !          1567: 
        !          1568:                /*
        !          1569:                 * Save char in raw input buffer.
        !          1570:                 */
        !          1571:                if (in_silo->SILO_CHAR_COUNT < MAX_SILO_CHARS) {
        !          1572:                        in_silo->si_buf[in_silo->si_ix] = c;
        !          1573:                        if (in_silo->si_ix < MAX_SILO_INDEX)
        !          1574:                                in_silo->si_ix++;
        !          1575:                        else
        !          1576:                                in_silo->si_ix = 0;
        !          1577:                        in_silo->SILO_CHAR_COUNT++;
        !          1578:                }
        !          1579: 
        !          1580:                /*
        !          1581:                 * If using hardware flow control, see if we need to drop RTS.
        !          1582:                 */
        !          1583:                if ((tp->t_flags & T_CFLOW)
        !          1584:                && (in_silo->SILO_CHAR_COUNT > SILO_HIGH_MARK)) {
        !          1585:                        unsigned char mcr = inb(port+MCR);
        !          1586:                        if (mcr & MC_RTS) {
        !          1587:                                outb(port+MCR, mcr & ~MC_RTS);
        !          1588:                        }
        !          1589:                }
        !          1590: 
        !          1591:                goto rescan;
        !          1592: 
        !          1593:        case Tx_INTR:
        !          1594:                T_HAL(0x800, printf("[%d:T]", chan));
        !          1595:                ret = 1;
        !          1596:                /*
        !          1597:                 * Do nothing if output is stopped.
        !          1598:                 */
        !          1599:                if (tp->t_flags & T_STOP) {
        !          1600:                        goto rescan;
        !          1601:                }
        !          1602:                if (a1->a_ohlt)
        !          1603:                        goto rescan;
        !          1604: 
        !          1605:                /*
        !          1606:                 * Transmit next char in raw output buffer.
        !          1607:                 */
        !          1608:                xmit_count = (a1->a_ut == US_16550A)?16:1;
        !          1609:                asy_send(out_silo, port+DREG, xmit_count);
        !          1610:                goto rescan;
        !          1611: 
        !          1612:        case MS_INTR:
        !          1613:                ret = 1;
        !          1614:                /*
        !          1615:                 * Get status (and clear interrupt).
        !          1616:                 */
        !          1617:                msr = inb(port+MSR);
        !          1618:                T_HAL(0x800, printf("[%d:M%x]", chan, msr));
        !          1619: 
        !          1620:                /*
        !          1621:                 * Hardware flow control.
        !          1622:                 *      Check CTS to see if we need to halt output.
        !          1623:                 */
        !          1624:                if (tp->t_flags & T_CFLOW) {
        !          1625:                        if (msr & MS_CTS)
        !          1626:                                a1->a_ohlt = 0;
        !          1627:                        else
        !          1628:                                a1->a_ohlt = 1;
        !          1629:                }
        !          1630: 
        !          1631:                goto rescan;
        !          1632:        default:
        !          1633:                return ret;
        !          1634:        } /* endswitch */
        !          1635: }
        !          1636: 
        !          1637: /*
        !          1638:  * asyclk()
        !          1639:  *
        !          1640:  * Called every time T0 interrupts.- if it returns 0,
        !          1641:  * the usual system timer interrupt stuff is done.
        !          1642:  * Poll all pollable ports.
        !          1643:  */
        !          1644: static int
        !          1645: asyclk()
        !          1646: {
        !          1647:        static  int count;
        !          1648:        int     ix;
        !          1649: 
        !          1650:        for (ix = 0; ix < ppnum; ix++)
        !          1651:                asysph(pptbl[ix]);
        !          1652: 
        !          1653:        count++;
        !          1654:        if (count >= poll_divisor)
        !          1655:                count = 0;
        !          1656:        return count;
        !          1657: }
        !          1658: 
        !          1659: /*
        !          1660:  * asyspr()
        !          1661:  *
        !          1662:  * asyspr is called when a port is opened or closed or changes speed
        !          1663:  * It sets the polling rate only as fast as needed, and shuts off polling
        !          1664:  * whenever possible.
        !          1665:  * It updates the links in irq1[0], which lists polled-mode ports.
        !          1666:  */
        !          1667: static void
        !          1668: asyspr()
        !          1669: {
        !          1670:        asy0_t  *a0;
        !          1671:        asy1_t  *a1;
        !          1672:        int     chan;
        !          1673:        int     s;
        !          1674:        int     ix, max_rate, port_rate;
        !          1675: 
        !          1676:        /*
        !          1677:         * Rebuild table of pollable ports.
        !          1678:         */
        !          1679:        s = sphi();
        !          1680:        ppnum = 0;
        !          1681:        for (chan = 0; chan < ASY_NUM; chan++) {
        !          1682:                a1 = asy1 + chan;
        !          1683:                if (a1->a_poll)
        !          1684:                        pptbl[ppnum++] = chan;
        !          1685:        }
        !          1686:        spl(s);
        !          1687: 
        !          1688:        /*
        !          1689:         * If another driver has the polling clock, do nothing.
        !          1690:         */
        !          1691:        if (poll_owner & ~ POLL_ASY)
        !          1692:                return;
        !          1693: 
        !          1694:        /*
        !          1695:         * Find highest valid polling rate in units of HZ/10.
        !          1696:         * If using FIFO chip, can poll at 1/16 the usual rate.
        !          1697:         */
        !          1698:        max_rate = 0;
        !          1699:        for (ix = 0; ix < ppnum; ix++) {
        !          1700:                chan = pptbl[ix];
        !          1701:                a0 = asy0 + chan;
        !          1702:                a1 = asy1 + chan;
        !          1703:                port_rate = alp_rate[a0->a_speed];
        !          1704:                if (a1->a_ut == US_16550A) {
        !          1705:                        port_rate /= 16;
        !          1706:                        if (port_rate % HZ)
        !          1707:                                port_rate += HZ - (port_rate % HZ);
        !          1708:                }
        !          1709:                if (max_rate < port_rate)
        !          1710:                        max_rate = port_rate;
        !          1711:        }
        !          1712: 
        !          1713:        /*
        !          1714:         * if max_rate is not current rate, adjust the system clock
        !          1715:         */
        !          1716:        if (max_rate != poll_rate) {
        !          1717:                poll_rate = max_rate;
        !          1718:                poll_divisor = poll_rate/HZ;  /* used in asyclk() */
        !          1719:                altclk_out();           /* stop previous polling */
        !          1720:                poll_owner &= ~ POLL_ASY;
        !          1721:                if (poll_rate) {  /* resume polling at new rate if needed */
        !          1722:                        poll_owner |= POLL_ASY;
        !          1723:                        altclk_in(poll_rate, asyclk);
        !          1724:                }
        !          1725:        }
        !          1726: }
        !          1727: 
        !          1728: /*
        !          1729:  * asysph()
        !          1730:  *
        !          1731:  * Serial polling handler.
        !          1732:  */
        !          1733: static void
        !          1734: asysph(chan)
        !          1735: int chan;
        !          1736: {
        !          1737:        asy0_t  *a0 = asy0 + chan;
        !          1738:        asy1_t  *a1 = asy1 + chan;
        !          1739:        TTY     *tp = &a1->a_tty;
        !          1740:        silo_t  *out_silo = &a1->a_out;
        !          1741:        silo_t  *in_silo = &a1->a_in;
        !          1742:        int     c, xmit_count;
        !          1743:        short   port = a0->a_port;
        !          1744:        char    lsr;
        !          1745: 
        !          1746:        /*
        !          1747:         * Check for received break first.
        !          1748:         * This status is wiped out on reading the LSR.
        !          1749:         */
        !          1750:        lsr = inb(port + LSR);
        !          1751:        if (lsr & LS_BREAK)
        !          1752:                defer(asybreak, chan);
        !          1753: 
        !          1754:        /*
        !          1755:         * Handle all incoming characters.
        !          1756:         */
        !          1757:        for (;;) {
        !          1758:                lsr = inb(port + LSR);
        !          1759:                if ((lsr & LS_RxRDY) == 0)
        !          1760:                        break;
        !          1761:                c = inb(port+DREG);
        !          1762:                if (tp->t_open == 0)
        !          1763:                        continue;
        !          1764:                /*
        !          1765:                 * Must recognize XOFF quickly to avoid transmit overrun.
        !          1766:                 * Recognize XON here as well to avoid race conditions.
        !          1767:                 */
        !          1768:                if (!ISRIN) {
        !          1769:                        /*
        !          1770:                         * XOFF.
        !          1771:                         */
        !          1772:                        if (ISSTOP) {
        !          1773:                                tp->t_flags |= T_STOP;
        !          1774:                                continue;
        !          1775:                        }
        !          1776: 
        !          1777:                        /*
        !          1778:                         * XON.
        !          1779:                         */
        !          1780:                        if (ISSTART) {
        !          1781:                                tp->t_flags &= ~T_STOP;
        !          1782:                                continue;
        !          1783:                        }
        !          1784:                }
        !          1785: 
        !          1786:                /*
        !          1787:                 * Save char in raw input buffer.
        !          1788:                 */
        !          1789:                if (in_silo->SILO_CHAR_COUNT < MAX_SILO_CHARS) {
        !          1790:                        in_silo->si_buf[in_silo->si_ix] = c;
        !          1791:                        if (in_silo->si_ix < MAX_SILO_INDEX)
        !          1792:                                in_silo->si_ix++;
        !          1793:                        else
        !          1794:                                in_silo->si_ix = 0;
        !          1795:                        in_silo->SILO_CHAR_COUNT++;
        !          1796:                }
        !          1797: 
        !          1798:                /*
        !          1799:                 * If using hardware flow control, see if we need to drop RTS.
        !          1800:                 */
        !          1801:                if ((tp->t_flags & T_CFLOW)
        !          1802:                  && (in_silo->SILO_CHAR_COUNT > SILO_HIGH_MARK)) {
        !          1803:                        unsigned char mcr = inb(port+MCR);
        !          1804:                        if (mcr & MC_RTS) {
        !          1805:                                outb(port+MCR, mcr & ~MC_RTS);
        !          1806:                        }
        !          1807:                }
        !          1808:        }
        !          1809: 
        !          1810:        /*
        !          1811:         * Handle outgoing characters.
        !          1812:         * Do nothing if output is stopped.
        !          1813:         */
        !          1814:        lsr = inb(port + LSR);
        !          1815:        if ((lsr & LS_TxRDY)
        !          1816:          && !(tp->t_flags & T_STOP)
        !          1817:          && !(a1->a_ohlt)) {
        !          1818:                /*
        !          1819:                 * Transmit next char in raw output buffer.
        !          1820:                 */
        !          1821:                xmit_count = (a1->a_ut == US_16550A)?16:1;
        !          1822:                asy_send(out_silo, port+DREG, xmit_count);
        !          1823:        }
        !          1824: 
        !          1825:        /*
        !          1826:         * Hardware flow control.
        !          1827:         *      Check CTS to see if we need to halt output.
        !          1828:         */
        !          1829:        if (tp->t_flags & T_CFLOW) {
        !          1830:                if (inb(port+MSR) & MS_CTS)
        !          1831:                        a1->a_ohlt = 0;
        !          1832:                else
        !          1833:                        a1->a_ohlt = 1;
        !          1834:        }
        !          1835: }
        !          1836: 
        !          1837: /*
        !          1838:  * asy_send()
        !          1839:  *
        !          1840:  * Write to xmit data register of the UART.
        !          1841:  * Assume all checking about whether it's time to send has been done already.
        !          1842:  * Called by time-critical IRQ and polling routines!
        !          1843:  *
        !          1844:  * "rawout" is the output silo for the TTY struct supplying data to the port.
        !          1845:  * "dreg" is the i/o address of the UART xmit data register.
        !          1846:  * "xmit_count" is the max number of chars we can write (16 for FIFO parts).
        !          1847:  */
        !          1848: static int
        !          1849: asy_send(rawout, dreg, xmit_count)
        !          1850: register silo_t * rawout;
        !          1851: int dreg, xmit_count;
        !          1852: {
        !          1853:        /*
        !          1854:         * Transmit next chars in raw output buffer.
        !          1855:         */
        !          1856:        for (;(rawout->si_ix != rawout->si_ox) && xmit_count; xmit_count--) {
        !          1857:                outb(dreg, rawout->si_buf[rawout->si_ox]);
        !          1858:                /*
        !          1859:                 * Adjust raw output buffer output index.
        !          1860:                 */
        !          1861:                if (++rawout->si_ox >= sizeof(rawout->si_buf))
        !          1862:                        rawout->si_ox = 0;
        !          1863:        }
        !          1864:        return xmit_count;
        !          1865: }
        !          1866: 
        !          1867: /*
        !          1868:  * p1()
        !          1869:  *
        !          1870:  * Interrupt handler for Comtrol-type port groups.
        !          1871:  * Status register has 1 in bit positions for interrupting ports.
        !          1872:  */
        !          1873: static int
        !          1874: p1(g)
        !          1875: int g;
        !          1876: {
        !          1877:        asy_gp_t        *gp = asy_gp + g;
        !          1878:        short           port = gp->stat_port;
        !          1879:        unsigned char   status, index, chan;
        !          1880:        int             safety = LOOP_LIMIT;
        !          1881:        int             ret = 0;
        !          1882: 
        !          1883: #if 0 /* DEBUG */
        !          1884: static int pxstat[2][8];
        !          1885:        int ci;
        !          1886:        int change_found=0;
        !          1887: 
        !          1888:        for (ci=0; ci<1; ci++) {
        !          1889:                index = inb(port+ci);
        !          1890:                outb(port+ci, 0);
        !          1891:                if (index != pxstat[g][ci]) {
        !          1892:                        if (!change_found) {
        !          1893:                                change_found = 1;
        !          1894:                                printf("<%d:", g);
        !          1895:                        } else
        !          1896:                                putchar(' ');
        !          1897:                        printf("%x:%x", port+ci, index);
        !          1898:                        pxstat[g][ci] = index;
        !          1899:                }
        !          1900:        }
        !          1901:        if (change_found)
        !          1902:                putchar('>');
        !          1903:        for (ci=0; ci<8; ci++)
        !          1904:                asyintr(4+ci);
        !          1905:        putchar('.');
        !          1906:        return 0;
        !          1907: #endif /* DEBUG */
        !          1908: 
        !          1909:        /*
        !          1910:         * while any port is active
        !          1911:         *      call simple interrupt handler for active channel
        !          1912:         */
        !          1913:        while (status = inb(port)) {
        !          1914:                ret = 1;
        !          1915:                index = 0;
        !          1916:                if (status & 0xf0) {
        !          1917:                        status &= 0xf0;
        !          1918:                        index +=4;
        !          1919:                } else
        !          1920:                        status &= 0x0f;
        !          1921:                if (status & 0xcc) {
        !          1922:                        status &= 0xcc;
        !          1923:                        index +=2;
        !          1924:                } else
        !          1925:                        status &= 0x33;
        !          1926:                if (status & 0xaa)
        !          1927:                        index++;
        !          1928:                chan = gp->chan_list[index];
        !          1929:                asyintr(chan);
        !          1930:                if (safety-- == 0) {
        !          1931:                        printf("asy: p1 runaway - status %x\n", status);
        !          1932:                        break;
        !          1933:                }
        !          1934:        }
        !          1935: 
        !          1936:        return ret;
        !          1937: }
        !          1938: 
        !          1939: /*
        !          1940:  * p2()
        !          1941:  *
        !          1942:  * Interrupt handler for Arnet-type port groups.
        !          1943:  * Status register has 0 in bit positions for interrupting ports.
        !          1944:  */
        !          1945: static int
        !          1946: p2(g)
        !          1947: int g;
        !          1948: {
        !          1949:        asy_gp_t        *gp = asy_gp + g;
        !          1950:        short           port = gp->stat_port;
        !          1951:        unsigned char   status, index, chan;
        !          1952:        int             safety = LOOP_LIMIT;
        !          1953:        int             ret = 0;
        !          1954: 
        !          1955:        /*
        !          1956:         * while any port is active
        !          1957:         *      call simple interrupt handler for active channel
        !          1958:         */
        !          1959:        while (status = ~inb(port)) {
        !          1960:                ret = 1;
        !          1961:                index = 0;
        !          1962:                if (status & 0xf0) {
        !          1963:                        status &= 0xf0;
        !          1964:                        index +=4;
        !          1965:                } else
        !          1966:                        status &= 0x0f;
        !          1967:                if (status & 0xcc) {
        !          1968:                        status &= 0xcc;
        !          1969:                        index +=2;
        !          1970:                } else
        !          1971:                        status &= 0x33;
        !          1972:                if (status & 0xaa)
        !          1973:                        index++;
        !          1974:                chan = gp->chan_list[index];
        !          1975:                asyintr(chan);
        !          1976:                if (safety-- == 0) {
        !          1977:                        printf("asy: p2 runaway - status %x\n", status);
        !          1978:                        break;
        !          1979:                }
        !          1980:        }
        !          1981:        return ret;
        !          1982: }
        !          1983: 
        !          1984: /*
        !          1985:  * p3()
        !          1986:  *
        !          1987:  * Interrupt handler for GTEK-type port groups.
        !          1988:  */
        !          1989: static int
        !          1990: p3(g)
        !          1991: int g;
        !          1992: {
        !          1993:        asy_gp_t        *gp = asy_gp + g;
        !          1994:        short           port = gp->stat_port;
        !          1995:        unsigned char   index, chan;
        !          1996: 
        !          1997:        /*
        !          1998:         * Call simple interrupt handler for active channel.
        !          1999:         */
        !          2000:        index = inb(port) & 7;
        !          2001:        chan = gp->chan_list[index];
        !          2002:        return asyintr(chan);
        !          2003: }
        !          2004: 
        !          2005: /*
        !          2006:  * p4()
        !          2007:  *
        !          2008:  * Interrupt handler for DigiBoard-type port groups.
        !          2009:  */
        !          2010: static int
        !          2011: p4(g)
        !          2012: int g;
        !          2013: {
        !          2014:        asy_gp_t        *gp = asy_gp + g;
        !          2015:        short           port = gp->stat_port;
        !          2016:        unsigned char   index, chan;
        !          2017:        int             ret = 0;
        !          2018:        int             safety = LOOP_LIMIT;
        !          2019: 
        !          2020:        /*
        !          2021:         * Status register has slot number for active port,
        !          2022:         * or 0xFF if no port is active.
        !          2023:         */
        !          2024: 
        !          2025:        for (;;) {
        !          2026:                index = inb(port);
        !          2027:                if (index == 0xFF)
        !          2028:                        break;
        !          2029:                if (safety-- == 0) {
        !          2030:                        printf("asy: p4 runaway - status %x\n", index);
        !          2031:                        break;
        !          2032:                }
        !          2033:                ret = 1;
        !          2034:                chan = gp->chan_list[index&0xF];
        !          2035:                asyintr(chan);
        !          2036:        }
        !          2037:        return ret;
        !          2038: }
        !          2039: 
        !          2040: #ifdef TRACER
        !          2041: void
        !          2042: asydump(chan, tag)
        !          2043: int chan;
        !          2044: char *tag;
        !          2045: {
        !          2046:        asy0_t  *a0 = asy0 + chan;
        !          2047:        asy1_t  *a1 = asy1 + chan;
        !          2048:        TTY     *tp = &a1->a_tty;
        !          2049: 
        !          2050:        printf("ch=%d  %s\n", chan, tag);
        !          2051:        printf("port=%x  irqno=%x  speed=%d  ",
        !          2052:          a0->a_port, a0->a_irqno, a0->a_speed);
        !          2053:        printf("outs=%x  gp=%d  xcl=%d\n",
        !          2054:          a0->a_outs, a0->a_asy_gp, a0->a_ixc);
        !          2055:        printf("in_use=%d  lcr=%x  irq=%d  has_irq=%d  ",
        !          2056:          a1->a_in_use, a1->a_lcr, a1->a_irq, a1->a_has_irq);
        !          2057:        printf("hop=%d  hcl=%d  ", a1->a_hopn, a1->a_hcls);
        !          2058:        printf("opn=%d  ier=%x\n", tp->t_open, inb(a0->a_port+IER));
        !          2059: }
        !          2060: #endif

unix.superglobalmegacorp.com

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