|  | /*	$Id: aurora.c,v 1.19 2002/01/08 16:00:16 davem Exp $ | 
|  | *	linux/drivers/sbus/char/aurora.c -- Aurora multiport driver | 
|  | * | 
|  | *	Copyright (c) 1999 by Oliver Aldulea (oli at bv dot ro) | 
|  | * | 
|  | *	This code is based on the RISCom/8 multiport serial driver written | 
|  | *	by Dmitry Gorodchanin (pgmdsg@ibi.com), based on the Linux serial | 
|  | *	driver, written by Linus Torvalds, Theodore T'so and others. | 
|  | *	The Aurora multiport programming info was obtained mainly from the | 
|  | *	Cirrus Logic CD180 documentation (available on the web), and by | 
|  | *	doing heavy tests on the board. Many thanks to Eddie C. Dost for the | 
|  | *	help on the sbus interface. | 
|  | * | 
|  | *	This program is free software; you can redistribute it and/or modify | 
|  | *	it under the terms of the GNU General Public License as published by | 
|  | *	the Free Software Foundation; either version 2 of the License, or | 
|  | *	(at your option) any later version. | 
|  | * | 
|  | *	This program is distributed in the hope that it will be useful, | 
|  | *	but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | *	MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
|  | *	GNU General Public License for more details. | 
|  | * | 
|  | *	You should have received a copy of the GNU General Public License | 
|  | *	along with this program; if not, write to the Free Software | 
|  | *	Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. | 
|  | * | 
|  | *	Revision 1.0 | 
|  | * | 
|  | *	This is the first public release. | 
|  | * | 
|  | *	Most of the information you need is in the aurora.h file. Please | 
|  | *	read that file before reading this one. | 
|  | * | 
|  | *	Several parts of the code do not have comments yet. | 
|  | * | 
|  | * n.b.  The board can support 115.2 bit rates, but only on a few | 
|  | * ports. The total badwidth of one chip (ports 0-7 or 8-15) is equal | 
|  | * to OSC_FREQ div 16. In case of my board, each chip can take 6 | 
|  | * channels of 115.2 kbaud.  This information is not well-tested. | 
|  | * | 
|  | * Fixed to use tty_get_baud_rate(). | 
|  | *   Theodore Ts'o <tytso@mit.edu>, 2001-Oct-12 | 
|  | */ | 
|  |  | 
|  | #include <linux/module.h> | 
|  |  | 
|  | #include <linux/errno.h> | 
|  | #include <linux/sched.h> | 
|  | #ifdef AURORA_INT_DEBUG | 
|  | #include <linux/timer.h> | 
|  | #endif | 
|  | #include <linux/interrupt.h> | 
|  | #include <linux/tty.h> | 
|  | #include <linux/tty_flip.h> | 
|  | #include <linux/major.h> | 
|  | #include <linux/string.h> | 
|  | #include <linux/fcntl.h> | 
|  | #include <linux/mm.h> | 
|  | #include <linux/kernel.h> | 
|  | #include <linux/init.h> | 
|  | #include <linux/delay.h> | 
|  | #include <linux/bitops.h> | 
|  |  | 
|  | #include <asm/io.h> | 
|  | #include <asm/irq.h> | 
|  | #include <asm/oplib.h> | 
|  | #include <asm/system.h> | 
|  | #include <asm/kdebug.h> | 
|  | #include <asm/sbus.h> | 
|  | #include <asm/uaccess.h> | 
|  |  | 
|  | #include "aurora.h" | 
|  | #include "cd180.h" | 
|  |  | 
|  | unsigned char irqs[4] = { | 
|  | 0, 0, 0, 0 | 
|  | }; | 
|  |  | 
|  | #ifdef AURORA_INT_DEBUG | 
|  | int irqhit=0; | 
|  | #endif | 
|  |  | 
|  | static struct tty_driver *aurora_driver; | 
|  | static struct Aurora_board aurora_board[AURORA_NBOARD] = { | 
|  | {0,}, | 
|  | }; | 
|  |  | 
|  | static struct Aurora_port aurora_port[AURORA_TNPORTS] =  { | 
|  | { 0, }, | 
|  | }; | 
|  |  | 
|  | /* no longer used. static struct Aurora_board * IRQ_to_board[16] = { NULL, } ;*/ | 
|  | static unsigned char * tmp_buf = NULL; | 
|  | static DECLARE_MUTEX(tmp_buf_sem); | 
|  |  | 
|  | DECLARE_TASK_QUEUE(tq_aurora); | 
|  |  | 
|  | static inline int aurora_paranoia_check(struct Aurora_port const * port, | 
|  | char *name, const char *routine) | 
|  | { | 
|  | #ifdef AURORA_PARANOIA_CHECK | 
|  | static const char *badmagic = | 
|  | KERN_DEBUG "aurora: Warning: bad aurora port magic number for device %s in %s\n"; | 
|  | static const char *badinfo = | 
|  | KERN_DEBUG "aurora: Warning: null aurora port for device %s in %s\n"; | 
|  |  | 
|  | if (!port) { | 
|  | printk(badinfo, name, routine); | 
|  | return 1; | 
|  | } | 
|  | if (port->magic != AURORA_MAGIC) { | 
|  | printk(badmagic, name, routine); | 
|  | return 1; | 
|  | } | 
|  | #endif | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * | 
|  | *  Service functions for aurora driver. | 
|  | * | 
|  | */ | 
|  |  | 
|  | /* Get board number from pointer */ | 
|  | extern inline int board_No (struct Aurora_board const * bp) | 
|  | { | 
|  | return bp - aurora_board; | 
|  | } | 
|  |  | 
|  | /* Get port number from pointer */ | 
|  | extern inline int port_No (struct Aurora_port const * port) | 
|  | { | 
|  | return AURORA_PORT(port - aurora_port); | 
|  | } | 
|  |  | 
|  | /* Get pointer to board from pointer to port */ | 
|  | extern inline struct Aurora_board * port_Board(struct Aurora_port const * port) | 
|  | { | 
|  | return &aurora_board[AURORA_BOARD(port - aurora_port)]; | 
|  | } | 
|  |  | 
|  | /* Wait for Channel Command Register ready */ | 
|  | extern inline void aurora_wait_CCR(struct aurora_reg128 * r) | 
|  | { | 
|  | unsigned long delay; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_wait_CCR\n"); | 
|  | #endif | 
|  | /* FIXME: need something more descriptive than 100000 :) */ | 
|  | for (delay = 100000; delay; delay--) | 
|  | if (!sbus_readb(&r->r[CD180_CCR])) | 
|  | return; | 
|  | printk(KERN_DEBUG "aurora: Timeout waiting for CCR.\n"); | 
|  | } | 
|  |  | 
|  | /* | 
|  | *  aurora probe functions. | 
|  | */ | 
|  |  | 
|  | /* Must be called with enabled interrupts */ | 
|  | extern inline void aurora_long_delay(unsigned long delay) | 
|  | { | 
|  | unsigned long i; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_long_delay: start\n"); | 
|  | #endif | 
|  | for (i = jiffies + delay; time_before(jiffies, i); ) ; | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_long_delay: end\n"); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | /* Reset and setup CD180 chip */ | 
|  | static int aurora_init_CD180(struct Aurora_board * bp, int chip) | 
|  | { | 
|  | unsigned long flags; | 
|  | int id; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_init_CD180: start %d:%d\n", | 
|  | board_No(bp), chip); | 
|  | #endif | 
|  | save_flags(flags); cli(); | 
|  | sbus_writeb(0, &bp->r[chip]->r[CD180_CAR]); | 
|  | sbus_writeb(0, &bp->r[chip]->r[CD180_GSVR]); | 
|  |  | 
|  | /* Wait for CCR ready        */ | 
|  | aurora_wait_CCR(bp->r[chip]); | 
|  |  | 
|  | /* Reset CD180 chip          */ | 
|  | sbus_writeb(CCR_HARDRESET, &bp->r[chip]->r[CD180_CCR]); | 
|  | udelay(1); | 
|  | sti(); | 
|  | id=1000; | 
|  | while((--id) && | 
|  | (sbus_readb(&bp->r[chip]->r[CD180_GSVR])!=0xff))udelay(100); | 
|  | if(!id) { | 
|  | printk(KERN_ERR "aurora%d: Chip %d failed init.\n", | 
|  | board_No(bp), chip); | 
|  | restore_flags(flags); | 
|  | return(-1); | 
|  | } | 
|  | cli(); | 
|  | sbus_writeb((board_No(bp)<<5)|((chip+1)<<3), | 
|  | &bp->r[chip]->r[CD180_GSVR]); /* Set ID for this chip      */ | 
|  | sbus_writeb(0x80|bp->ACK_MINT, | 
|  | &bp->r[chip]->r[CD180_MSMR]); /* Prio for modem intr       */ | 
|  | sbus_writeb(0x80|bp->ACK_TINT, | 
|  | &bp->r[chip]->r[CD180_TSMR]); /* Prio for transmitter intr */ | 
|  | sbus_writeb(0x80|bp->ACK_RINT, | 
|  | &bp->r[chip]->r[CD180_RSMR]); /* Prio for receiver intr    */ | 
|  | /* Setting up prescaler. We need 4 tick per 1 ms */ | 
|  | sbus_writeb((bp->oscfreq/(1000000/AURORA_TPS)) >> 8, | 
|  | &bp->r[chip]->r[CD180_PPRH]); | 
|  | sbus_writeb((bp->oscfreq/(1000000/AURORA_TPS)) & 0xff, | 
|  | &bp->r[chip]->r[CD180_PPRL]); | 
|  |  | 
|  | sbus_writeb(SRCR_AUTOPRI|SRCR_GLOBPRI, | 
|  | &bp->r[chip]->r[CD180_SRCR]); | 
|  |  | 
|  | id = sbus_readb(&bp->r[chip]->r[CD180_GFRCR]); | 
|  | printk(KERN_INFO "aurora%d: Chip %d id %02x: ", | 
|  | board_No(bp), chip,id); | 
|  | if(sbus_readb(&bp->r[chip]->r[CD180_SRCR]) & 128) { | 
|  | switch (id) { | 
|  | case 0x82:printk("CL-CD1864 rev A\n");break; | 
|  | case 0x83:printk("CL-CD1865 rev A\n");break; | 
|  | case 0x84:printk("CL-CD1865 rev B\n");break; | 
|  | case 0x85:printk("CL-CD1865 rev C\n");break; | 
|  | default:printk("Unknown.\n"); | 
|  | }; | 
|  | } else { | 
|  | switch (id) { | 
|  | case 0x81:printk("CL-CD180 rev B\n");break; | 
|  | case 0x82:printk("CL-CD180 rev C\n");break; | 
|  | default:printk("Unknown.\n"); | 
|  | }; | 
|  | } | 
|  | restore_flags(flags); | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_init_CD180: end\n"); | 
|  | #endif | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int valid_irq(unsigned char irq) | 
|  | { | 
|  | int i; | 
|  | for(i=0;i<TYPE_1_IRQS;i++) | 
|  | if (type_1_irq[i]==irq) return 1; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static irqreturn_t aurora_interrupt(int irq, void * dev_id, struct pt_regs * regs); | 
|  |  | 
|  | /* Main probing routine, also sets irq. */ | 
|  | static int aurora_probe(void) | 
|  | { | 
|  | struct sbus_bus *sbus; | 
|  | struct sbus_dev *sdev; | 
|  | int grrr; | 
|  | char buf[30]; | 
|  | int bn = 0; | 
|  | struct Aurora_board *bp; | 
|  |  | 
|  | for_each_sbus(sbus) { | 
|  | for_each_sbusdev(sdev, sbus) { | 
|  | /*			printk("Try: %x %s\n",sdev,sdev->prom_name);*/ | 
|  | if (!strcmp(sdev->prom_name, "sio16")) { | 
|  | #ifdef AURORA_DEBUG | 
|  | printk(KERN_INFO "aurora: sio16 at %p\n",sdev); | 
|  | #endif | 
|  | if((sdev->reg_addrs[0].reg_size!=1) && | 
|  | (sdev->reg_addrs[1].reg_size!=128) && | 
|  | (sdev->reg_addrs[2].reg_size!=128) && | 
|  | (sdev->reg_addrs[3].reg_size!=4)) { | 
|  | printk(KERN_ERR "aurora%d: registers' sizes " | 
|  | "do not match.\n", bn); | 
|  | break; | 
|  | } | 
|  | bp = &aurora_board[bn]; | 
|  | bp->r0 = (struct aurora_reg1 *) | 
|  | sbus_ioremap(&sdev->resource[0], 0, | 
|  | sdev->reg_addrs[0].reg_size, | 
|  | "sio16"); | 
|  | if (bp->r0 == NULL) { | 
|  | printk(KERN_ERR "aurora%d: can't map " | 
|  | "reg_addrs[0]\n", bn); | 
|  | break; | 
|  | } | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("Map reg 0: %p\n", bp->r0); | 
|  | #endif | 
|  | bp->r[0] = (struct aurora_reg128 *) | 
|  | sbus_ioremap(&sdev->resource[1], 0, | 
|  | sdev->reg_addrs[1].reg_size, | 
|  | "sio16"); | 
|  | if (bp->r[0] == NULL) { | 
|  | printk(KERN_ERR "aurora%d: can't map " | 
|  | "reg_addrs[1]\n", bn); | 
|  | break; | 
|  | } | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("Map reg 1: %p\n", bp->r[0]); | 
|  | #endif | 
|  | bp->r[1] = (struct aurora_reg128 *) | 
|  | sbus_ioremap(&sdev->resource[2], 0, | 
|  | sdev->reg_addrs[2].reg_size, | 
|  | "sio16"); | 
|  | if (bp->r[1] == NULL) { | 
|  | printk(KERN_ERR "aurora%d: can't map " | 
|  | "reg_addrs[2]\n", bn); | 
|  | break; | 
|  | } | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("Map reg 2: %p\n", bp->r[1]); | 
|  | #endif | 
|  | bp->r3 = (struct aurora_reg4 *) | 
|  | sbus_ioremap(&sdev->resource[3], 0, | 
|  | sdev->reg_addrs[3].reg_size, | 
|  | "sio16"); | 
|  | if (bp->r3 == NULL) { | 
|  | printk(KERN_ERR "aurora%d: can't map " | 
|  | "reg_addrs[3]\n", bn); | 
|  | break; | 
|  | } | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("Map reg 3: %p\n", bp->r3); | 
|  | #endif | 
|  | /* Variables setup */ | 
|  | bp->flags = 0; | 
|  | #ifdef AURORA_DEBUG | 
|  | grrr=prom_getint(sdev->prom_node,"intr"); | 
|  | printk("intr pri %d\n", grrr); | 
|  | #endif | 
|  | if ((bp->irq=irqs[bn]) && valid_irq(bp->irq) && | 
|  | !request_irq(bp->irq|0x30, aurora_interrupt, SA_SHIRQ, "sio16", bp)) { | 
|  | free_irq(bp->irq|0x30, bp); | 
|  | } else | 
|  | if ((bp->irq=prom_getint(sdev->prom_node, "bintr")) && valid_irq(bp->irq) && | 
|  | !request_irq(bp->irq|0x30, aurora_interrupt, SA_SHIRQ, "sio16", bp)) { | 
|  | free_irq(bp->irq|0x30, bp); | 
|  | } else | 
|  | if ((bp->irq=prom_getint(sdev->prom_node, "intr")) && valid_irq(bp->irq) && | 
|  | !request_irq(bp->irq|0x30, aurora_interrupt, SA_SHIRQ, "sio16", bp)) { | 
|  | free_irq(bp->irq|0x30, bp); | 
|  | } else | 
|  | for(grrr=0;grrr<TYPE_1_IRQS;grrr++) { | 
|  | if ((bp->irq=type_1_irq[grrr])&&!request_irq(bp->irq|0x30, aurora_interrupt, SA_SHIRQ, "sio16", bp)) { | 
|  | free_irq(bp->irq|0x30, bp); | 
|  | break; | 
|  | } else { | 
|  | printk(KERN_ERR "aurora%d: Could not get an irq for this board !!!\n",bn); | 
|  | bp->flags=0xff; | 
|  | } | 
|  | } | 
|  | if(bp->flags==0xff)break; | 
|  | printk(KERN_INFO "aurora%d: irq %d\n",bn,bp->irq&0x0f); | 
|  | buf[0]=0; | 
|  | grrr=prom_getproperty(sdev->prom_node,"dtr_rts",buf,sizeof(buf)); | 
|  | if(!strcmp(buf,"swapped")){ | 
|  | printk(KERN_INFO "aurora%d: Swapped DTR and RTS\n",bn); | 
|  | bp->DTR=MSVR_RTS; | 
|  | bp->RTS=MSVR_DTR; | 
|  | bp->MSVDTR=CD180_MSVRTS; | 
|  | bp->MSVRTS=CD180_MSVDTR; | 
|  | bp->flags|=AURORA_BOARD_DTR_FLOW_OK; | 
|  | }else{ | 
|  | #ifdef AURORA_FORCE_DTR_FLOW | 
|  | printk(KERN_INFO "aurora%d: Forcing swapped DTR-RTS\n",bn); | 
|  | bp->DTR=MSVR_RTS; | 
|  | bp->RTS=MSVR_DTR; | 
|  | bp->MSVDTR=CD180_MSVRTS; | 
|  | bp->MSVRTS=CD180_MSVDTR; | 
|  | bp->flags|=AURORA_BOARD_DTR_FLOW_OK; | 
|  | #else | 
|  | printk(KERN_INFO "aurora%d: Normal DTR and RTS\n",bn); | 
|  | bp->DTR=MSVR_DTR; | 
|  | bp->RTS=MSVR_RTS; | 
|  | bp->MSVDTR=CD180_MSVDTR; | 
|  | bp->MSVRTS=CD180_MSVRTS; | 
|  | #endif | 
|  | } | 
|  | bp->oscfreq=prom_getint(sdev->prom_node,"clk")*100; | 
|  | printk(KERN_INFO "aurora%d: Oscillator: %d Hz\n",bn,bp->oscfreq); | 
|  | grrr=prom_getproperty(sdev->prom_node,"chip",buf,sizeof(buf)); | 
|  | printk(KERN_INFO "aurora%d: Chips: %s\n",bn,buf); | 
|  | grrr=prom_getproperty(sdev->prom_node,"manu",buf,sizeof(buf)); | 
|  | printk(KERN_INFO "aurora%d: Manufacturer: %s\n",bn,buf); | 
|  | grrr=prom_getproperty(sdev->prom_node,"model",buf,sizeof(buf)); | 
|  | printk(KERN_INFO "aurora%d: Model: %s\n",bn,buf); | 
|  | grrr=prom_getproperty(sdev->prom_node,"rev",buf,sizeof(buf)); | 
|  | printk(KERN_INFO "aurora%d: Revision: %s\n",bn,buf); | 
|  | grrr=prom_getproperty(sdev->prom_node,"mode",buf,sizeof(buf)); | 
|  | printk(KERN_INFO "aurora%d: Mode: %s\n",bn,buf); | 
|  | #ifdef MODULE | 
|  | bp->count=0; | 
|  | #endif | 
|  | bp->flags = AURORA_BOARD_PRESENT; | 
|  | /* hardware ack */ | 
|  | bp->ACK_MINT=1; | 
|  | bp->ACK_TINT=2; | 
|  | bp->ACK_RINT=3; | 
|  | bn++; | 
|  | } | 
|  | } | 
|  | } | 
|  | return bn; | 
|  | } | 
|  |  | 
|  | static void aurora_release_io_range(struct Aurora_board *bp) | 
|  | { | 
|  | sbus_iounmap((unsigned long)bp->r0, 1); | 
|  | sbus_iounmap((unsigned long)bp->r[0], 128); | 
|  | sbus_iounmap((unsigned long)bp->r[1], 128); | 
|  | sbus_iounmap((unsigned long)bp->r3, 4); | 
|  | } | 
|  |  | 
|  | extern inline void aurora_mark_event(struct Aurora_port * port, int event) | 
|  | { | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_mark_event: start\n"); | 
|  | #endif | 
|  | set_bit(event, &port->event); | 
|  | queue_task(&port->tqueue, &tq_aurora); | 
|  | mark_bh(AURORA_BH); | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_mark_event: end\n"); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | static __inline__ struct Aurora_port * aurora_get_port(struct Aurora_board const * bp, | 
|  | int chip, | 
|  | unsigned char const *what) | 
|  | { | 
|  | unsigned char channel; | 
|  | struct Aurora_port * port; | 
|  |  | 
|  | channel = ((chip << 3) | | 
|  | ((sbus_readb(&bp->r[chip]->r[CD180_GSCR]) & GSCR_CHAN) >> GSCR_CHAN_OFF)); | 
|  | port = &aurora_port[board_No(bp) * AURORA_NPORT * AURORA_NCD180 + channel]; | 
|  | if (port->flags & ASYNC_INITIALIZED) | 
|  | return port; | 
|  |  | 
|  | printk(KERN_DEBUG "aurora%d: %s interrupt from invalid port %d\n", | 
|  | board_No(bp), what, channel); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | static void aurora_receive_exc(struct Aurora_board const * bp, int chip) | 
|  | { | 
|  | struct Aurora_port *port; | 
|  | struct tty_struct *tty; | 
|  | unsigned char status; | 
|  | unsigned char ch; | 
|  |  | 
|  | if (!(port = aurora_get_port(bp, chip, "Receive_x"))) | 
|  | return; | 
|  |  | 
|  | tty = port->tty; | 
|  | if (tty->flip.count >= TTY_FLIPBUF_SIZE)  { | 
|  | #ifdef AURORA_INTNORM | 
|  | printk("aurora%d: port %d: Working around flip buffer overflow.\n", | 
|  | board_No(bp), port_No(port)); | 
|  | #endif | 
|  | return; | 
|  | } | 
|  |  | 
|  | #ifdef AURORA_REPORT_OVERRUN | 
|  | status = sbus_readb(&bp->r[chip]->r[CD180_RCSR]); | 
|  | if (status & RCSR_OE)  { | 
|  | port->overrun++; | 
|  | #if 1 | 
|  | printk("aurora%d: port %d: Overrun. Total %ld overruns.\n", | 
|  | board_No(bp), port_No(port), port->overrun); | 
|  | #endif | 
|  | } | 
|  | status &= port->mark_mask; | 
|  | #else | 
|  | status = sbus_readb(&bp->r[chip]->r[CD180_RCSR]) & port->mark_mask; | 
|  | #endif | 
|  | ch = sbus_readb(&bp->r[chip]->r[CD180_RDR]); | 
|  | if (!status) | 
|  | return; | 
|  |  | 
|  | if (status & RCSR_TOUT)  { | 
|  | /*		printk("aurora%d: port %d: Receiver timeout. Hardware problems ?\n", | 
|  | board_No(bp), port_No(port));*/ | 
|  | return; | 
|  |  | 
|  | } else if (status & RCSR_BREAK)  { | 
|  | printk(KERN_DEBUG "aurora%d: port %d: Handling break...\n", | 
|  | board_No(bp), port_No(port)); | 
|  | *tty->flip.flag_buf_ptr++ = TTY_BREAK; | 
|  | if (port->flags & ASYNC_SAK) | 
|  | do_SAK(tty); | 
|  |  | 
|  | } else if (status & RCSR_PE) | 
|  | *tty->flip.flag_buf_ptr++ = TTY_PARITY; | 
|  |  | 
|  | else if (status & RCSR_FE) | 
|  | *tty->flip.flag_buf_ptr++ = TTY_FRAME; | 
|  |  | 
|  | else if (status & RCSR_OE) | 
|  | *tty->flip.flag_buf_ptr++ = TTY_OVERRUN; | 
|  |  | 
|  | else | 
|  | *tty->flip.flag_buf_ptr++ = 0; | 
|  |  | 
|  | *tty->flip.char_buf_ptr++ = ch; | 
|  | tty->flip.count++; | 
|  | queue_task(&tty->flip.tqueue, &tq_timer); | 
|  | } | 
|  |  | 
|  | static void aurora_receive(struct Aurora_board const * bp, int chip) | 
|  | { | 
|  | struct Aurora_port *port; | 
|  | struct tty_struct *tty; | 
|  | unsigned char count,cnt; | 
|  |  | 
|  | if (!(port = aurora_get_port(bp, chip, "Receive"))) | 
|  | return; | 
|  |  | 
|  | tty = port->tty; | 
|  |  | 
|  | count = sbus_readb(&bp->r[chip]->r[CD180_RDCR]); | 
|  |  | 
|  | #ifdef AURORA_REPORT_FIFO | 
|  | port->hits[count > 8 ? 9 : count]++; | 
|  | #endif | 
|  |  | 
|  | while (count--)  { | 
|  | if (tty->flip.count >= TTY_FLIPBUF_SIZE)  { | 
|  | #ifdef AURORA_INTNORM | 
|  | printk("aurora%d: port %d: Working around flip buffer overflow.\n", | 
|  | board_No(bp), port_No(port)); | 
|  | #endif | 
|  | break; | 
|  | } | 
|  | cnt = sbus_readb(&bp->r[chip]->r[CD180_RDR]); | 
|  | *tty->flip.char_buf_ptr++ = cnt; | 
|  | *tty->flip.flag_buf_ptr++ = 0; | 
|  | tty->flip.count++; | 
|  | } | 
|  | queue_task(&tty->flip.tqueue, &tq_timer); | 
|  | } | 
|  |  | 
|  | static void aurora_transmit(struct Aurora_board const * bp, int chip) | 
|  | { | 
|  | struct Aurora_port *port; | 
|  | struct tty_struct *tty; | 
|  | unsigned char count; | 
|  |  | 
|  | if (!(port = aurora_get_port(bp, chip, "Transmit"))) | 
|  | return; | 
|  |  | 
|  | tty = port->tty; | 
|  |  | 
|  | if (port->SRER & SRER_TXEMPTY)  { | 
|  | /* FIFO drained */ | 
|  | sbus_writeb(port_No(port) & 7, | 
|  | &bp->r[chip]->r[CD180_CAR]); | 
|  | udelay(1); | 
|  | port->SRER &= ~SRER_TXEMPTY; | 
|  | sbus_writeb(port->SRER, &bp->r[chip]->r[CD180_SRER]); | 
|  | return; | 
|  | } | 
|  |  | 
|  | if ((port->xmit_cnt <= 0 && !port->break_length) | 
|  | || tty->stopped || tty->hw_stopped)  { | 
|  | sbus_writeb(port_No(port) & 7, | 
|  | &bp->r[chip]->r[CD180_CAR]); | 
|  | udelay(1); | 
|  | port->SRER &= ~SRER_TXRDY; | 
|  | sbus_writeb(port->SRER, | 
|  | &bp->r[chip]->r[CD180_SRER]); | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (port->break_length)  { | 
|  | if (port->break_length > 0)  { | 
|  | if (port->COR2 & COR2_ETC)  { | 
|  | sbus_writeb(CD180_C_ESC, | 
|  | &bp->r[chip]->r[CD180_TDR]); | 
|  | sbus_writeb(CD180_C_SBRK, | 
|  | &bp->r[chip]->r[CD180_TDR]); | 
|  | port->COR2 &= ~COR2_ETC; | 
|  | } | 
|  | count = min(port->break_length, 0xff); | 
|  | sbus_writeb(CD180_C_ESC, | 
|  | &bp->r[chip]->r[CD180_TDR]); | 
|  | sbus_writeb(CD180_C_DELAY, | 
|  | &bp->r[chip]->r[CD180_TDR]); | 
|  | sbus_writeb(count, | 
|  | &bp->r[chip]->r[CD180_TDR]); | 
|  | if (!(port->break_length -= count)) | 
|  | port->break_length--; | 
|  | } else  { | 
|  | sbus_writeb(CD180_C_ESC, | 
|  | &bp->r[chip]->r[CD180_TDR]); | 
|  | sbus_writeb(CD180_C_EBRK, | 
|  | &bp->r[chip]->r[CD180_TDR]); | 
|  | sbus_writeb(port->COR2, | 
|  | &bp->r[chip]->r[CD180_COR2]); | 
|  | aurora_wait_CCR(bp->r[chip]); | 
|  | sbus_writeb(CCR_CORCHG2, | 
|  | &bp->r[chip]->r[CD180_CCR]); | 
|  | port->break_length = 0; | 
|  | } | 
|  | return; | 
|  | } | 
|  |  | 
|  | count = CD180_NFIFO; | 
|  | do { | 
|  | u8 byte = port->xmit_buf[port->xmit_tail++]; | 
|  |  | 
|  | sbus_writeb(byte, &bp->r[chip]->r[CD180_TDR]); | 
|  | port->xmit_tail = port->xmit_tail & (SERIAL_XMIT_SIZE-1); | 
|  | if (--port->xmit_cnt <= 0) | 
|  | break; | 
|  | } while (--count > 0); | 
|  |  | 
|  | if (port->xmit_cnt <= 0)  { | 
|  | sbus_writeb(port_No(port) & 7, | 
|  | &bp->r[chip]->r[CD180_CAR]); | 
|  | udelay(1); | 
|  | port->SRER &= ~SRER_TXRDY; | 
|  | sbus_writeb(port->SRER, | 
|  | &bp->r[chip]->r[CD180_SRER]); | 
|  | } | 
|  | if (port->xmit_cnt <= port->wakeup_chars) | 
|  | aurora_mark_event(port, RS_EVENT_WRITE_WAKEUP); | 
|  | } | 
|  |  | 
|  | static void aurora_check_modem(struct Aurora_board const * bp, int chip) | 
|  | { | 
|  | struct Aurora_port *port; | 
|  | struct tty_struct *tty; | 
|  | unsigned char mcr; | 
|  |  | 
|  | if (!(port = aurora_get_port(bp, chip, "Modem"))) | 
|  | return; | 
|  |  | 
|  | tty = port->tty; | 
|  |  | 
|  | mcr = sbus_readb(&bp->r[chip]->r[CD180_MCR]); | 
|  | if (mcr & MCR_CDCHG)  { | 
|  | if (sbus_readb(&bp->r[chip]->r[CD180_MSVR]) & MSVR_CD) | 
|  | wake_up_interruptible(&port->open_wait); | 
|  | else | 
|  | schedule_task(&port->tqueue_hangup); | 
|  | } | 
|  |  | 
|  | /* We don't have such things yet. My aurora board has DTR and RTS swapped, but that doesn't count in this driver. Let's hope | 
|  | * Aurora didn't made any boards with CTS or DSR broken... | 
|  | */ | 
|  | /* #ifdef AURORA_BRAIN_DAMAGED_CTS | 
|  | if (mcr & MCR_CTSCHG)  { | 
|  | if (aurora_in(bp, CD180_MSVR) & MSVR_CTS)  { | 
|  | tty->hw_stopped = 0; | 
|  | port->SRER |= SRER_TXRDY; | 
|  | if (port->xmit_cnt <= port->wakeup_chars) | 
|  | aurora_mark_event(port, RS_EVENT_WRITE_WAKEUP); | 
|  | } else  { | 
|  | tty->hw_stopped = 1; | 
|  | port->SRER &= ~SRER_TXRDY; | 
|  | } | 
|  | sbus_writeb(port->SRER, &bp->r[chip]->r[CD180_SRER]); | 
|  | } | 
|  | if (mcr & MCR_DSRCHG)  { | 
|  | if (aurora_in(bp, CD180_MSVR) & MSVR_DSR)  { | 
|  | tty->hw_stopped = 0; | 
|  | port->SRER |= SRER_TXRDY; | 
|  | if (port->xmit_cnt <= port->wakeup_chars) | 
|  | aurora_mark_event(port, RS_EVENT_WRITE_WAKEUP); | 
|  | } else  { | 
|  | tty->hw_stopped = 1; | 
|  | port->SRER &= ~SRER_TXRDY; | 
|  | } | 
|  | sbus_writeb(port->SRER, &bp->r[chip]->r[CD180_SRER]); | 
|  | } | 
|  | #endif AURORA_BRAIN_DAMAGED_CTS */ | 
|  |  | 
|  | /* Clear change bits */ | 
|  | sbus_writeb(0, &bp->r[chip]->r[CD180_MCR]); | 
|  | } | 
|  |  | 
|  | /* The main interrupt processing routine */ | 
|  | static irqreturn_t aurora_interrupt(int irq, void * dev_id, struct pt_regs * regs) | 
|  | { | 
|  | unsigned char status; | 
|  | unsigned char ack,chip/*,chip_id*/; | 
|  | struct Aurora_board * bp = (struct Aurora_board *) dev_id; | 
|  | unsigned long loop = 0; | 
|  |  | 
|  | #ifdef AURORA_INT_DEBUG | 
|  | printk("IRQ%d %d\n",irq,++irqhit); | 
|  | #ifdef AURORA_FLOODPRO | 
|  | if (irqhit>=AURORA_FLOODPRO) | 
|  | sbus_writeb(8, &bp->r0->r); | 
|  | #endif | 
|  | #endif | 
|  |  | 
|  | /* old	bp = IRQ_to_board[irq&0x0f];*/ | 
|  |  | 
|  | if (!bp || !(bp->flags & AURORA_BOARD_ACTIVE)) | 
|  | return IRQ_NONE; | 
|  |  | 
|  | /*	The while() below takes care of this. | 
|  | status = sbus_readb(&bp->r[0]->r[CD180_SRSR]); | 
|  | #ifdef AURORA_INT_DEBUG | 
|  | printk("mumu: %02x\n", status); | 
|  | #endif | 
|  | if (!(status&SRSR_ANYINT)) | 
|  | return IRQ_NONE; * Nobody has anything to say, so exit * | 
|  | */ | 
|  | while ((loop++ < 48) && | 
|  | (status = sbus_readb(&bp->r[0]->r[CD180_SRSR]) & SRSR_ANYINT)){ | 
|  | #ifdef AURORA_INT_DEBUG | 
|  | printk("SRSR: %02x\n", status); | 
|  | #endif | 
|  | if (status & SRSR_REXT) { | 
|  | ack = sbus_readb(&bp->r3->r[bp->ACK_RINT]); | 
|  | #ifdef AURORA_INT_DEBUG | 
|  | printk("R-ACK %02x\n", ack); | 
|  | #endif | 
|  | if ((ack >> 5) == board_No(bp)) { | 
|  | if ((chip=((ack>>3)&3)-1) < AURORA_NCD180) { | 
|  | if ((ack&GSVR_ITMASK)==GSVR_IT_RGD) { | 
|  | aurora_receive(bp,chip); | 
|  | sbus_writeb(0, | 
|  | &bp->r[chip]->r[CD180_EOSRR]); | 
|  | } else if ((ack & GSVR_ITMASK) == GSVR_IT_REXC) { | 
|  | aurora_receive_exc(bp,chip); | 
|  | sbus_writeb(0, | 
|  | &bp->r[chip]->r[CD180_EOSRR]); | 
|  | } | 
|  | } | 
|  | } | 
|  | } else if (status & SRSR_TEXT) { | 
|  | ack = sbus_readb(&bp->r3->r[bp->ACK_TINT]); | 
|  | #ifdef AURORA_INT_DEBUG | 
|  | printk("T-ACK %02x\n", ack); | 
|  | #endif | 
|  | if ((ack >> 5) == board_No(bp)) { | 
|  | if ((chip=((ack>>3)&3)-1) < AURORA_NCD180) { | 
|  | if ((ack&GSVR_ITMASK)==GSVR_IT_TX) { | 
|  | aurora_transmit(bp,chip); | 
|  | sbus_writeb(0, | 
|  | &bp->r[chip]->r[CD180_EOSRR]); | 
|  | } | 
|  | } | 
|  | } | 
|  | } else if (status & SRSR_MEXT) { | 
|  | ack = sbus_readb(&bp->r3->r[bp->ACK_MINT]); | 
|  | #ifdef AURORA_INT_DEBUG | 
|  | printk("M-ACK %02x\n", ack); | 
|  | #endif | 
|  | if ((ack >> 5) == board_No(bp)) { | 
|  | if ((chip = ((ack>>3)&3)-1) < AURORA_NCD180) { | 
|  | if ((ack&GSVR_ITMASK)==GSVR_IT_MDM) { | 
|  | aurora_check_modem(bp,chip); | 
|  | sbus_writeb(0, | 
|  | &bp->r[chip]->r[CD180_EOSRR]); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | /* I guess this faster code can be used with CD1865, using AUROPRI and GLOBPRI. */ | 
|  | #if 0 | 
|  | while ((loop++ < 48)&&(status=bp->r[0]->r[CD180_SRSR]&SRSR_ANYINT)){ | 
|  | #ifdef AURORA_INT_DEBUG | 
|  | printk("SRSR: %02x\n",status); | 
|  | #endif | 
|  | ack = sbus_readb(&bp->r3->r[0]); | 
|  | #ifdef AURORA_INT_DEBUG | 
|  | printk("ACK: %02x\n",ack); | 
|  | #endif | 
|  | if ((ack>>5)==board_No(bp)) { | 
|  | if ((chip=((ack>>3)&3)-1) < AURORA_NCD180) { | 
|  | ack&=GSVR_ITMASK; | 
|  | if (ack==GSVR_IT_RGD) { | 
|  | aurora_receive(bp,chip); | 
|  | sbus_writeb(0, | 
|  | &bp->r[chip]->r[CD180_EOSRR]); | 
|  | } else if (ack==GSVR_IT_REXC) { | 
|  | aurora_receive_exc(bp,chip); | 
|  | sbus_writeb(0, | 
|  | &bp->r[chip]->r[CD180_EOSRR]); | 
|  | } else if (ack==GSVR_IT_TX) { | 
|  | aurora_transmit(bp,chip); | 
|  | sbus_writeb(0, | 
|  | &bp->r[chip]->r[CD180_EOSRR]); | 
|  | } else if (ack==GSVR_IT_MDM) { | 
|  | aurora_check_modem(bp,chip); | 
|  | sbus_writeb(0, | 
|  | &bp->r[chip]->r[CD180_EOSRR]); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | #endif | 
|  |  | 
|  | /* This is the old handling routine, used in riscom8 for only one CD180. I keep it here for reference. */ | 
|  | #if 0 | 
|  | for(chip=0;chip<AURORA_NCD180;chip++){ | 
|  | chip_id=(board_No(bp)<<5)|((chip+1)<<3); | 
|  | loop=0; | 
|  | while ((loop++ < 1) && | 
|  | ((status = sbus_readb(&bp->r[chip]->r[CD180_SRSR])) & | 
|  | (SRSR_TEXT | SRSR_MEXT | SRSR_REXT))) { | 
|  |  | 
|  | if (status & SRSR_REXT) { | 
|  | ack = sbus_readb(&bp->r3->r[bp->ACK_RINT]); | 
|  | if (ack == (chip_id | GSVR_IT_RGD)) { | 
|  | #ifdef AURORA_INTMSG | 
|  | printk("RX ACK\n"); | 
|  | #endif | 
|  | aurora_receive(bp,chip); | 
|  | } else if (ack == (chip_id | GSVR_IT_REXC)) { | 
|  | #ifdef AURORA_INTMSG | 
|  | printk("RXC ACK\n"); | 
|  | #endif | 
|  | aurora_receive_exc(bp,chip); | 
|  | } else { | 
|  | #ifdef AURORA_INTNORM | 
|  | printk("aurora%d-%d: Bad receive ack 0x%02x.\n", | 
|  | board_No(bp), chip, ack); | 
|  | #endif | 
|  | } | 
|  | } else if (status & SRSR_TEXT) { | 
|  | ack = sbus_readb(&bp->r3->r[bp->ACK_TINT]); | 
|  | if (ack == (chip_id | GSVR_IT_TX)){ | 
|  | #ifdef AURORA_INTMSG | 
|  | printk("TX ACK\n"); | 
|  | #endif | 
|  | aurora_transmit(bp,chip); | 
|  | } else { | 
|  | #ifdef AURORA_INTNORM | 
|  | printk("aurora%d-%d: Bad transmit ack 0x%02x.\n", | 
|  | board_No(bp), chip, ack); | 
|  | #endif | 
|  | } | 
|  | } else  if (status & SRSR_MEXT)  { | 
|  | ack = sbus_readb(&bp->r3->r[bp->ACK_MINT]); | 
|  | if (ack == (chip_id | GSVR_IT_MDM)){ | 
|  | #ifdef AURORA_INTMSG | 
|  | printk("MDM ACK\n"); | 
|  | #endif | 
|  | aurora_check_modem(bp,chip); | 
|  | } else { | 
|  | #ifdef AURORA_INTNORM | 
|  | printk("aurora%d-%d: Bad modem ack 0x%02x.\n", | 
|  | board_No(bp), chip, ack); | 
|  | #endif | 
|  | } | 
|  | } | 
|  | sbus_writeb(0, &bp->r[chip]->r[CD180_EOSRR]); | 
|  | } | 
|  | } | 
|  | #endif | 
|  |  | 
|  | return IRQ_HANDLED; | 
|  | } | 
|  |  | 
|  | #ifdef AURORA_INT_DEBUG | 
|  | static void aurora_timer (unsigned long ignored); | 
|  |  | 
|  | static struct timer_list aurora_poll_timer = | 
|  | TIMER_INITIALIZER(aurora_timer, 0, 0); | 
|  |  | 
|  | static void | 
|  | aurora_timer (unsigned long ignored) | 
|  | { | 
|  | unsigned long flags; | 
|  | int i; | 
|  |  | 
|  | save_flags(flags); cli(); | 
|  |  | 
|  | printk("SRSR: %02x,%02x - ", | 
|  | sbus_readb(&aurora_board[0].r[0]->r[CD180_SRSR]), | 
|  | sbus_readb(&aurora_board[0].r[1]->r[CD180_SRSR])); | 
|  | for (i = 0; i < 4; i++) { | 
|  | udelay(1); | 
|  | printk("%02x ", | 
|  | sbus_readb(&aurora_board[0].r3->r[i])); | 
|  | } | 
|  | printk("\n"); | 
|  |  | 
|  | aurora_poll_timer.expires = jiffies + 300; | 
|  | add_timer (&aurora_poll_timer); | 
|  |  | 
|  | restore_flags(flags); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | /* | 
|  | *  Routines for open & close processing. | 
|  | */ | 
|  |  | 
|  | /* Called with disabled interrupts */ | 
|  | static int aurora_setup_board(struct Aurora_board * bp) | 
|  | { | 
|  | int error; | 
|  |  | 
|  | #ifdef AURORA_ALLIRQ | 
|  | int i; | 
|  | for (i = 0; i < AURORA_ALLIRQ; i++) { | 
|  | error = request_irq(allirq[i]|0x30, aurora_interrupt, SA_SHIRQ, | 
|  | "sio16", bp); | 
|  | if (error) | 
|  | printk(KERN_ERR "IRQ%d request error %d\n", | 
|  | allirq[i], error); | 
|  | } | 
|  | #else | 
|  | error = request_irq(bp->irq|0x30, aurora_interrupt, SA_SHIRQ, | 
|  | "sio16", bp); | 
|  | if (error) { | 
|  | printk(KERN_ERR "IRQ request error %d\n", error); | 
|  | return error; | 
|  | } | 
|  | #endif | 
|  | /* Board reset */ | 
|  | sbus_writeb(0, &bp->r0->r); | 
|  | udelay(1); | 
|  | if (bp->flags & AURORA_BOARD_TYPE_2) { | 
|  | /* unknown yet */ | 
|  | } else { | 
|  | sbus_writeb((AURORA_CFG_ENABLE_IO | AURORA_CFG_ENABLE_IRQ | | 
|  | (((bp->irq)&0x0f)>>2)), | 
|  | &bp->r0->r); | 
|  | } | 
|  | udelay(10000); | 
|  |  | 
|  | if (aurora_init_CD180(bp,0))error=1;error=0; | 
|  | if (aurora_init_CD180(bp,1))error++; | 
|  | if (error == AURORA_NCD180) { | 
|  | printk(KERN_ERR "Both chips failed initialisation.\n"); | 
|  | return -EIO; | 
|  | } | 
|  |  | 
|  | #ifdef AURORA_INT_DEBUG | 
|  | aurora_poll_timer.expires= jiffies + 1; | 
|  | add_timer(&aurora_poll_timer); | 
|  | #endif | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_setup_board: end\n"); | 
|  | #endif | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* Called with disabled interrupts */ | 
|  | static void aurora_shutdown_board(struct Aurora_board *bp) | 
|  | { | 
|  | int i; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_shutdown_board: start\n"); | 
|  | #endif | 
|  |  | 
|  | #ifdef AURORA_INT_DEBUG | 
|  | del_timer(&aurora_poll_timer); | 
|  | #endif | 
|  |  | 
|  | #ifdef AURORA_ALLIRQ | 
|  | for(i=0;i<AURORA_ALLIRQ;i++){ | 
|  | free_irq(allirq[i]|0x30, bp); | 
|  | /*		IRQ_to_board[allirq[i]&0xf] = NULL;*/ | 
|  | } | 
|  | #else | 
|  | free_irq(bp->irq|0x30, bp); | 
|  | /*	IRQ_to_board[bp->irq&0xf] = NULL;*/ | 
|  | #endif | 
|  | /* Drop all DTR's */ | 
|  | for(i=0;i<16;i++){ | 
|  | sbus_writeb(i & 7, &bp->r[i>>3]->r[CD180_CAR]); | 
|  | udelay(1); | 
|  | sbus_writeb(0, &bp->r[i>>3]->r[CD180_MSVR]); | 
|  | udelay(1); | 
|  | } | 
|  | /* Board shutdown */ | 
|  | sbus_writeb(0, &bp->r0->r); | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_shutdown_board: end\n"); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | /* Setting up port characteristics. | 
|  | * Must be called with disabled interrupts | 
|  | */ | 
|  | static void aurora_change_speed(struct Aurora_board *bp, struct Aurora_port *port) | 
|  | { | 
|  | struct tty_struct *tty; | 
|  | unsigned long baud; | 
|  | long tmp; | 
|  | unsigned char cor1 = 0, cor3 = 0; | 
|  | unsigned char mcor1 = 0, mcor2 = 0,chip; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_change_speed: start\n"); | 
|  | #endif | 
|  | if (!(tty = port->tty) || !tty->termios) | 
|  | return; | 
|  |  | 
|  | chip = AURORA_CD180(port_No(port)); | 
|  |  | 
|  | port->SRER  = 0; | 
|  | port->COR2 = 0; | 
|  | port->MSVR = MSVR_RTS|MSVR_DTR; | 
|  |  | 
|  | baud = tty_get_baud_rate(tty); | 
|  |  | 
|  | /* Select port on the board */ | 
|  | sbus_writeb(port_No(port) & 7, | 
|  | &bp->r[chip]->r[CD180_CAR]); | 
|  | udelay(1); | 
|  |  | 
|  | if (!baud)  { | 
|  | /* Drop DTR & exit */ | 
|  | port->MSVR &= ~(bp->DTR|bp->RTS); | 
|  | sbus_writeb(port->MSVR, | 
|  | &bp->r[chip]->r[CD180_MSVR]); | 
|  | return; | 
|  | } else  { | 
|  | /* Set DTR on */ | 
|  | port->MSVR |= bp->DTR; | 
|  | sbus_writeb(port->MSVR, | 
|  | &bp->r[chip]->r[CD180_MSVR]); | 
|  | } | 
|  |  | 
|  | /* Now we must calculate some speed dependent things. */ | 
|  |  | 
|  | /* Set baud rate for port. */ | 
|  | tmp = (((bp->oscfreq + baud/2) / baud + | 
|  | CD180_TPC/2) / CD180_TPC); | 
|  |  | 
|  | /*	tmp = (bp->oscfreq/7)/baud; | 
|  | if((tmp%10)>4)tmp=tmp/10+1;else tmp=tmp/10;*/ | 
|  | /*	printk("Prescaler period: %d\n",tmp);*/ | 
|  |  | 
|  | sbus_writeb((tmp >> 8) & 0xff, | 
|  | &bp->r[chip]->r[CD180_RBPRH]); | 
|  | sbus_writeb((tmp >> 8) & 0xff, | 
|  | &bp->r[chip]->r[CD180_TBPRH]); | 
|  | sbus_writeb(tmp & 0xff, &bp->r[chip]->r[CD180_RBPRL]); | 
|  | sbus_writeb(tmp & 0xff, &bp->r[chip]->r[CD180_TBPRL]); | 
|  |  | 
|  | baud = (baud + 5) / 10;   /* Estimated CPS */ | 
|  |  | 
|  | /* Two timer ticks seems enough to wakeup something like SLIP driver */ | 
|  | tmp = ((baud + HZ/2) / HZ) * 2 - CD180_NFIFO; | 
|  | port->wakeup_chars = (tmp < 0) ? 0 : ((tmp >= SERIAL_XMIT_SIZE) ? | 
|  | SERIAL_XMIT_SIZE - 1 : tmp); | 
|  |  | 
|  | /* Receiver timeout will be transmission time for 1.5 chars */ | 
|  | tmp = (AURORA_TPS + AURORA_TPS/2 + baud/2) / baud; | 
|  | tmp = (tmp > 0xff) ? 0xff : tmp; | 
|  | sbus_writeb(tmp, &bp->r[chip]->r[CD180_RTPR]); | 
|  |  | 
|  | switch (C_CSIZE(tty))  { | 
|  | case CS5: | 
|  | cor1 |= COR1_5BITS; | 
|  | break; | 
|  | case CS6: | 
|  | cor1 |= COR1_6BITS; | 
|  | break; | 
|  | case CS7: | 
|  | cor1 |= COR1_7BITS; | 
|  | break; | 
|  | case CS8: | 
|  | cor1 |= COR1_8BITS; | 
|  | break; | 
|  | } | 
|  |  | 
|  | if (C_CSTOPB(tty)) | 
|  | cor1 |= COR1_2SB; | 
|  |  | 
|  | cor1 |= COR1_IGNORE; | 
|  | if (C_PARENB(tty))  { | 
|  | cor1 |= COR1_NORMPAR; | 
|  | if (C_PARODD(tty)) | 
|  | cor1 |= COR1_ODDP; | 
|  | if (I_INPCK(tty)) | 
|  | cor1 &= ~COR1_IGNORE; | 
|  | } | 
|  | /* Set marking of some errors */ | 
|  | port->mark_mask = RCSR_OE | RCSR_TOUT; | 
|  | if (I_INPCK(tty)) | 
|  | port->mark_mask |= RCSR_FE | RCSR_PE; | 
|  | if (I_BRKINT(tty) || I_PARMRK(tty)) | 
|  | port->mark_mask |= RCSR_BREAK; | 
|  | if (I_IGNPAR(tty)) | 
|  | port->mark_mask &= ~(RCSR_FE | RCSR_PE); | 
|  | if (I_IGNBRK(tty))  { | 
|  | port->mark_mask &= ~RCSR_BREAK; | 
|  | if (I_IGNPAR(tty)) | 
|  | /* Real raw mode. Ignore all */ | 
|  | port->mark_mask &= ~RCSR_OE; | 
|  | } | 
|  | /* Enable Hardware Flow Control */ | 
|  | if (C_CRTSCTS(tty))  { | 
|  | /*#ifdef AURORA_BRAIN_DAMAGED_CTS | 
|  | port->SRER |= SRER_DSR | SRER_CTS; | 
|  | mcor1 |= MCOR1_DSRZD | MCOR1_CTSZD; | 
|  | mcor2 |= MCOR2_DSROD | MCOR2_CTSOD; | 
|  | tty->hw_stopped = !(aurora_in(bp, CD180_MSVR) & (MSVR_CTS|MSVR_DSR)); | 
|  | #else*/ | 
|  | port->COR2 |= COR2_CTSAE; | 
|  | /*#endif*/ | 
|  | if (bp->flags&AURORA_BOARD_DTR_FLOW_OK) { | 
|  | mcor1 |= AURORA_RXTH; | 
|  | } | 
|  | } | 
|  | /* Enable Software Flow Control. FIXME: I'm not sure about this */ | 
|  | /* Some people reported that it works, but I still doubt */ | 
|  | if (I_IXON(tty))  { | 
|  | port->COR2 |= COR2_TXIBE; | 
|  | cor3 |= (COR3_FCT | COR3_SCDE); | 
|  | if (I_IXANY(tty)) | 
|  | port->COR2 |= COR2_IXM; | 
|  | sbus_writeb(START_CHAR(tty), | 
|  | &bp->r[chip]->r[CD180_SCHR1]); | 
|  | sbus_writeb(STOP_CHAR(tty), | 
|  | &bp->r[chip]->r[CD180_SCHR2]); | 
|  | sbus_writeb(START_CHAR(tty), | 
|  | &bp->r[chip]->r[CD180_SCHR3]); | 
|  | sbus_writeb(STOP_CHAR(tty), | 
|  | &bp->r[chip]->r[CD180_SCHR4]); | 
|  | } | 
|  | if (!C_CLOCAL(tty))  { | 
|  | /* Enable CD check */ | 
|  | port->SRER |= SRER_CD; | 
|  | mcor1 |= MCOR1_CDZD; | 
|  | mcor2 |= MCOR2_CDOD; | 
|  | } | 
|  |  | 
|  | if (C_CREAD(tty)) | 
|  | /* Enable receiver */ | 
|  | port->SRER |= SRER_RXD; | 
|  |  | 
|  | /* Set input FIFO size (1-8 bytes) */ | 
|  | cor3 |= AURORA_RXFIFO; | 
|  | /* Setting up CD180 channel registers */ | 
|  | sbus_writeb(cor1, &bp->r[chip]->r[CD180_COR1]); | 
|  | sbus_writeb(port->COR2, &bp->r[chip]->r[CD180_COR2]); | 
|  | sbus_writeb(cor3, &bp->r[chip]->r[CD180_COR3]); | 
|  | /* Make CD180 know about registers change */ | 
|  | aurora_wait_CCR(bp->r[chip]); | 
|  | sbus_writeb(CCR_CORCHG1 | CCR_CORCHG2 | CCR_CORCHG3, | 
|  | &bp->r[chip]->r[CD180_CCR]); | 
|  | /* Setting up modem option registers */ | 
|  | sbus_writeb(mcor1, &bp->r[chip]->r[CD180_MCOR1]); | 
|  | sbus_writeb(mcor2, &bp->r[chip]->r[CD180_MCOR2]); | 
|  | /* Enable CD180 transmitter & receiver */ | 
|  | aurora_wait_CCR(bp->r[chip]); | 
|  | sbus_writeb(CCR_TXEN | CCR_RXEN, &bp->r[chip]->r[CD180_CCR]); | 
|  | /* Enable interrupts */ | 
|  | sbus_writeb(port->SRER, &bp->r[chip]->r[CD180_SRER]); | 
|  | /* And finally set RTS on */ | 
|  | sbus_writeb(port->MSVR, &bp->r[chip]->r[CD180_MSVR]); | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_change_speed: end\n"); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | /* Must be called with interrupts enabled */ | 
|  | static int aurora_setup_port(struct Aurora_board *bp, struct Aurora_port *port) | 
|  | { | 
|  | unsigned long flags; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_setup_port: start %d\n",port_No(port)); | 
|  | #endif | 
|  | if (port->flags & ASYNC_INITIALIZED) | 
|  | return 0; | 
|  |  | 
|  | if (!port->xmit_buf) { | 
|  | /* We may sleep in get_zeroed_page() */ | 
|  | unsigned long tmp; | 
|  |  | 
|  | if (!(tmp = get_zeroed_page(GFP_KERNEL))) | 
|  | return -ENOMEM; | 
|  |  | 
|  | if (port->xmit_buf) { | 
|  | free_page(tmp); | 
|  | return -ERESTARTSYS; | 
|  | } | 
|  | port->xmit_buf = (unsigned char *) tmp; | 
|  | } | 
|  |  | 
|  | save_flags(flags); cli(); | 
|  |  | 
|  | if (port->tty) | 
|  | clear_bit(TTY_IO_ERROR, &port->tty->flags); | 
|  |  | 
|  | #ifdef MODULE | 
|  | if ((port->count == 1) && ((++bp->count) == 1)) | 
|  | bp->flags |= AURORA_BOARD_ACTIVE; | 
|  | #endif | 
|  |  | 
|  | port->xmit_cnt = port->xmit_head = port->xmit_tail = 0; | 
|  | aurora_change_speed(bp, port); | 
|  | port->flags |= ASYNC_INITIALIZED; | 
|  |  | 
|  | restore_flags(flags); | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_setup_port: end\n"); | 
|  | #endif | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* Must be called with interrupts disabled */ | 
|  | static void aurora_shutdown_port(struct Aurora_board *bp, struct Aurora_port *port) | 
|  | { | 
|  | struct tty_struct *tty; | 
|  | unsigned char chip; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_shutdown_port: start\n"); | 
|  | #endif | 
|  | if (!(port->flags & ASYNC_INITIALIZED)) | 
|  | return; | 
|  |  | 
|  | chip = AURORA_CD180(port_No(port)); | 
|  |  | 
|  | #ifdef AURORA_REPORT_OVERRUN | 
|  | printk("aurora%d: port %d: Total %ld overruns were detected.\n", | 
|  | board_No(bp), port_No(port), port->overrun); | 
|  | #endif | 
|  | #ifdef AURORA_REPORT_FIFO | 
|  | { | 
|  | int i; | 
|  |  | 
|  | printk("aurora%d: port %d: FIFO hits [ ", | 
|  | board_No(bp), port_No(port)); | 
|  | for (i = 0; i < 10; i++)  { | 
|  | printk("%ld ", port->hits[i]); | 
|  | } | 
|  | printk("].\n"); | 
|  | } | 
|  | #endif | 
|  | if (port->xmit_buf)  { | 
|  | free_page((unsigned long) port->xmit_buf); | 
|  | port->xmit_buf = NULL; | 
|  | } | 
|  |  | 
|  | if (!(tty = port->tty) || C_HUPCL(tty))  { | 
|  | /* Drop DTR */ | 
|  | port->MSVR &= ~(bp->DTR|bp->RTS); | 
|  | sbus_writeb(port->MSVR, | 
|  | &bp->r[chip]->r[CD180_MSVR]); | 
|  | } | 
|  |  | 
|  | /* Select port */ | 
|  | sbus_writeb(port_No(port) & 7, | 
|  | &bp->r[chip]->r[CD180_CAR]); | 
|  | udelay(1); | 
|  |  | 
|  | /* Reset port */ | 
|  | aurora_wait_CCR(bp->r[chip]); | 
|  | sbus_writeb(CCR_SOFTRESET, &bp->r[chip]->r[CD180_CCR]); | 
|  |  | 
|  | /* Disable all interrupts from this port */ | 
|  | port->SRER = 0; | 
|  | sbus_writeb(port->SRER, &bp->r[chip]->r[CD180_SRER]); | 
|  |  | 
|  | if (tty) | 
|  | set_bit(TTY_IO_ERROR, &tty->flags); | 
|  | port->flags &= ~ASYNC_INITIALIZED; | 
|  |  | 
|  | #ifdef MODULE | 
|  | if (--bp->count < 0)  { | 
|  | printk(KERN_DEBUG "aurora%d: aurora_shutdown_port: " | 
|  | "bad board count: %d\n", | 
|  | board_No(bp), bp->count); | 
|  | bp->count = 0; | 
|  | } | 
|  |  | 
|  | if (!bp->count) | 
|  | bp->flags &= ~AURORA_BOARD_ACTIVE; | 
|  | #endif | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_shutdown_port: end\n"); | 
|  | #endif | 
|  | } | 
|  |  | 
|  |  | 
|  | static int block_til_ready(struct tty_struct *tty, struct file * filp, | 
|  | struct Aurora_port *port) | 
|  | { | 
|  | DECLARE_WAITQUEUE(wait, current); | 
|  | struct Aurora_board *bp = port_Board(port); | 
|  | int    retval; | 
|  | int    do_clocal = 0; | 
|  | int    CD; | 
|  | unsigned char chip; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("block_til_ready: start\n"); | 
|  | #endif | 
|  | chip = AURORA_CD180(port_No(port)); | 
|  |  | 
|  | /* If the device is in the middle of being closed, then block | 
|  | * until it's done, and then try again. | 
|  | */ | 
|  | if (tty_hung_up_p(filp) || port->flags & ASYNC_CLOSING) { | 
|  | interruptible_sleep_on(&port->close_wait); | 
|  | if (port->flags & ASYNC_HUP_NOTIFY) | 
|  | return -EAGAIN; | 
|  | else | 
|  | return -ERESTARTSYS; | 
|  | } | 
|  |  | 
|  | /* If non-blocking mode is set, or the port is not enabled, | 
|  | * then make the check up front and then exit. | 
|  | */ | 
|  | if ((filp->f_flags & O_NONBLOCK) || | 
|  | (tty->flags & (1 << TTY_IO_ERROR))) { | 
|  | port->flags |= ASYNC_NORMAL_ACTIVE; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | if (C_CLOCAL(tty)) | 
|  | do_clocal = 1; | 
|  |  | 
|  | /* Block waiting for the carrier detect and the line to become | 
|  | * free (i.e., not in use by the callout).  While we are in | 
|  | * this loop, info->count is dropped by one, so that | 
|  | * rs_close() knows when to free things.  We restore it upon | 
|  | * exit, either normal or abnormal. | 
|  | */ | 
|  | retval = 0; | 
|  | add_wait_queue(&port->open_wait, &wait); | 
|  | cli(); | 
|  | if (!tty_hung_up_p(filp)) | 
|  | port->count--; | 
|  | sti(); | 
|  | port->blocked_open++; | 
|  | while (1) { | 
|  | cli(); | 
|  | sbus_writeb(port_No(port) & 7, | 
|  | &bp->r[chip]->r[CD180_CAR]); | 
|  | udelay(1); | 
|  | CD = sbus_readb(&bp->r[chip]->r[CD180_MSVR]) & MSVR_CD; | 
|  | port->MSVR=bp->RTS; | 
|  |  | 
|  | /* auto drops DTR */ | 
|  | sbus_writeb(port->MSVR, &bp->r[chip]->r[CD180_MSVR]); | 
|  | sti(); | 
|  | set_current_state(TASK_INTERRUPTIBLE); | 
|  | if (tty_hung_up_p(filp) || | 
|  | !(port->flags & ASYNC_INITIALIZED)) { | 
|  | if (port->flags & ASYNC_HUP_NOTIFY) | 
|  | retval = -EAGAIN; | 
|  | else | 
|  | retval = -ERESTARTSYS; | 
|  | break; | 
|  | } | 
|  | if (!(port->flags & ASYNC_CLOSING) && | 
|  | (do_clocal || CD)) | 
|  | break; | 
|  | if (signal_pending(current)) { | 
|  | retval = -ERESTARTSYS; | 
|  | break; | 
|  | } | 
|  | schedule(); | 
|  | } | 
|  | current->state = TASK_RUNNING; | 
|  | remove_wait_queue(&port->open_wait, &wait); | 
|  | if (!tty_hung_up_p(filp)) | 
|  | port->count++; | 
|  | port->blocked_open--; | 
|  | if (retval) | 
|  | return retval; | 
|  |  | 
|  | port->flags |= ASYNC_NORMAL_ACTIVE; | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("block_til_ready: end\n"); | 
|  | #endif | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int aurora_open(struct tty_struct * tty, struct file * filp) | 
|  | { | 
|  | int board; | 
|  | int error; | 
|  | struct Aurora_port * port; | 
|  | struct Aurora_board * bp; | 
|  | unsigned long flags; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_open: start\n"); | 
|  | #endif | 
|  |  | 
|  | board = AURORA_BOARD(tty->index); | 
|  | if (board > AURORA_NBOARD || | 
|  | !(aurora_board[board].flags & AURORA_BOARD_PRESENT)) { | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_open: error board %d present %d\n", | 
|  | board, aurora_board[board].flags & AURORA_BOARD_PRESENT); | 
|  | #endif | 
|  | return -ENODEV; | 
|  | } | 
|  |  | 
|  | bp = &aurora_board[board]; | 
|  | port = aurora_port + board * AURORA_NPORT * AURORA_NCD180 + AURORA_PORT(tty->index); | 
|  | if ((aurora_paranoia_check(port, tty->name, "aurora_open")) { | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_open: error paranoia check\n"); | 
|  | #endif | 
|  | return -ENODEV; | 
|  | } | 
|  |  | 
|  | port->count++; | 
|  | tty->driver_data = port; | 
|  | port->tty = tty; | 
|  |  | 
|  | if ((error = aurora_setup_port(bp, port))) { | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_open: error aurora_setup_port ret %d\n",error); | 
|  | #endif | 
|  | return error; | 
|  | } | 
|  |  | 
|  | if ((error = block_til_ready(tty, filp, port))) { | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_open: error block_til_ready ret %d\n",error); | 
|  | #endif | 
|  | return error; | 
|  | } | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_open: end\n"); | 
|  | #endif | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void aurora_close(struct tty_struct * tty, struct file * filp) | 
|  | { | 
|  | struct Aurora_port *port = (struct Aurora_port *) tty->driver_data; | 
|  | struct Aurora_board *bp; | 
|  | unsigned long flags; | 
|  | unsigned long timeout; | 
|  | unsigned char chip; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_close: start\n"); | 
|  | #endif | 
|  |  | 
|  | if (!port || (aurora_paranoia_check(port, tty->name, "close")) | 
|  | return; | 
|  |  | 
|  | chip = AURORA_CD180(port_No(port)); | 
|  |  | 
|  | save_flags(flags); cli(); | 
|  | if (tty_hung_up_p(filp))  { | 
|  | restore_flags(flags); | 
|  | return; | 
|  | } | 
|  |  | 
|  | bp = port_Board(port); | 
|  | if ((tty->count == 1) && (port->count != 1))  { | 
|  | printk(KERN_DEBUG "aurora%d: aurora_close: bad port count; " | 
|  | "tty->count is 1, port count is %d\n", | 
|  | board_No(bp), port->count); | 
|  | port->count = 1; | 
|  | } | 
|  | if (--port->count < 0)  { | 
|  | printk(KERN_DEBUG "aurora%d: aurora_close: bad port " | 
|  | "count for tty%d: %d\n", | 
|  | board_No(bp), port_No(port), port->count); | 
|  | port->count = 0; | 
|  | } | 
|  | if (port->count)  { | 
|  | restore_flags(flags); | 
|  | return; | 
|  | } | 
|  | port->flags |= ASYNC_CLOSING; | 
|  |  | 
|  | /* Now we wait for the transmit buffer to clear; and we notify | 
|  | * the line discipline to only process XON/XOFF characters. | 
|  | */ | 
|  | tty->closing = 1; | 
|  | if (port->closing_wait != ASYNC_CLOSING_WAIT_NONE){ | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_close: waiting to flush...\n"); | 
|  | #endif | 
|  | tty_wait_until_sent(tty, port->closing_wait); | 
|  | } | 
|  |  | 
|  | /* At this point we stop accepting input.  To do this, we | 
|  | * disable the receive line status interrupts, and tell the | 
|  | * interrupt driver to stop checking the data ready bit in the | 
|  | * line status register. | 
|  | */ | 
|  | port->SRER &= ~SRER_RXD; | 
|  | if (port->flags & ASYNC_INITIALIZED) { | 
|  | port->SRER &= ~SRER_TXRDY; | 
|  | port->SRER |= SRER_TXEMPTY; | 
|  | sbus_writeb(port_No(port) & 7, | 
|  | &bp->r[chip]->r[CD180_CAR]); | 
|  | udelay(1); | 
|  | sbus_writeb(port->SRER, &bp->r[chip]->r[CD180_SRER]); | 
|  | /* | 
|  | * Before we drop DTR, make sure the UART transmitter | 
|  | * has completely drained; this is especially | 
|  | * important if there is a transmit FIFO! | 
|  | */ | 
|  | timeout = jiffies+HZ; | 
|  | while(port->SRER & SRER_TXEMPTY)  { | 
|  | msleep_interruptible(jiffies_to_msecs(port->timeout)); | 
|  | if (time_after(jiffies, timeout)) | 
|  | break; | 
|  | } | 
|  | } | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_close: shutdown_port\n"); | 
|  | #endif | 
|  | aurora_shutdown_port(bp, port); | 
|  | if (tty->driver->flush_buffer) | 
|  | tty->driver->flush_buffer(tty); | 
|  | tty_ldisc_flush(tty); | 
|  | tty->closing = 0; | 
|  | port->event = 0; | 
|  | port->tty = 0; | 
|  | if (port->blocked_open) { | 
|  | if (port->close_delay) { | 
|  | msleep_interruptible(jiffies_to_msecs(port->close_delay)); | 
|  | } | 
|  | wake_up_interruptible(&port->open_wait); | 
|  | } | 
|  | port->flags &= ~(ASYNC_NORMAL_ACTIVE|ASYNC_CLOSING); | 
|  | wake_up_interruptible(&port->close_wait); | 
|  | restore_flags(flags); | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_close: end\n"); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | static int aurora_write(struct tty_struct * tty, | 
|  | const unsigned char *buf, int count) | 
|  | { | 
|  | struct Aurora_port *port = (struct Aurora_port *) tty->driver_data; | 
|  | struct Aurora_board *bp; | 
|  | int c, total = 0; | 
|  | unsigned long flags; | 
|  | unsigned char chip; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_write: start %d\n",count); | 
|  | #endif | 
|  | if ((aurora_paranoia_check(port, tty->name, "aurora_write")) | 
|  | return 0; | 
|  |  | 
|  | chip = AURORA_CD180(port_No(port)); | 
|  |  | 
|  | bp = port_Board(port); | 
|  |  | 
|  | if (!tty || !port->xmit_buf || !tmp_buf) | 
|  | return 0; | 
|  |  | 
|  | save_flags(flags); | 
|  | while (1) { | 
|  | cli(); | 
|  | c = min(count, min(SERIAL_XMIT_SIZE - port->xmit_cnt - 1, | 
|  | SERIAL_XMIT_SIZE - port->xmit_head)); | 
|  | if (c <= 0) { | 
|  | restore_flags(flags); | 
|  | break; | 
|  | } | 
|  | memcpy(port->xmit_buf + port->xmit_head, buf, c); | 
|  | port->xmit_head = (port->xmit_head + c) & (SERIAL_XMIT_SIZE-1); | 
|  | port->xmit_cnt += c; | 
|  | restore_flags(flags); | 
|  |  | 
|  | buf += c; | 
|  | count -= c; | 
|  | total += c; | 
|  | } | 
|  |  | 
|  | cli(); | 
|  | if (port->xmit_cnt && !tty->stopped && !tty->hw_stopped && | 
|  | !(port->SRER & SRER_TXRDY)) { | 
|  | port->SRER |= SRER_TXRDY; | 
|  | sbus_writeb(port_No(port) & 7, | 
|  | &bp->r[chip]->r[CD180_CAR]); | 
|  | udelay(1); | 
|  | sbus_writeb(port->SRER, &bp->r[chip]->r[CD180_SRER]); | 
|  | } | 
|  | restore_flags(flags); | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_write: end %d\n",total); | 
|  | #endif | 
|  | return total; | 
|  | } | 
|  |  | 
|  | static void aurora_put_char(struct tty_struct * tty, unsigned char ch) | 
|  | { | 
|  | struct Aurora_port *port = (struct Aurora_port *) tty->driver_data; | 
|  | unsigned long flags; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_put_char: start %c\n",ch); | 
|  | #endif | 
|  | if ((aurora_paranoia_check(port, tty->name, "aurora_put_char")) | 
|  | return; | 
|  |  | 
|  | if (!tty || !port->xmit_buf) | 
|  | return; | 
|  |  | 
|  | save_flags(flags); cli(); | 
|  |  | 
|  | if (port->xmit_cnt >= SERIAL_XMIT_SIZE - 1) { | 
|  | restore_flags(flags); | 
|  | return; | 
|  | } | 
|  |  | 
|  | port->xmit_buf[port->xmit_head++] = ch; | 
|  | port->xmit_head &= SERIAL_XMIT_SIZE - 1; | 
|  | port->xmit_cnt++; | 
|  | restore_flags(flags); | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_put_char: end\n"); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | static void aurora_flush_chars(struct tty_struct * tty) | 
|  | { | 
|  | struct Aurora_port *port = (struct Aurora_port *) tty->driver_data; | 
|  | unsigned long flags; | 
|  | unsigned char chip; | 
|  |  | 
|  | /*#ifdef AURORA_DEBUG | 
|  | printk("aurora_flush_chars: start\n"); | 
|  | #endif*/ | 
|  | if ((aurora_paranoia_check(port, tty->name, "aurora_flush_chars")) | 
|  | return; | 
|  |  | 
|  | chip = AURORA_CD180(port_No(port)); | 
|  |  | 
|  | if (port->xmit_cnt <= 0 || tty->stopped || tty->hw_stopped || | 
|  | !port->xmit_buf) | 
|  | return; | 
|  |  | 
|  | save_flags(flags); cli(); | 
|  | port->SRER |= SRER_TXRDY; | 
|  | sbus_writeb(port_No(port) & 7, | 
|  | &port_Board(port)->r[chip]->r[CD180_CAR]); | 
|  | udelay(1); | 
|  | sbus_writeb(port->SRER, | 
|  | &port_Board(port)->r[chip]->r[CD180_SRER]); | 
|  | restore_flags(flags); | 
|  | /*#ifdef AURORA_DEBUG | 
|  | printk("aurora_flush_chars: end\n"); | 
|  | #endif*/ | 
|  | } | 
|  |  | 
|  | static int aurora_write_room(struct tty_struct * tty) | 
|  | { | 
|  | struct Aurora_port *port = (struct Aurora_port *) tty->driver_data; | 
|  | int	ret; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_write_room: start\n"); | 
|  | #endif | 
|  | if ((aurora_paranoia_check(port, tty->name, "aurora_write_room")) | 
|  | return 0; | 
|  |  | 
|  | ret = SERIAL_XMIT_SIZE - port->xmit_cnt - 1; | 
|  | if (ret < 0) | 
|  | ret = 0; | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_write_room: end\n"); | 
|  | #endif | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static int aurora_chars_in_buffer(struct tty_struct *tty) | 
|  | { | 
|  | struct Aurora_port *port = (struct Aurora_port *) tty->driver_data; | 
|  |  | 
|  | if ((aurora_paranoia_check(port, tty->name, "aurora_chars_in_buffer")) | 
|  | return 0; | 
|  |  | 
|  | return port->xmit_cnt; | 
|  | } | 
|  |  | 
|  | static void aurora_flush_buffer(struct tty_struct *tty) | 
|  | { | 
|  | struct Aurora_port *port = (struct Aurora_port *) tty->driver_data; | 
|  | unsigned long flags; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_flush_buffer: start\n"); | 
|  | #endif | 
|  | if ((aurora_paranoia_check(port, tty->name, "aurora_flush_buffer")) | 
|  | return; | 
|  |  | 
|  | save_flags(flags); cli(); | 
|  | port->xmit_cnt = port->xmit_head = port->xmit_tail = 0; | 
|  | restore_flags(flags); | 
|  |  | 
|  | tty_wakeup(tty); | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_flush_buffer: end\n"); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | static int aurora_tiocmget(struct tty_struct *tty, struct file *file) | 
|  | { | 
|  | struct Aurora_port *port = (struct Aurora_port *) tty->driver_data; | 
|  | struct Aurora_board * bp; | 
|  | unsigned char status,chip; | 
|  | unsigned int result; | 
|  | unsigned long flags; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_get_modem_info: start\n"); | 
|  | #endif | 
|  | if ((aurora_paranoia_check(port, tty->name, __FUNCTION__)) | 
|  | return -ENODEV; | 
|  |  | 
|  | chip = AURORA_CD180(port_No(port)); | 
|  |  | 
|  | bp = port_Board(port); | 
|  |  | 
|  | save_flags(flags); cli(); | 
|  |  | 
|  | sbus_writeb(port_No(port) & 7, &bp->r[chip]->r[CD180_CAR]); | 
|  | udelay(1); | 
|  |  | 
|  | status = sbus_readb(&bp->r[chip]->r[CD180_MSVR]); | 
|  | result = 0/*bp->r[chip]->r[AURORA_RI] & (1u << port_No(port)) ? 0 : TIOCM_RNG*/; | 
|  |  | 
|  | restore_flags(flags); | 
|  |  | 
|  | result |= ((status & bp->RTS) ? TIOCM_RTS : 0) | 
|  | | ((status & bp->DTR) ? TIOCM_DTR : 0) | 
|  | | ((status & MSVR_CD)  ? TIOCM_CAR : 0) | 
|  | | ((status & MSVR_DSR) ? TIOCM_DSR : 0) | 
|  | | ((status & MSVR_CTS) ? TIOCM_CTS : 0); | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_get_modem_info: end\n"); | 
|  | #endif | 
|  | return result; | 
|  | } | 
|  |  | 
|  | static int aurora_tiocmset(struct tty_struct *tty, struct file *file, | 
|  | unsigned int set, unsigned int clear) | 
|  | { | 
|  | struct Aurora_port *port = (struct Aurora_port *) tty->driver_data; | 
|  | unsigned int arg; | 
|  | unsigned long flags; | 
|  | struct Aurora_board *bp = port_Board(port); | 
|  | unsigned char chip; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_set_modem_info: start\n"); | 
|  | #endif | 
|  | if ((aurora_paranoia_check(port, tty->name, __FUNCTION__)) | 
|  | return -ENODEV; | 
|  |  | 
|  | chip = AURORA_CD180(port_No(port)); | 
|  |  | 
|  | save_flags(flags); cli(); | 
|  | if (set & TIOCM_RTS) | 
|  | port->MSVR |= bp->RTS; | 
|  | if (set & TIOCM_DTR) | 
|  | port->MSVR |= bp->DTR; | 
|  | if (clear & TIOCM_RTS) | 
|  | port->MSVR &= ~bp->RTS; | 
|  | if (clear & TIOCM_DTR) | 
|  | port->MSVR &= ~bp->DTR; | 
|  |  | 
|  | sbus_writeb(port_No(port) & 7, &bp->r[chip]->r[CD180_CAR]); | 
|  | udelay(1); | 
|  |  | 
|  | sbus_writeb(port->MSVR, &bp->r[chip]->r[CD180_MSVR]); | 
|  |  | 
|  | restore_flags(flags); | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_set_modem_info: end\n"); | 
|  | #endif | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void aurora_send_break(struct Aurora_port * port, unsigned long length) | 
|  | { | 
|  | struct Aurora_board *bp = port_Board(port); | 
|  | unsigned long flags; | 
|  | unsigned char chip; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_send_break: start\n"); | 
|  | #endif | 
|  | chip = AURORA_CD180(port_No(port)); | 
|  |  | 
|  | save_flags(flags); cli(); | 
|  |  | 
|  | port->break_length = AURORA_TPS / HZ * length; | 
|  | port->COR2 |= COR2_ETC; | 
|  | port->SRER  |= SRER_TXRDY; | 
|  | sbus_writeb(port_No(port) & 7, &bp->r[chip]->r[CD180_CAR]); | 
|  | udelay(1); | 
|  |  | 
|  | sbus_writeb(port->COR2, &bp->r[chip]->r[CD180_COR2]); | 
|  | sbus_writeb(port->SRER, &bp->r[chip]->r[CD180_SRER]); | 
|  | aurora_wait_CCR(bp->r[chip]); | 
|  |  | 
|  | sbus_writeb(CCR_CORCHG2, &bp->r[chip]->r[CD180_CCR]); | 
|  | aurora_wait_CCR(bp->r[chip]); | 
|  |  | 
|  | restore_flags(flags); | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_send_break: end\n"); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | static int aurora_set_serial_info(struct Aurora_port * port, | 
|  | struct serial_struct * newinfo) | 
|  | { | 
|  | struct serial_struct tmp; | 
|  | struct Aurora_board *bp = port_Board(port); | 
|  | int change_speed; | 
|  | unsigned long flags; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_set_serial_info: start\n"); | 
|  | #endif | 
|  | if (copy_from_user(&tmp, newinfo, sizeof(tmp))) | 
|  | return -EFAULT; | 
|  | #if 0 | 
|  | if ((tmp.irq != bp->irq) || | 
|  | (tmp.port != bp->base) || | 
|  | (tmp.type != PORT_CIRRUS) || | 
|  | (tmp.baud_base != (bp->oscfreq + CD180_TPC/2) / CD180_TPC) || | 
|  | (tmp.custom_divisor != 0) || | 
|  | (tmp.xmit_fifo_size != CD180_NFIFO) || | 
|  | (tmp.flags & ~AURORA_LEGAL_FLAGS)) | 
|  | return -EINVAL; | 
|  | #endif | 
|  |  | 
|  | change_speed = ((port->flags & ASYNC_SPD_MASK) != | 
|  | (tmp.flags & ASYNC_SPD_MASK)); | 
|  |  | 
|  | if (!capable(CAP_SYS_ADMIN)) { | 
|  | if ((tmp.close_delay != port->close_delay) || | 
|  | (tmp.closing_wait != port->closing_wait) || | 
|  | ((tmp.flags & ~ASYNC_USR_MASK) != | 
|  | (port->flags & ~ASYNC_USR_MASK))) | 
|  | return -EPERM; | 
|  | port->flags = ((port->flags & ~ASYNC_USR_MASK) | | 
|  | (tmp.flags & ASYNC_USR_MASK)); | 
|  | } else  { | 
|  | port->flags = ((port->flags & ~ASYNC_FLAGS) | | 
|  | (tmp.flags & ASYNC_FLAGS)); | 
|  | port->close_delay = tmp.close_delay; | 
|  | port->closing_wait = tmp.closing_wait; | 
|  | } | 
|  | if (change_speed)  { | 
|  | save_flags(flags); cli(); | 
|  | aurora_change_speed(bp, port); | 
|  | restore_flags(flags); | 
|  | } | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_set_serial_info: end\n"); | 
|  | #endif | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | extern int aurora_get_serial_info(struct Aurora_port * port, | 
|  | struct serial_struct * retinfo) | 
|  | { | 
|  | struct serial_struct tmp; | 
|  | struct Aurora_board *bp = port_Board(port); | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_get_serial_info: start\n"); | 
|  | #endif | 
|  | if (!access_ok(VERIFY_WRITE, (void *) retinfo, sizeof(tmp))) | 
|  | return -EFAULT; | 
|  |  | 
|  | memset(&tmp, 0, sizeof(tmp)); | 
|  | tmp.type = PORT_CIRRUS; | 
|  | tmp.line = port - aurora_port; | 
|  | tmp.port = 0; | 
|  | tmp.irq  = bp->irq; | 
|  | tmp.flags = port->flags; | 
|  | tmp.baud_base = (bp->oscfreq + CD180_TPC/2) / CD180_TPC; | 
|  | tmp.close_delay = port->close_delay * HZ/100; | 
|  | tmp.closing_wait = port->closing_wait * HZ/100; | 
|  | tmp.xmit_fifo_size = CD180_NFIFO; | 
|  | copy_to_user(retinfo, &tmp, sizeof(tmp)); | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_get_serial_info: end\n"); | 
|  | #endif | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int aurora_ioctl(struct tty_struct * tty, struct file * filp, | 
|  | unsigned int cmd, unsigned long arg) | 
|  |  | 
|  | { | 
|  | struct Aurora_port *port = (struct Aurora_port *) tty->driver_data; | 
|  | int retval; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_ioctl: start\n"); | 
|  | #endif | 
|  | if ((aurora_paranoia_check(port, tty->name, "aurora_ioctl")) | 
|  | return -ENODEV; | 
|  |  | 
|  | switch (cmd) { | 
|  | case TCSBRK:	/* SVID version: non-zero arg --> no break */ | 
|  | retval = tty_check_change(tty); | 
|  | if (retval) | 
|  | return retval; | 
|  | tty_wait_until_sent(tty, 0); | 
|  | if (!arg) | 
|  | aurora_send_break(port, HZ/4);	/* 1/4 second */ | 
|  | return 0; | 
|  | case TCSBRKP:	/* support for POSIX tcsendbreak() */ | 
|  | retval = tty_check_change(tty); | 
|  | if (retval) | 
|  | return retval; | 
|  | tty_wait_until_sent(tty, 0); | 
|  | aurora_send_break(port, arg ? arg*(HZ/10) : HZ/4); | 
|  | return 0; | 
|  | case TIOCGSOFTCAR: | 
|  | return put_user(C_CLOCAL(tty) ? 1 : 0, (unsigned long *)arg); | 
|  | case TIOCSSOFTCAR: | 
|  | if (get_user(arg,(unsigned long *)arg)) | 
|  | return -EFAULT; | 
|  | tty->termios->c_cflag = | 
|  | ((tty->termios->c_cflag & ~CLOCAL) | | 
|  | (arg ? CLOCAL : 0)); | 
|  | return 0; | 
|  | case TIOCGSERIAL: | 
|  | return aurora_get_serial_info(port, (struct serial_struct *) arg); | 
|  | case TIOCSSERIAL: | 
|  | return aurora_set_serial_info(port, (struct serial_struct *) arg); | 
|  | default: | 
|  | return -ENOIOCTLCMD; | 
|  | }; | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_ioctl: end\n"); | 
|  | #endif | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void aurora_throttle(struct tty_struct * tty) | 
|  | { | 
|  | struct Aurora_port *port = (struct Aurora_port *) tty->driver_data; | 
|  | struct Aurora_board *bp; | 
|  | unsigned long flags; | 
|  | unsigned char chip; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_throttle: start\n"); | 
|  | #endif | 
|  | if ((aurora_paranoia_check(port, tty->name, "aurora_throttle")) | 
|  | return; | 
|  |  | 
|  | bp = port_Board(port); | 
|  | chip = AURORA_CD180(port_No(port)); | 
|  |  | 
|  | save_flags(flags); cli(); | 
|  | port->MSVR &= ~bp->RTS; | 
|  | sbus_writeb(port_No(port) & 7, &bp->r[chip]->r[CD180_CAR]); | 
|  | udelay(1); | 
|  | if (I_IXOFF(tty))  { | 
|  | aurora_wait_CCR(bp->r[chip]); | 
|  | sbus_writeb(CCR_SSCH2, &bp->r[chip]->r[CD180_CCR]); | 
|  | aurora_wait_CCR(bp->r[chip]); | 
|  | } | 
|  | sbus_writeb(port->MSVR, &bp->r[chip]->r[CD180_MSVR]); | 
|  | restore_flags(flags); | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_throttle: end\n"); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | static void aurora_unthrottle(struct tty_struct * tty) | 
|  | { | 
|  | struct Aurora_port *port = (struct Aurora_port *) tty->driver_data; | 
|  | struct Aurora_board *bp; | 
|  | unsigned long flags; | 
|  | unsigned char chip; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_unthrottle: start\n"); | 
|  | #endif | 
|  | if ((aurora_paranoia_check(port, tty->name, "aurora_unthrottle")) | 
|  | return; | 
|  |  | 
|  | bp = port_Board(port); | 
|  |  | 
|  | chip = AURORA_CD180(port_No(port)); | 
|  |  | 
|  | save_flags(flags); cli(); | 
|  | port->MSVR |= bp->RTS; | 
|  | sbus_writeb(port_No(port) & 7, | 
|  | &bp->r[chip]->r[CD180_CAR]); | 
|  | udelay(1); | 
|  | if (I_IXOFF(tty))  { | 
|  | aurora_wait_CCR(bp->r[chip]); | 
|  | sbus_writeb(CCR_SSCH1, | 
|  | &bp->r[chip]->r[CD180_CCR]); | 
|  | aurora_wait_CCR(bp->r[chip]); | 
|  | } | 
|  | sbus_writeb(port->MSVR, &bp->r[chip]->r[CD180_MSVR]); | 
|  | restore_flags(flags); | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_unthrottle: end\n"); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | static void aurora_stop(struct tty_struct * tty) | 
|  | { | 
|  | struct Aurora_port *port = (struct Aurora_port *) tty->driver_data; | 
|  | struct Aurora_board *bp; | 
|  | unsigned long flags; | 
|  | unsigned char chip; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_stop: start\n"); | 
|  | #endif | 
|  | if ((aurora_paranoia_check(port, tty->name, "aurora_stop")) | 
|  | return; | 
|  |  | 
|  | bp = port_Board(port); | 
|  |  | 
|  | chip = AURORA_CD180(port_No(port)); | 
|  |  | 
|  | save_flags(flags); cli(); | 
|  | port->SRER &= ~SRER_TXRDY; | 
|  | sbus_writeb(port_No(port) & 7, | 
|  | &bp->r[chip]->r[CD180_CAR]); | 
|  | udelay(1); | 
|  | sbus_writeb(port->SRER, | 
|  | &bp->r[chip]->r[CD180_SRER]); | 
|  | restore_flags(flags); | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_stop: end\n"); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | static void aurora_start(struct tty_struct * tty) | 
|  | { | 
|  | struct Aurora_port *port = (struct Aurora_port *) tty->driver_data; | 
|  | struct Aurora_board *bp; | 
|  | unsigned long flags; | 
|  | unsigned char chip; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_start: start\n"); | 
|  | #endif | 
|  | if ((aurora_paranoia_check(port, tty->name, "aurora_start")) | 
|  | return; | 
|  |  | 
|  | bp = port_Board(port); | 
|  |  | 
|  | chip = AURORA_CD180(port_No(port)); | 
|  |  | 
|  | save_flags(flags); cli(); | 
|  | if (port->xmit_cnt && port->xmit_buf && !(port->SRER & SRER_TXRDY))  { | 
|  | port->SRER |= SRER_TXRDY; | 
|  | sbus_writeb(port_No(port) & 7, | 
|  | &bp->r[chip]->r[CD180_CAR]); | 
|  | udelay(1); | 
|  | sbus_writeb(port->SRER, | 
|  | &bp->r[chip]->r[CD180_SRER]); | 
|  | } | 
|  | restore_flags(flags); | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_start: end\n"); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | /* | 
|  | * This routine is called from the scheduler tqueue when the interrupt | 
|  | * routine has signalled that a hangup has occurred.  The path of | 
|  | * hangup processing is: | 
|  | * | 
|  | * 	serial interrupt routine -> (scheduler tqueue) -> | 
|  | * 	do_aurora_hangup() -> tty->hangup() -> aurora_hangup() | 
|  | * | 
|  | */ | 
|  | static void do_aurora_hangup(void *private_) | 
|  | { | 
|  | struct Aurora_port	*port = (struct Aurora_port *) private_; | 
|  | struct tty_struct	*tty; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("do_aurora_hangup: start\n"); | 
|  | #endif | 
|  | tty = port->tty; | 
|  | if (tty != NULL) { | 
|  | tty_hangup(tty);	/* FIXME: module removal race - AKPM */ | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("do_aurora_hangup: end\n"); | 
|  | #endif | 
|  | } | 
|  | } | 
|  |  | 
|  | static void aurora_hangup(struct tty_struct * tty) | 
|  | { | 
|  | struct Aurora_port *port = (struct Aurora_port *) tty->driver_data; | 
|  | struct Aurora_board *bp; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_hangup: start\n"); | 
|  | #endif | 
|  | if ((aurora_paranoia_check(port, tty->name, "aurora_hangup")) | 
|  | return; | 
|  |  | 
|  | bp = port_Board(port); | 
|  |  | 
|  | aurora_shutdown_port(bp, port); | 
|  | port->event = 0; | 
|  | port->count = 0; | 
|  | port->flags &= ~ASYNC_NORMAL_ACTIVE; | 
|  | port->tty = 0; | 
|  | wake_up_interruptible(&port->open_wait); | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_hangup: end\n"); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | static void aurora_set_termios(struct tty_struct * tty, struct termios * old_termios) | 
|  | { | 
|  | struct Aurora_port *port = (struct Aurora_port *) tty->driver_data; | 
|  | unsigned long flags; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_set_termios: start\n"); | 
|  | #endif | 
|  | if ((aurora_paranoia_check(port, tty->name, "aurora_set_termios")) | 
|  | return; | 
|  |  | 
|  | if (tty->termios->c_cflag == old_termios->c_cflag && | 
|  | tty->termios->c_iflag == old_termios->c_iflag) | 
|  | return; | 
|  |  | 
|  | save_flags(flags); cli(); | 
|  | aurora_change_speed(port_Board(port), port); | 
|  | restore_flags(flags); | 
|  |  | 
|  | if ((old_termios->c_cflag & CRTSCTS) && | 
|  | !(tty->termios->c_cflag & CRTSCTS)) { | 
|  | tty->hw_stopped = 0; | 
|  | aurora_start(tty); | 
|  | } | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_set_termios: end\n"); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | static void do_aurora_bh(void) | 
|  | { | 
|  | run_task_queue(&tq_aurora); | 
|  | } | 
|  |  | 
|  | static void do_softint(void *private_) | 
|  | { | 
|  | struct Aurora_port	*port = (struct Aurora_port *) private_; | 
|  | struct tty_struct	*tty; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("do_softint: start\n"); | 
|  | #endif | 
|  | tty = port->tty; | 
|  | if (tty == NULL) | 
|  | return; | 
|  |  | 
|  | if (test_and_clear_bit(RS_EVENT_WRITE_WAKEUP, &port->event)) { | 
|  | tty_wakeup(tty); | 
|  | } | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("do_softint: end\n"); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | static struct tty_operations aurora_ops = { | 
|  | .open  = aurora_open, | 
|  | .close = aurora_close, | 
|  | .write = aurora_write, | 
|  | .put_char = aurora_put_char, | 
|  | .flush_chars = aurora_flush_chars, | 
|  | .write_room = aurora_write_room, | 
|  | .chars_in_buffer = aurora_chars_in_buffer, | 
|  | .flush_buffer = aurora_flush_buffer, | 
|  | .ioctl = aurora_ioctl, | 
|  | .throttle = aurora_throttle, | 
|  | .unthrottle = aurora_unthrottle, | 
|  | .set_termios = aurora_set_termios, | 
|  | .stop = aurora_stop, | 
|  | .start = aurora_start, | 
|  | .hangup = aurora_hangup, | 
|  | .tiocmget = aurora_tiocmget, | 
|  | .tiocmset = aurora_tiocmset, | 
|  | }; | 
|  |  | 
|  | static int aurora_init_drivers(void) | 
|  | { | 
|  | int error; | 
|  | int i; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_init_drivers: start\n"); | 
|  | #endif | 
|  | tmp_buf = (unsigned char *) get_zeroed_page(GFP_KERNEL); | 
|  | if (tmp_buf == NULL) { | 
|  | printk(KERN_ERR "aurora: Couldn't get free page.\n"); | 
|  | return 1; | 
|  | } | 
|  | init_bh(AURORA_BH, do_aurora_bh); | 
|  | aurora_driver = alloc_tty_driver(AURORA_INPORTS); | 
|  | if (!aurora_driver) { | 
|  | printk(KERN_ERR "aurora: Couldn't allocate tty driver.\n"); | 
|  | free_page((unsigned long) tmp_buf); | 
|  | return 1; | 
|  | } | 
|  | aurora_driver->owner = THIS_MODULE; | 
|  | aurora_driver->name = "ttyA"; | 
|  | aurora_driver->major = AURORA_MAJOR; | 
|  | aurora_driver->type = TTY_DRIVER_TYPE_SERIAL; | 
|  | aurora_driver->subtype = SERIAL_TYPE_NORMAL; | 
|  | aurora_driver->init_termios = tty_std_termios; | 
|  | aurora_driver->init_termios.c_cflag = | 
|  | B9600 | CS8 | CREAD | HUPCL | CLOCAL; | 
|  | aurora_driver->flags = TTY_DRIVER_REAL_RAW; | 
|  | tty_set_operations(aurora_driver, &aurora_ops); | 
|  | error = tty_register_driver(aurora_driver); | 
|  | if (error) { | 
|  | put_tty_driver(aurora_driver); | 
|  | free_page((unsigned long) tmp_buf); | 
|  | printk(KERN_ERR "aurora: Couldn't register aurora driver, error = %d\n", | 
|  | error); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | memset(aurora_port, 0, sizeof(aurora_port)); | 
|  | for (i = 0; i < AURORA_TNPORTS; i++)  { | 
|  | aurora_port[i].magic = AURORA_MAGIC; | 
|  | aurora_port[i].tqueue.routine = do_softint; | 
|  | aurora_port[i].tqueue.data = &aurora_port[i]; | 
|  | aurora_port[i].tqueue_hangup.routine = do_aurora_hangup; | 
|  | aurora_port[i].tqueue_hangup.data = &aurora_port[i]; | 
|  | aurora_port[i].close_delay = 50 * HZ/100; | 
|  | aurora_port[i].closing_wait = 3000 * HZ/100; | 
|  | init_waitqueue_head(&aurora_port[i].open_wait); | 
|  | init_waitqueue_head(&aurora_port[i].close_wait); | 
|  | } | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_init_drivers: end\n"); | 
|  | #endif | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void aurora_release_drivers(void) | 
|  | { | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_release_drivers: start\n"); | 
|  | #endif | 
|  | free_page((unsigned long)tmp_buf); | 
|  | tty_unregister_driver(aurora_driver); | 
|  | put_tty_driver(aurora_driver); | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("aurora_release_drivers: end\n"); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Called at boot time. | 
|  | * | 
|  | * You can specify IO base for up to RC_NBOARD cards, | 
|  | * using line "riscom8=0xiobase1,0xiobase2,.." at LILO prompt. | 
|  | * Note that there will be no probing at default | 
|  | * addresses in this case. | 
|  | * | 
|  | */ | 
|  | void __init aurora_setup(char *str, int *ints) | 
|  | { | 
|  | int i; | 
|  |  | 
|  | for(i=0;(i<ints[0])&&(i<4);i++) { | 
|  | if (ints[i+1]) irqs[i]=ints[i+1]; | 
|  | } | 
|  | } | 
|  |  | 
|  | static int __init aurora_real_init(void) | 
|  | { | 
|  | int found; | 
|  | int i; | 
|  |  | 
|  | printk(KERN_INFO "aurora: Driver starting.\n"); | 
|  | if(aurora_init_drivers()) | 
|  | return -EIO; | 
|  | found = aurora_probe(); | 
|  | if(!found) { | 
|  | aurora_release_drivers(); | 
|  | printk(KERN_INFO "aurora: No Aurora Multiport boards detected.\n"); | 
|  | return -EIO; | 
|  | } else { | 
|  | printk(KERN_INFO "aurora: %d boards found.\n", found); | 
|  | } | 
|  | for (i = 0; i < found; i++) { | 
|  | int ret = aurora_setup_board(&aurora_board[i]); | 
|  |  | 
|  | if (ret) { | 
|  | #ifdef AURORA_DEBUG | 
|  | printk(KERN_ERR "aurora_init: error aurora_setup_board ret %d\n", | 
|  | ret); | 
|  | #endif | 
|  | return ret; | 
|  | } | 
|  | } | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | int irq  = 0; | 
|  | int irq1 = 0; | 
|  | int irq2 = 0; | 
|  | int irq3 = 0; | 
|  | module_param(irq , int, 0); | 
|  | module_param(irq1, int, 0); | 
|  | module_param(irq2, int, 0); | 
|  | module_param(irq3, int, 0); | 
|  |  | 
|  | static int __init aurora_init(void) | 
|  | { | 
|  | if (irq ) irqs[0]=irq ; | 
|  | if (irq1) irqs[1]=irq1; | 
|  | if (irq2) irqs[2]=irq2; | 
|  | if (irq3) irqs[3]=irq3; | 
|  | return aurora_real_init(); | 
|  | } | 
|  |  | 
|  | static void __exit aurora_cleanup(void) | 
|  | { | 
|  | int i; | 
|  |  | 
|  | #ifdef AURORA_DEBUG | 
|  | printk("cleanup_module: aurora_release_drivers\n"); | 
|  | #endif | 
|  |  | 
|  | aurora_release_drivers(); | 
|  | for (i = 0; i < AURORA_NBOARD; i++) | 
|  | if (aurora_board[i].flags & AURORA_BOARD_PRESENT) { | 
|  | aurora_shutdown_board(&aurora_board[i]); | 
|  | aurora_release_io_range(&aurora_board[i]); | 
|  | } | 
|  | } | 
|  |  | 
|  | module_init(aurora_init); | 
|  | module_exit(aurora_cleanup); | 
|  | MODULE_LICENSE("GPL"); |