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

unix.superglobalmegacorp.com

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