Annotation of coherent/d/286_KERNEL/USRSRC/io/al.c, revision 1.1

1.1     ! root        1: /* (-lgl
        !             2:  *     COHERENT Device Driver Kit version 1.2.0
        !             3:  *     Copyright (c) 1982, 1991 by Mark Williams Company.
        !             4:  *     All rights reserved. May not be copied without permission.
        !             5:  *
        !             6:  * $Log:       al.c,v $
        !             7:  * Revision 1.12  92/02/14  10:12:27  bin
        !             8:  * update by hal (post 321)
        !             9:  * 
        !            10:  * Revision 1.11  92/01/13  08:37:52  hal
        !            11:  * alclose() - decrement open count in alx.c
        !            12:  * 
        !            13:  * Revision 1.10  91/12/20  14:09:50  hal
        !            14:  * Don't use loopback during chip sense.
        !            15:  * 
        !            16:  * Revision 1.9  91/12/10  08:01:11  hal
        !            17:  * Set ALCNT automatically.
        !            18:  * Set interrupt vector before calling uart_sense().
        !            19:  * 
        !            20:  * Revision 1.8  91/12/05  09:35:25  hal
        !            21:  * Working 16550A code.  Nfg on GeeSee.
        !            22:  * 
        !            23:  * Revision 1.7  91/12/02  19:22:00  hal
        !            24:  * Last version before FIFO testing.
        !            25:  * 
        !            26:  -lgl) */
        !            27: /*
        !            28:  * Driver for an IBM PC asyncronous
        !            29:  * line, using interrupts. The interface
        !            30:  * uses a Natty/WD 8250 chip.
        !            31:  */
        !            32: 
        !            33: #include <sys/coherent.h>
        !            34: #ifndef _I386
        !            35: #include <sys/i8086.h>
        !            36: #endif
        !            37: #include <sys/con.h>
        !            38: #include <errno.h>
        !            39: #include <sys/stat.h>
        !            40: #include <sys/tty.h>
        !            41: #include <sys/uproc.h>
        !            42: #include <sys/clist.h>
        !            43: #include <sys/ins8250.h>
        !            44: #include <sys/sched.h>
        !            45: #include <sys/al.h>
        !            46: #include <sys/devices.h>
        !            47: 
        !            48: #define        minor_st(dev)   (dev & 0x0f)    /* up to 16 ports per driver */
        !            49: #define        DEV_TTY         (alttab[minor_st(dev)])
        !            50: #define ALPORT         (((COM_DDP *)(DEV_TTY.t_ddp))->port)
        !            51: #define TESTBAUD       0x03A5
        !            52: 
        !            53: /*
        !            54:  * This driver can be compiled to drive any possible
        !            55:  * async port by appropriate definitions of:
        !            56:  *     ALPORT[ab]      the io port address(es)
        !            57:  *     ALNUM[ab]       com index number (0..3 for com[1..4])
        !            58:  *     ALINT   the interrupt level
        !            59:  *     ALNAME  the xxcon name
        !            60:  *     ALMAJ   the major device number
        !            61:  *      ALCNT  number of ports sharing the interrupt
        !            62:  *
        !            63:  *     NOTE:   if ALCNT is changed, alttab and alintr will need hacking
        !            64:  * Common code for the different ports is handled by alx.c
        !            65:  */
        !            66: 
        !            67: #ifdef ALCOM1                  /* COM1_3 definitions */
        !            68: #define ALPORTa        0x3F8           /* Base of com1 port */
        !            69: #define ALPORTb        0x3E8           /* Base of com3 port */
        !            70: #define ALNUMa 0               /* com1 has com number of 0 */
        !            71: #define ALNUMb 2               /* com3 has com number of 2 */
        !            72: #define ALINT  4               /* Interrupt level of com1_3 ports */
        !            73: #define        ALNAME  a0con           /* CON name of com1_3 ports */
        !            74: #define ALMAJ  AL0_MAJOR       /* Major number of com1_3 port */
        !            75: #define ALCNT  A0CNT           /* Number of ports for this IRQ */
        !            76: #define ALSPEEDa C1BAUD                /* Name of patchable variable for com1 speed */
        !            77: #define ALSPEEDb C3BAUD                /* Name of patchable variable for com3 speed */
        !            78: #endif
        !            79: 
        !            80: #ifdef ALCOM2                  /* COM2_4 definitions */
        !            81: #define ALPORTa        0x2F8           /* Base of com2 port */
        !            82: #define ALPORTb        0x2E8           /* Base of com4 port */
        !            83: #define ALNUMa 1               /* com2 has com number of 1 */
        !            84: #define ALNUMb 3               /* com4 has com number of 3 */
        !            85: #define ALINT  3               /* Interrupt level of com2_4 ports */
        !            86: #define ALNAME a1con           /* CON name of com2_4 ports */
        !            87: #define ALMAJ  AL1_MAJOR       /* Major number of com2_4 ports */
        !            88: #define ALCNT  A1CNT           /* Number of ports for this IRQ */
        !            89: #define ALSPEEDa C2BAUD                /* Name of patchable variable for com2 speed */
        !            90: #define ALSPEEDb C4BAUD                /* Name of patchable variable for com4 speed */
        !            91: #endif
        !            92: 
        !            93: /*
        !            94:  * Functions.
        !            95:  */
        !            96: int    alxopen();
        !            97: int    alxclose();
        !            98: int    alxioctl();
        !            99: int    alxtimer();
        !           100: int    alxparam();
        !           101: int    alxcycle();
        !           102: int    alxstart();
        !           103: int    alxbreak();
        !           104: 
        !           105: int    alintr();
        !           106: int    alopen();
        !           107: int    alclose();
        !           108: int    alread();
        !           109: int    alwrite();
        !           110: int    alioctl();
        !           111: int    alload();
        !           112: int    alunload();
        !           113: int    alpoll();
        !           114: int    nulldev();
        !           115: int    nonedev();
        !           116: static int uart_sense();
        !           117: 
        !           118: /*
        !           119:  * Configuration table.
        !           120:  */
        !           121: CON ALNAME ={
        !           122:        DFCHR|DFPOL,                    /* Flags */
        !           123:        ALMAJ,                          /* Major index */
        !           124:        alopen,                         /* Open */
        !           125:        alclose,                        /* Close */
        !           126:        nulldev,                        /* Block */
        !           127:        alread,                         /* Read */
        !           128:        alwrite,                        /* Write */
        !           129:        alioctl,                        /* Ioctl */
        !           130:        nulldev,                        /* Powerfail */
        !           131:        alxtimer,                       /* Timeout */
        !           132:        alload,                         /* Load */
        !           133:        alunload,                       /* Unload */
        !           134:        alpoll                          /* Poll */
        !           135: };
        !           136: 
        !           137: /*
        !           138:  * Terminal structures.
        !           139:  */
        !           140: static COM_DDP * ddp;
        !           141: static TTY     * alttab;
        !           142: static TTY     * irqtty;  /* point to alttab entry which is IRQ-enabled */
        !           143: 
        !           144: /*
        !           145:  * to change default speeds - patch kernel variables C1BAUD..C4BAUD
        !           146:  *   new value should be one of B0..B9600 in /usr/include/sgtty.h
        !           147:  */
        !           148: int ALSPEEDa = B9600;
        !           149: int ALSPEEDb = B9600;
        !           150: 
        !           151: /*
        !           152:  * to enable com[34], patch here
        !           153:  *     A0CNT should be 2 if you want com3, 1 otherwise
        !           154:  *     A1CNT should be 2 if you want com4, 1 otherwise
        !           155:  */
        !           156: int ALCNT = 2;
        !           157: 
        !           158: static
        !           159: alload()
        !           160: {
        !           161:        register int s;
        !           162:        static int init;
        !           163:        extern int albaud[];
        !           164:        int port, i;
        !           165:        int usa, usb;
        !           166: 
        !           167:        /*
        !           168:         * Set interrupt vector early in case uart_sense() causes bogus irpts.
        !           169:         */
        !           170:        setivec(ALINT, alintr);     /* set interrupt vector */
        !           171:        usa = uart_sense(ALPORTa);
        !           172:        usb = uart_sense(ALPORTb);
        !           173:        if (usa == US_NONE && usb == US_NONE) {
        !           174:                ALCNT = 0;
        !           175:        } else {
        !           176:                if (usb == US_NONE)
        !           177:                        ALCNT = 1;
        !           178:                else
        !           179:                        ALCNT = 2;
        !           180:        }
        !           181:        if (init == 0 && ALCNT
        !           182:          && (alttab = (TTY *)kalloc(ALCNT * sizeof(TTY)))
        !           183:          && (ddp = (COM_DDP *)kalloc(ALCNT * sizeof(COM_DDP)))) {
        !           184:                kclear(alttab, ALCNT*sizeof(TTY));
        !           185:                kclear(ddp, ALCNT*sizeof(COM_DDP));
        !           186:                ++init;
        !           187: 
        !           188:                s = sphi();
        !           189:                alttab[0].t_dispeed = alttab[0].t_dospeed = ALSPEEDa;
        !           190:                alttab[0].t_ddp = (char *)&ddp[0];
        !           191:                tp_table[ALNUMa] = alttab; /* set TTY pointers for polling */
        !           192:                ddp[0].port = ALPORTa;
        !           193:                ddp[0].com_num = ALNUMa;
        !           194:                com_usage[ALNUMa].uart_type = usa;
        !           195: 
        !           196:                if (ALCNT > 1) {
        !           197:                        alttab[1].t_dispeed = alttab[1].t_dospeed = ALSPEEDb;
        !           198:                        alttab[1].t_ddp = (char *)&ddp[1];
        !           199:                        tp_table[ALNUMb] = alttab+1;
        !           200:                        ddp[1].port = ALPORTb;
        !           201:                        ddp[1].com_num = ALNUMb;
        !           202:                        com_usage[ALNUMb].uart_type = usb;
        !           203:                }
        !           204: 
        !           205:                for (i = 0;  i < ALCNT; i++) {
        !           206:                        int speed = alttab[i].t_dospeed;
        !           207: 
        !           208:                        /* port = base I/O address */
        !           209:                        port = ((COM_DDP *)(alttab[i].t_ddp))->port;
        !           210:                        outb(port+IER, 0);      /* disable port interrupts */
        !           211:                        outb(port+MCR, 0);  /* hangup port */
        !           212:                        outb(port+LCR, LC_DLAB);
        !           213:                        outb(port+DLL, albaud[speed]);
        !           214:                        outb(port+DLH, albaud[speed] >> 8);
        !           215:                        outb(port+LCR, LC_CS8);
        !           216:                        alttab[i].t_start = alxstart;
        !           217:                        alttab[i].t_param = alxparam;
        !           218:                        alttab[i].t_cs_sel= cs_sel();
        !           219:                }
        !           220: 
        !           221:                spl(s);
        !           222:        } else {        /* Load failed - no ports or no RAM available! */
        !           223:                clrivec(ALINT);
        !           224:        }
        !           225:        return; 
        !           226: }
        !           227: 
        !           228: static
        !           229: alunload()
        !           230: {
        !           231:        int port, i;
        !           232: 
        !           233:        for (i = 0;  i < ALCNT; i++) {
        !           234:                port = ((COM_DDP *)(alttab[i].t_ddp))->port;
        !           235:                outb(port+IER, 0);      /* disable port interrupts */
        !           236:                outb(port+MCR, 0);      /* hangup port */
        !           237:                timeout(alttab[i].t_rawtim, 0, NULL, 0);/* cancel timer */
        !           238:        }
        !           239:        if (ALCNT) {
        !           240:                clrivec(ALINT);         /* release interrupt vector */
        !           241:                kfree(alttab);
        !           242:                kfree(ddp);
        !           243:        }
        !           244: }
        !           245: 
        !           246: static
        !           247: alopen(dev, mode)
        !           248: dev_t  dev;
        !           249: int    mode;
        !           250: {
        !           251:        if (minor_st(dev) < ALCNT) {
        !           252:                alxopen(dev, mode, &DEV_TTY, &irqtty);
        !           253:        } else
        !           254:                u.u_error = ENXIO;
        !           255: }
        !           256: 
        !           257: static
        !           258: alclose(dev, mode)
        !           259: dev_t  dev;
        !           260: int    mode;
        !           261: {
        !           262:        /*
        !           263:         * The real work is in alx.c.
        !           264:         */
        !           265:        alxclose(dev, mode, &DEV_TTY);
        !           266: }
        !           267: 
        !           268: static
        !           269: alread(dev, iop)
        !           270: dev_t  dev;
        !           271: IO     *iop;
        !           272: {
        !           273:        ttread(&DEV_TTY, iop, 0);
        !           274: }
        !           275: 
        !           276: static
        !           277: alwrite(dev, iop)
        !           278: dev_t  dev;
        !           279: register IO    *iop;
        !           280: {
        !           281:        register int c;
        !           282: 
        !           283:        /*
        !           284:         * Treat user writes through tty driver.
        !           285:         */
        !           286:        if (iop->io_seg != IOSYS) {
        !           287:                ttwrite(&DEV_TTY, iop, 0);
        !           288:                return;
        !           289:        }
        !           290: 
        !           291:        /*
        !           292:         * Treat kernel writes by blocking on transmit buffer.
        !           293:         */
        !           294:        while ((c = iogetc(iop)) >= 0) {
        !           295:                /*
        !           296:                 * Wait until transmit buffer is empty.
        !           297:                 * Check twice to prevent critical race with interrupt handler.
        !           298:                 */
        !           299:                for (;;) {
        !           300:                        if (inb(ALPORT+LSR) & LS_TxRDY)
        !           301:                                if (inb(ALPORT+LSR) & LS_TxRDY)
        !           302:                                        break;
        !           303:                }
        !           304: 
        !           305:                /*
        !           306:                 * Output the next character.
        !           307:                 */
        !           308:                outb(ALPORT+DREG, c);
        !           309:        }
        !           310: }
        !           311: 
        !           312: static
        !           313: alioctl(dev, com, vec)
        !           314: dev_t  dev;
        !           315: struct sgttyb *vec;
        !           316: {
        !           317:        alxioctl(dev, com, vec, &DEV_TTY);
        !           318: }
        !           319: 
        !           320: static
        !           321: alpoll(dev, ev, msec)
        !           322: dev_t dev;
        !           323: int ev;
        !           324: int msec;
        !           325: {
        !           326:        return ttpoll(&DEV_TTY, ev, msec);
        !           327: }
        !           328: 
        !           329: static
        !           330: alintr()
        !           331: {
        !           332:        alxintr(irqtty);
        !           333: }
        !           334: 
        !           335: /*
        !           336:  * uart_sense()
        !           337:  *
        !           338:  * Given port address, return what type of 8250-family chip is found there.
        !           339:  *
        !           340:  * 0 - no chip
        !           341:  * 1 - 8250 or 8250B
        !           342:  * 2 - 8250A or 16450
        !           343:  * 3 - 16550
        !           344:  * 4 - 16550A
        !           345:  *
        !           346:  * Only the last of these has usable on-chip FIFO.
        !           347:  */
        !           348: static int uart_sense(port)
        !           349: int port;
        !           350: {
        !           351:        int ret;
        !           352:        unsigned ch;
        !           353:        short testbaud;
        !           354: 
        !           355:        /*
        !           356:         * See if UART is detected at port address.
        !           357:         * UART should have IER = 0000 xxxx
        !           358:         *                  MCR = 000x xxxx
        !           359:         *                  IIR = xx00 xxxx
        !           360:         * and should be write and read back the baud rate regs.
        !           361:         */
        !           362:        if (inb(port+IER) & 0xF0
        !           363:          || inb(port+MCR) & 0xE0
        !           364:          || inb(port+IIR) & 0x30) {
        !           365:                ret = US_NONE;
        !           366:                goto done;
        !           367:        }
        !           368:        outb(port+LCR, LC_DLAB);
        !           369:        outb(port+DLL, TESTBAUD & 0xFF);
        !           370:        outb(port+DLH, TESTBAUD >> 8);
        !           371:        testbaud = inb(port+DLL) | inb(port+DLH) << 8;
        !           372:        outb(port+LCR, LC_CS8);
        !           373:        if (testbaud != TESTBAUD){
        !           374:                ret = US_NONE;
        !           375:                goto done;
        !           376:        }
        !           377: 
        !           378:        /*
        !           379:         * Scratch register NOT found on 8250/8250B.
        !           380:         */
        !           381:        outb(port+SCR, 0x55);
        !           382:        ch = inb(port+SCR);
        !           383:        if (ch != 0x55) {
        !           384:                ret = US_8250;
        !           385:                goto done;
        !           386:        }
        !           387: 
        !           388:        /*
        !           389:         * After trying to turn on FIFO mode,
        !           390:         * If IIR is 00xx xxxx, it's 8250A/16450 (no FIFO).
        !           391:         * If IIR is 10xx xxxx, it's 16550 (broken FIFO).
        !           392:         * If IIR is 11xx xxxx, it's 16550A (usable FIFO).
        !           393:         */
        !           394:        outb(port+FCR, 0x01);
        !           395:        ch = inb(port+FCR);
        !           396:        switch (ch & 0xC0) {
        !           397:        case 0x00:
        !           398:                ret = US_16450;
        !           399:                break;
        !           400:        case 0x80:
        !           401:                ret = US_16550;
        !           402:                break;
        !           403:        case 0xC0:
        !           404:                ret = US_16550A;
        !           405:                break;
        !           406:        }
        !           407:        outb(port+FCR, 0x00);
        !           408: done:
        !           409: if (ret == US_NONE)
        !           410:        goto really_done;       
        !           411: switch(port){
        !           412: case 0x3F8:
        !           413:        printf("com1 ");
        !           414:        break;
        !           415: case 0x2F8:
        !           416:        printf("com2 ");
        !           417:        break;
        !           418: case 0x3E8:
        !           419:        printf("com3 ");
        !           420:        break;
        !           421: case 0x2E8:
        !           422:        printf("com4 ");
        !           423:        break;
        !           424: }
        !           425: printf("port %x: ", port);     
        !           426: switch (ret) {
        !           427: case US_NONE:
        !           428:        printf("no UART\n");
        !           429:        break;
        !           430: case US_8250:
        !           431:        printf("8250/8250B\n");
        !           432:        break;
        !           433: case US_16450:
        !           434:        printf("8250A/16450\n");
        !           435:        break;
        !           436: case US_16550:
        !           437:        printf("16550 - no FIFO\n");
        !           438:        break;
        !           439: case US_16550A:
        !           440:        printf("16550A - FIFO\n");
        !           441:        break;
        !           442: }
        !           443: really_done:
        !           444:        return ret;
        !           445: }

unix.superglobalmegacorp.com

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