www.pudn.com > 44b0-232-ucLINUX-driver.rar > 44b0232-uclinux-driver.c, change:2004-12-13,size:98096b


/* 
 *  linux/drivers/char/serial_44b0.c 
 * 
 *  Copyright (C) 1991, 1992  Linus Torvalds 
 *  Copyright (C) 1992, 1993, 1994, 1995, 1996, 1997,  
 * 		1998, 1999  Theodore Ts'o 
 * 
 *  Extensively rewritten by Theodore Ts'o, 8/16/92 -- 9/14/92.  Now 
 *  much more extensible to support other serial cards based on the 
 *  16450/16550A UART's.  Added support for the AST FourPort and the 
 *  Accent Async board.   
 * 
 *  set_serial_info fixed to set the flags, custom divisor, and uart 
 * 	type fields.  Fix suggested by Michael K. Johnson 12/12/92. 
 * 
 *  11/95: TIOCMIWAIT, TIOCGICOUNT by Angelo Haritsis  
 * 
 *  03/96: Modularised by Angelo Haritsis  
 * 
 *  rs_set_termios fixed to look also for changes of the input 
 *      flags INPCK, BRKINT, PARMRK, IGNPAR and IGNBRK. 
 *                                            Bernd Anh鋟pl 05/17/96. 
 * 
 *  1/97:  Extended dumb serial ports are a config option now.   
 *         Saves 4k.   Michael A. Griffith  
 *  
 *  8/97: Fix bug in rs_set_termios with RTS 
 *        Stanislav V. Voronyi  
 * 
 *  3/98: Change the IRQ detection, use of probe_irq_o*(), 
 *	  suppress TIOCSERGWILD and TIOCSERSWILD 
 *	  Etienne Lorrain  
 * 
 *  4/98: Added changes to support the ARM architecture proposed by 
 * 	  Russell King 
 * 
 *  5/99: Updated to include support for the XR16C850 and ST16C654 
 *        uarts.  Stuart MacDonald  
 * 
 *  8/99: Generalized PCI support added.  Theodore Ts'o 
 *  
 *  3/00: Rid circular buffer of redundant xmit_cnt.  Fix a 
 *	  few races on freeing buffers too. 
 *	  Alan Modra  
 * 
 *  5/00: Support for the RSA-DV II/S card added. 
 *	  Kiyokazu SUTO  
 *  
 *  6/00: Remove old-style timer, use timer_list 
 *        Andrew Morton  
 * 
 *  7/00: Support Timedia/Sunix/Exsys PCI cards 
 * 
 *  7/00: fix some returns on failure not using MOD_DEC_USE_COUNT. 
 *	  Arnaldo Carvalho de Melo  
 * 
 * 10/00: add in optional software flow control for serial console. 
 *	  Kanoj Sarcar   (Modified by Theodore Ts'o) 
 *  
 * 09/02: modify by wesley king for s3c44b0's serial console. 
 *	  Qin Wei  
 * 
 * 01/03: modify by wesley king for s3c44b0's serial console, support ev44b0ii. 
 *	  Qin Wei  
 * 
 */ 
 
static char *serial_version = "1.10"; 
static char *serial_revdate = "2003-01-15"; 
 
/* 
 * Serial driver configuration section.  Here are the various options: 
 * 
 * CONFIG_HUB6 
 *		Enables support for the venerable Bell Technologies 
 *		HUB6 card. 
 * 
 * CONFIG_SERIAL_MANY_PORTS 
 * 		Enables support for ports beyond the standard, stupid 
 * 		COM 1/2/3/4. 
 * 
 * CONFIG_SERIAL_MULTIPORT 
 * 		Enables support for special multiport board support. 
 * 
 * CONFIG_SERIAL_SHARE_IRQ 
 * 		Enables support for multiple serial ports on one IRQ 
 * 
 * CONFIG_SERIAL_DETECT_IRQ 
 *		Enable the autodetection of IRQ on standart ports 
 * 
 * SERIAL_PARANOIA_CHECK 
 * 		Check the magic number for the async_structure where 
 * 		ever possible. 
 */ 
 
#include  
#include  
 
#undef SERIAL_PARANOIA_CHECK 
#define CONFIG_SERIAL_NOPAUSE_IO 
#define SERIAL_DO_RESTART 
 
/* Set of debugging defines */ 
 
#undef SERIAL_DEBUG_INTR 
#undef SERIAL_DEBUG_OPEN 
#undef SERIAL_DEBUG_FLOW 
#undef SERIAL_DEBUG_RS_WAIT_UNTIL_SENT 
#undef SERIAL_DEBUG_PCI 
#undef SERIAL_DEBUG_AUTOCONF 
 
/* Sanity checks */ 
// Qinwei : for using polling mode to send char on serial port 
//#define UARTPOLLING 
#undef UARTPOLLING 
 
#ifdef UARTPOLLING 
#define RS_STROBE_TIME 10*HZ 
#else 
#define RS_STROBE_TIME 10 
#endif 
#define RS_ISR_PASS_LIMIT 256 
 
#if defined(__i386__) && (defined(CONFIG_M386) || defined(CONFIG_M486)) 
#define SERIAL_INLINE 
#endif 
   
/* 
 * End of serial driver configuration section. 
 */ 
 
#include  
 
#include  
 
#include  
#include  
 
#ifdef LOCAL_HEADERS 
#include "serial_local.h" 
#else 
#include  
#include  
//#include  
#include  
#define LOCAL_VERSTRING "" 
#endif 
 
#include  
#include  
#include  
#include  
#include  
#include  
#include  
#include  
#include  
#include  
#include  
#include  
#include  
#include  
#if (LINUX_VERSION_CODE >= 131343) 
#include  
#endif 
#if (LINUX_VERSION_CODE >= 131336) 
#include  
#endif 
#include  
#ifdef CONFIG_SERIAL_SAMSUNG_CONSOLE 
#include  
#endif 
#ifdef CONFIG_MAGIC_SYSRQ 
#include  
#endif 
#ifdef CONFIG_LEDMAN 
#include  
#endif 
 
/* 
 * All of the compatibilty code so we can compile serial.c against 
 * older kernels is hidden in serial_compat.h 
 */ 
#if defined(LOCAL_HEADERS) || (LINUX_VERSION_CODE < 0x020317) /* 2.3.23 */ 
#include "serial_compat.h" 
#endif 
 
#include  
#include  
#include  
#include  
 
#ifdef CONFIG_MAC_SERIAL 
#define SERIAL_DEV_OFFSET	2 
#else 
#define SERIAL_DEV_OFFSET	0 
#endif 
 
#ifdef SERIAL_INLINE 
#define _INLINE_ inline 
#else 
#define _INLINE_ 
#endif 
 
#define RdURXH2()	(*(volatile unsigned char *)ADDR_SERIAL_A) 
#define RdURXH3()	(*(volatile unsigned char *)ADDR_SERIAL_B) 
#define WrUTXH2(ch)	(*(volatile unsigned char *)ADDR_SERIAL_A)=(unsigned char)(ch) 
#define WrUTXH3(ch)	(*(volatile unsigned char *)ADDR_SERIAL_B)=(unsigned char)(ch) 
 
#define	rEXTINTPND  (0x01D20054) 
 
#define	EX_UART_GCR_RX_INT		0x01 
#define	EX_UART_GCR_TX_INT		0x04 
 
#define EX_UART_LCR	3 
#define	EX_UART_LSR	5 
#define	EX_UART_BDR	0 
#define	EX_UART_GCR	1 
#define	EX_UART_TX	0 
#define	EX_UART_RX	0 
 
static char *serial_name = "EV44B0II Serial driver"; 
extern unsigned int		dwCpuClk; 
extern unsigned int dwTestBaud; 
static DECLARE_TASK_QUEUE(tq_serial); 
 
static struct tty_driver serial_driver, callout_driver; 
static int serial_refcount; 
 
static struct timer_list serial_timer; 
_INLINE_ void disable_uart_tx_interrupt(int line);	 
_INLINE_ void disable_uart_rx_interrupt(int line); 
_INLINE_ void enable_uart_tx_interrupt(int line); 
_INLINE_ void enable_uart_rx_interrupt(int line); 
static _INLINE_ void wait_for_xmitr(struct async_struct *info); 
static _INLINE_ void transmit_chars(struct async_struct *info, int *intr_done); 
 
 
/* serial subtype definitions */ 
#ifndef SERIAL_TYPE_NORMAL 
#define SERIAL_TYPE_NORMAL	1 
#define SERIAL_TYPE_CALLOUT	2 
#endif 
 
/* number of characters left in xmit buffer before we ask for more */ 
#define WAKEUP_CHARS 256 
 
/* 
 * IRQ_timeout		- How long the timeout should be for each IRQ 
 * 				should be after the IRQ has been active. 
 */ 
 
static struct async_struct *IRQ_ports[NR_IRQS]; 
static int IRQ_timeout[NR_IRQS]; 
#ifdef CONFIG_SERIAL_SAMSUNG_CONSOLE 
static struct console sercons; 
static int lsr_break_flag; 
#endif 
#if defined(CONFIG_SERIAL_SAMSUNG_CONSOLE) && defined(CONFIG_MAGIC_SYSRQ) 
static unsigned long break_pressed; /* break, really ... */ 
#endif 
 
static unsigned detect_uart_irq (struct serial_state * state); 
static void autoconfig(struct serial_state * state); 
static void change_speed(struct async_struct *info, struct termios *old); 
static void rs_wait_until_sent(struct tty_struct *tty, int timeout); 
 
/* 
 * Here we define the default xmit fifo size used for each type of 
 * UART 
 */ 
static struct serial_uart_config uart_config[] = { 
	{ "unknown", 1, 0 },  
	{ "8250", 1, 0 },  
	{ "16450", 1, 0 },  
	{ "16550", 1, 0 },  
	{ "16550A", 16, UART_CLEAR_FIFO | UART_USE_FIFO },  
	{ "cirrus", 1, 0 }, 	// usurped by cyclades.c 
	{ "ST16650", 1, UART_CLEAR_FIFO | UART_STARTECH },  
	{ "ST16650V2", 32, UART_CLEAR_FIFO | UART_USE_FIFO | 
		  UART_STARTECH },  
	{ "TI16750", 64, UART_CLEAR_FIFO | UART_USE_FIFO}, 
	{ "Startech", 1, 0},	// usurped by cyclades.c 
	{ "16C950/954", 128, UART_CLEAR_FIFO | UART_USE_FIFO}, 
	{ "ST16654", 64, UART_CLEAR_FIFO | UART_USE_FIFO | 
		  UART_STARTECH },  
	{ "XR16850", 128, UART_CLEAR_FIFO | UART_USE_FIFO | 
		  UART_STARTECH }, 
	{ "RSA", 2048, UART_CLEAR_FIFO | UART_USE_FIFO },  
	{ "S3C44B0", 1, 0},	/* Samsung S3C44B0 uart support */ 
	{ 0, 0} 
}; 
 
static struct serial_state rs_table[RS_TABLE_SIZE] = { 
	SERIAL_PORT_DFNS	// Defined in serial.h 
}; 
 
 
#define NR_PORTS	(sizeof(rs_table)/sizeof(struct serial_state)) 
 
#ifndef PREPARE_FUNC 
#define PREPARE_FUNC(dev)  (dev->prepare) 
#define ACTIVATE_FUNC(dev)  (dev->activate) 
#define DEACTIVATE_FUNC(dev)  (dev->deactivate) 
#endif 
 
#define HIGH_BITS_OFFSET ((sizeof(long)-sizeof(int))*8) 
 
static struct tty_struct *serial_table[NR_PORTS]; 
static struct termios *serial_termios[NR_PORTS]; 
static struct termios *serial_termios_locked[NR_PORTS]; 
 
 
#if defined(MODULE) && defined(SERIAL_DEBUG_MCOUNT) 
#define DBG_CNT(s) printk("(%s): [%x] refc=%d, serc=%d, ttyc=%d -> %s\n", \ 
 kdevname(tty->device), (info->flags), serial_refcount,info->count,tty->count,s) 
#else 
#define DBG_CNT(s) 
#endif 
 
/* 
 * tmp_buf is used as a temporary buffer by serial_write.  We need to 
 * lock it in case the copy_from_user blocks while swapping in a page, 
 * and some other program tries to do a serial write at the same time. 
 * Since the lock will only come under contention when the system is 
 * swapping and available memory is low, it makes sense to share one 
 * buffer across all the serial ports, since it significantly saves 
 * memory if large numbers of serial ports are open. 
 */ 
static unsigned char *tmp_buf; 
#ifdef DECLARE_MUTEX 
static DECLARE_MUTEX(tmp_buf_sem); 
#else 
static struct semaphore tmp_buf_sem = MUTEX; 
#endif 
 
static _INLINE_ int serial_paranoia_check(struct async_struct *info, 
					kdev_t device, const char *routine) 
{ 
#ifdef SERIAL_PARANOIA_CHECK 
	static const char *badmagic = 
		"Warning: bad magic number for serial struct (%s) in %s\n"; 
	static const char *badinfo = 
		"Warning: null async_struct for (%s) in %s\n"; 
 
	if (!info) { 
		printk(badinfo, kdevname(device), routine); 
		return 1; 
	} 
	if (info->magic != SERIAL_MAGIC) { 
		printk(badmagic, kdevname(device), routine); 
		return 1; 
	} 
#endif 
	return 0; 
} 
 
static _INLINE_ unsigned int serial_in(struct async_struct *info, int offset) 
{ 
//	return CSR_READ(info->port + offset); 
	if(info->line==0||info->line==1) 
	{ 
		return CSR_READ(info->port + offset); 
	} 
	/*else                                                                   */ 
	/*{                                                                      */ 
		/*return (CSR_READ(GetOutAddr(GetInterAddr(info->port)+offset)));*/ 
	/*}                                                                      */ 
} 
 
static _INLINE_ void serial_out(struct async_struct *info, int offset, 
				int value) 
{ 
	if(info->line==0||info->line==1) 
	{ 
		CSR_WRITE(info->port+offset, value); 
	} 
	/*else                                                                 */ 
	/*{                                                                    */ 
		/*CSR_WRITE(GetOutAddr(GetInterAddr(info->port)+offset),value);*/ 
	/*}                                                                    */ 
} 
 
static _INLINE_ unsigned int uart_in(struct async_struct *info, int offset) 
{ 
	switch(info->line) 
	{ 
		case 0: 
		    return RdURXH0(); 
		case 1: 
		    return RdURXH1(); 
		case 2: 
		    return RdURXH2(); 
		default: 
		    return RdURXH3(); 
	    	     
	} 
 
} 
 
static _INLINE_ void uart_out(struct async_struct *info, int offset, 
				int value) 
{ 
	switch(info->line) 
	{ 
		case 0: 
			WrUTXH0(value); 
			break; 
		case 1: 
			WrUTXH1(value); 
			break; 
		default: 
			break; 
		/*case 2:                */ 
			/*WrUTXH2(value);*/ 
			/*break;         */ 
		/*default:               */ 
			/*WrUTXH3(value);*/ 
	} 
} 
 
/* 
 * We used to support using pause I/O for certain machines.  We 
 * haven't supported this for a while, but just in case it's badly 
 * needed for certain old 386 machines, I've left these #define's 
 * in.... 
 */ 
#define serial_inp(info, offset)		serial_in(info, offset) 
#define serial_outp(info, offset, value)	serial_out(info, offset, value) 
 
unsigned int baudrate_div(unsigned int baudrate) 
{ 
	dwTestBaud=((unsigned int)((float)dwCpuClk/16/baudrate + 0.5)) -1 ; 
	return dwTestBaud; 
} 
/* 
 * ------------------------------------------------------------ 
 * rs_stop() and rs_start() 
 * 
 * This routines are called before setting or resetting tty->stopped. 
 * They enable or disable transmitter interrupts, as necessary. 
 * ------------------------------------------------------------ 
 */ 
static void rs_stop(struct tty_struct *tty) 
{ 
	struct async_struct *info = (struct async_struct *)tty->driver_data; 
	unsigned long flags; 
 
	if (serial_paranoia_check(info, tty->device, "rs_stop")) 
		return; 
	 
	save_flags(flags); cli(); 
	disable_uart_tx_interrupt(info->line);  
	restore_flags(flags); 
} 
_INLINE_ void disable_uart_tx_interrupt(int line)	 
{ 
	unsigned int tmp; 
	unsigned int addr; 
	switch(line) 
	{ 
	    case 0: 
		INT_DISABLE(INT_UTXD0);	 
		CLEAR_PEND_INT(INT_UTXD0);	 
		break; 
	    case 1: 
		INT_DISABLE(INT_UTXD1);	 
		CLEAR_PEND_INT(INT_UTXD1);	 
		break; 
	    /*case 2:                                            */ 
		/*addr=GetOutAddr(GetInterAddr(ADDR_SERIAL_A)+1);*/ 
		/*tmp=CSR_READ(addr);                            */ 
		/*tmp&=~0x02;                                    */ 
		/*CSR_WRITE(addr,tmp);                           */ 
		/*break;                                         */ 
	    /*case 3:                                            */ 
		/*addr=GetOutAddr(GetInterAddr(ADDR_SERIAL_B)+1);*/ 
		/*tmp=CSR_READ(addr);                            */ 
		/*tmp&=~0x02;                                    */ 
		/*CSR_WRITE(addr,tmp);                           */ 
		/*break;                                         */ 
	    default: 
		break; 
	} 
} 
 
_INLINE_ void disable_uart_rx_interrupt(int line)	 
{ 
	unsigned int tmp; 
	unsigned int addr; 
	switch(line) 
	{ 
	    case 0: 
		INT_DISABLE(INT_URXD0); 
		CLEAR_PEND_INT(INT_URXD0); 
		break; 
	    case 1: 
		INT_DISABLE(INT_URXD1); 
		CLEAR_PEND_INT(INT_URXD1); 
		break; 
	    /*case 2:                                            */ 
		/*addr=GetOutAddr(GetInterAddr(ADDR_SERIAL_A)+1);*/ 
		/*tmp=CSR_READ(addr);                            */ 
		/*tmp&=~0x01;                                    */ 
		/*CSR_WRITE(addr,tmp);                           */ 
		/*break;                                         */ 
	    /*case 3:                                            */ 
		/*addr=GetOutAddr(GetInterAddr(ADDR_SERIAL_B)+1);*/ 
		/*tmp=CSR_READ(addr);                            */ 
		/*tmp&=~0x01;                                    */ 
		/*CSR_WRITE(addr,tmp);                           */ 
		/*break;                                         */ 
	    default: 
		break; 
	} 
} 
 
_INLINE_ void enable_uart_tx_interrupt(int line)	 
{ 
	unsigned int tmp; 
	unsigned int addr; 
 
	switch(line) 
	{ 
	    case 0: 
		INT_ENABLE(INT_UTXD0);	 
		break; 
	    case 1: 
		INT_ENABLE(INT_UTXD1); 
		break; 
	    /*case 2:                                            */ 
		/*addr=GetOutAddr(GetInterAddr(ADDR_SERIAL_A)+1);*/ 
		/*tmp=CSR_READ(addr);                            */ 
		/*tmp|=0x02;                                     */ 
		/*CSR_WRITE(addr,tmp);                           */ 
		/*break;                                         */ 
	    /*case 3:                                            */ 
		/*addr=GetOutAddr(GetInterAddr(ADDR_SERIAL_B)+1);*/ 
		/*tmp=CSR_READ(addr);                            */ 
		/*tmp|=0x02;                                     */ 
		/*CSR_WRITE(addr,tmp);                           */ 
		/*break;                                         */ 
	    default: 
		break; 
	} 
} 
 
_INLINE_ void enable_uart_rx_interrupt(int line) 
{ 
	unsigned int tmp; 
	unsigned int addr; 
	switch(line) 
	{ 
	    case 0: 
		INT_ENABLE(INT_URXD0); 
		break; 
	    case 1: 
		INT_ENABLE(INT_URXD1); 
		break; 
	    /*case 2:                                            */ 
		/*addr=GetOutAddr(GetInterAddr(ADDR_SERIAL_A)+1);*/ 
		/*tmp=CSR_READ(addr);                            */ 
		/*tmp|=0x01;                                     */ 
		/*CSR_WRITE(addr,tmp);                           */ 
		/*break;                                         */ 
	    /*case 3:                                            */ 
		/*addr=GetOutAddr(GetInterAddr(ADDR_SERIAL_B)+1);*/ 
		/*tmp=CSR_READ(addr);                            */ 
		/*tmp|=0x01;                                     */ 
		/*CSR_WRITE(addr,tmp);                           */ 
		/*break;                                         */ 
	    default: 
		break; 
	} 
} 
 
 
static void rs_start(struct tty_struct *tty) 
{ 
	struct async_struct *info = (struct async_struct *)tty->driver_data; 
	unsigned long flags; 
	 
	if (serial_paranoia_check(info, tty->device, "rs_start")) 
		return; 
	 
	save_flags(flags); cli(); 
	if (info->xmit.head != info->xmit.tail 
	    && info->xmit.buf) { 
		enable_uart_tx_interrupt(info->line); 
	} 
	restore_flags(flags); 
} 
 
/* 
 * ---------------------------------------------------------------------- 
 * 
 * Here starts the interrupt handling routines.  All of the following 
 * subroutines are declared as inline and are folded into 
 * rs_interrupt().  They were separated out for readability's sake. 
 * 
 * Note: rs_interrupt() is a "fast" interrupt, which means that it 
 * runs with interrupts turned off.  People who may want to modify 
 * rs_interrupt() should try to keep the interrupt handler as fast as 
 * possible.  After you are done making modifications, it is not a bad 
 * idea to do: 
 *  
 * gcc -S -DKERNEL -Wall -Wstrict-prototypes -O6 -fomit-frame-pointer serial.c 
 * 
 * and look at the resulting assemble code in serial.s. 
 * 
 * 				- Ted Ts'o (tytso@mit.edu), 7-Mar-93 
 * ----------------------------------------------------------------------- 
 */ 
 
/* 
 * This routine is used by the interrupt handler to schedule 
 * processing in the software interrupt portion of the driver. 
 */ 
static _INLINE_ void rs_sched_event(struct async_struct *info, 
				  int event) 
{ 
	info->event |= 1 << event; 
	queue_task(&info->tqueue, &tq_serial); 
	mark_bh(SERIAL_BH); 
} 
 
static _INLINE_ void receive_chars(struct async_struct *info, 
				 int *status, struct pt_regs * regs) 
{ 
	struct tty_struct *tty = info->tty; 
	unsigned char ch; 
	struct	async_icount *icount; 
	int	max_count = 256; 
 
#ifdef CONFIG_LEDMAN 
	ledman_cmd(LEDMAN_CMD_SET, 
		(info->line == 0) ? LEDMAN_COM1_RX : LEDMAN_COM2_RX); 
#endif 
 
	icount = &info->state->icount; 
	do { 
		if (tty->flip.count >= TTY_FLIPBUF_SIZE) { 
			tty->flip.tqueue.routine((void *) tty); 
			if (tty->flip.count >= TTY_FLIPBUF_SIZE) 
				return;		// if TTY_DONT_FLIP is set 
		} 
		if(info->line==0||info->line==1) 
			ch = uart_in(info, UART_RX); 
		/*                                       */ 
		/*else                                   */ 
		/*        ch = uart_in(info, EX_UART_RX);*/ 
		*tty->flip.char_buf_ptr = ch; 
		icount->rx++; 
		 
#ifdef SERIAL_DEBUG_INTR 
		printk("DR%02x:%02x...", ch, *status); 
#endif 
		*tty->flip.flag_buf_ptr = 0; 
/* 
		if (*status & (UART_LSR_BI | UART_LSR_PE | 
			       UART_LSR_FE | UART_LSR_OE)) { 
			// 
			// For statistics only 
			// 
			if (*status & UART_LSR_BI) { 
				*status &= ~(UART_LSR_FE | UART_LSR_PE); 
				icount->brk++; 
				// 
				// We do the SysRQ and SAK checking 
				// here because otherwise the break 
				// may get masked by ignore_status_mask 
				// or read_status_mask. 
				// 
#if defined(CONFIG_SERIAL_SAMSUNG_CONSOLE) && defined(CONFIG_MAGIC_SYSRQ) 
				if (info->line == sercons.index) { 
					if (!break_pressed) { 
						break_pressed = jiffies; 
						goto ignore_char; 
					} 
					break_pressed = 0; 
				} 
#endif 
				if (info->flags & ASYNC_SAK) 
					do_SAK(tty); 
			} else if (*status & UART_LSR_PE) 
				icount->parity++; 
			else if (*status & UART_LSR_FE) 
				icount->frame++; 
			if (*status & UART_LSR_OE) 
				icount->overrun++; 
*/ 
			// 
			// Mask off conditions which should be ignored. 
			// 
			*status &= info->read_status_mask; 
 
#ifdef CONFIG_SERIAL_SAMSUNG_CONSOLE 
			if (info->line == sercons.index) { 
				// Recover the break flag from console xmit 
				*status |= lsr_break_flag; 
				lsr_break_flag = 0; 
			} 
#endif 
 
/*			if (*status & (UART_LSR_BI)) { 
#ifdef SERIAL_DEBUG_INTR 
				printk("handling break...."); 
#endif 
				*tty->flip.flag_buf_ptr = TTY_BREAK; 
			} else if (*status & UART_LSR_PE) 
				*tty->flip.flag_buf_ptr = TTY_PARITY; 
			else if (*status & UART_LSR_FE) 
				*tty->flip.flag_buf_ptr = TTY_FRAME; 
		}*/ 
#if defined(CONFIG_SERIAL_SAMSUNG_CONSOLE) && defined(CONFIG_MAGIC_SYSRQ) 
		if (break_pressed && info->line == sercons.index) { 
			if (ch != 0 && 
			    time_before(jiffies, break_pressed + HZ*5)) { 
				handle_sysrq(ch, regs, NULL, NULL); 
				break_pressed = 0; 
				goto ignore_char; 
			} 
			break_pressed = 0; 
		} 
#endif 
		if ((*status & info->ignore_status_mask) == 0) { 
			tty->flip.flag_buf_ptr++; 
			tty->flip.char_buf_ptr++; 
			tty->flip.count++; 
		} 
/*		if ((*status & UART_LSR_OE) && 
		    (tty->flip.count < TTY_FLIPBUF_SIZE)) { 
			// 
			// Overrun is special, since it's reported 
			// immediately, and doesn't affect the current 
			// character 
			// 
			*tty->flip.flag_buf_ptr = TTY_OVERRUN; 
			tty->flip.count++; 
			tty->flip.flag_buf_ptr++; 
			tty->flip.char_buf_ptr++; 
		}*/ 
#if defined(CONFIG_SERIAL_SAMSUNG_CONSOLE) && defined(CONFIG_MAGIC_SYSRQ) 
	ignore_char: 
#endif 
		if(info->line==0||info->line==1) 
			*status = serial_inp(info, UART_LSR); 
		else 
			*status = serial_inp(info, EX_UART_LSR); 
		     
	} while ((*status & UART_LSR_DR) && (max_count-- > 0)); 
#if (LINUX_VERSION_CODE > 131394) // 2.1.66 
	tty_flip_buffer_push(tty); 
#else 
	queue_task_irq_off(&tty->flip.tqueue, &tq_timer); 
#endif	 
} 
 
static _INLINE_ void transmit_chars(struct async_struct *info, int *intr_done) 
{ 
	int count; 
	unsigned int status; 
	 
#ifdef CONFIG_LEDMAN 
	ledman_cmd(LEDMAN_CMD_SET, 
		(info->line == 0) ? LEDMAN_COM1_TX : LEDMAN_COM2_TX); 
#endif 
 
	/*if(info->line==2)             */ 
	    /*printk("串口2发送\n");    */ 
	/*else                          */ 
	    /*if(info->line==3)         */ 
		/*printk("串口3发送\n");*/ 
	if (info->x_char) { 
		if(info->line==0||info->line==1) 
			uart_out(info, UART_TX, info->x_char); 
		/*else                                             */ 
			/*uart_out(info, EX_UART_TX, info->x_char);*/ 
		info->state->icount.tx++; 
		info->x_char = 0; 
		if (intr_done) 
			*intr_done = 0; 
		return; 
	} 
	if (info->xmit.head == info->xmit.tail 
	    || info->tty->stopped 
	    || info->tty->hw_stopped) { 
		disable_uart_tx_interrupt(info->line); 
		return; 
	} 
	 
	count = info->xmit_fifo_size; 
	do { 
		if(info->line==0||info->line==1) 
			uart_out(info, UART_TX, info->xmit.buf[info->xmit.tail]); 
		/*else                                                                */ 
			/*uart_out(info, EX_UART_TX, info->xmit.buf[info->xmit.tail]);*/ 
		     
		info->xmit.tail = (info->xmit.tail + 1) & (SERIAL_XMIT_SIZE-1); 
		info->state->icount.tx++; 
		if (info->xmit.head == info->xmit.tail) 
			break; 
		wait_for_xmitr(info); 
		status&=(~(UART_LSR_TEMT | UART_LSR_THRE)); 
		if(info->line==0||info->line==1) 
			serial_out(info,UART_LSR,status); 
		/*else                                        */ 
			/*serial_out(info,EX_UART_LSR,status);*/ 
		     
	} while (--count > 0); 
	 
	if (CIRC_CNT(info->xmit.head, 
		     info->xmit.tail, 
		     SERIAL_XMIT_SIZE) < WAKEUP_CHARS) 
		rs_sched_event(info, RS_EVENT_WRITE_WAKEUP); 
 
#ifdef SERIAL_DEBUG_INTR 
	printk("THRE..."); 
#endif 
	if (intr_done) 
		*intr_done = 0; 
 
	if (info->xmit.head == info->xmit.tail) { 
		disable_uart_tx_interrupt(info->line); 
	} 
} 
 
 
/* 
 * This is the serial driver's interrupt routine for a single port 
 */ 
static void rs_interrupt_single(int irq, void *dev_id, struct pt_regs * regs) 
{ 
	int status; 
	int pass_counter = 0; 
	struct async_struct * info; 
 
	info = IRQ_ports[irq]; 
	if (!info || !info->tty) 
		return; 
 
    if ((irq == INT_URXD1) || (irq == INT_URXD0)){ // Qinwei 2002.09.16		if (status & UART_LSR_THRE) 
	    do { 
		if(info->line==0||info->line==1) 
			status = serial_inp(info, UART_LSR); 
		else 
    			status = serial_inp(info, EX_UART_LSR); 
 
    		if (status & UART_LSR_DR) 
    			receive_chars(info, &status, regs); 
     
    		if (pass_counter++ > RS_ISR_PASS_LIMIT) { 
    			break; 
    		} 
    	} while (status & UART_LSR_DR ); 
	    info->last_active = jiffies; 
    	return; 
    } 
     
   if ((irq == INT_UTXD1) || (irq == INT_UTXD0)){ // Qinwei 2002.09.16		if (status & UART_LSR_THRE) 
        transmit_chars(info, 0); 
        info->last_active = jiffies; 
        return; 
    } 
} 
/****************************************************************************** 
 * 	功能:		83977串行中断程序 
 ******************************************************************************/ 
/*inline void do_com_interrupt(struct async_struct * info,int * status,struct pt_regs * regs)*/ 
/*{                                                                                          */ 
/*        unsigned char bInterruptNo;                                                        */ 
/*        //bInterruptNo=0x07&inb(GetOutAddr(wInterAddr+2));                                 */ 
/*        bInterruptNo=0x07&serial_in(info,2);                                               */ 
/*        switch(bInterruptNo)                                                               */ 
/*        {                                                                                  */ 
/*                case 0:                                                                    */ 
/*                        //modem_status                                                     */ 
/*                        serial_in(info,6);                                                 */ 
/*                        break;                                                             */ 
/*                case 2:                                                                    */ 
/*                        //write_char                                                       */ 
/*                        transmit_chars(info, 0);	                                     */ 
/*                        break;                                                             */ 
/*                case 4:                                                                    */ 
/*                        //read_char                                                        */ 
/*                        receive_chars(info, status, regs);                                 */ 
/*                        break;                                                             */ 
/*                case 6:                                                                    */ 
/*                        //line_status                                                      */ 
/*                        serial_in(info,5);                                                 */ 
/*                        break;                                                             */ 
/*                default:                                                                   */ 
/*                        break;                                                             */ 
/*        }                                                                                  */ 
/*}                                                                                          */ 
/* 
static void rs_83977_interrupt(int irq, void *dev_id, struct pt_regs * regs) 
{ 
 
	unsigned int 	tmp; 
	int status; 
	struct async_struct * info; 
	info = IRQ_ports[irq]; 
	if (!info || !info->tty) 
		return; 
	tmp=CSR_READ(rEXTINTPND); 
	if(tmp&0x01) 
	{ 
		//串口3中断 
		printk("串口3中断\n"); 
		do_com_interrupt(info,&status,regs); 
		CSR_WRITE(rEXTINTPND,tmp&(~0x01)); 
	} 
	if(tmp&0x02) 
	{ 
		//串口4中断 
		printk("串口4中断\n"); 
		do_com_interrupt(info,&status,regs); 
		CSR_WRITE(rEXTINTPND,tmp&(~0x02)); 
	} 
} 
*/ 
/* 
 * ------------------------------------------------------------------- 
 * Here ends the serial interrupt routines. 
 * ------------------------------------------------------------------- 
 */ 
 
/* 
 * This routine is used to handle the "bottom half" processing for the 
 * serial driver, known also the "software interrupt" processing. 
 * This processing is done at the kernel interrupt level, after the 
 * rs_interrupt() has returned, BUT WITH INTERRUPTS TURNED ON.  This 
 * is where time-consuming activities which can not be done in the 
 * interrupt driver proper are done; the interrupt driver schedules 
 * them using rs_sched_event(), and they get done here. 
 */ 
static void do_serial_bh(void) 
{ 
	run_task_queue(&tq_serial); 
} 
 
static void do_softint(void *private_) 
{ 
	struct async_struct	*info = (struct async_struct *) private_; 
	struct tty_struct	*tty; 
	 
	tty = info->tty; 
	if (!tty) 
		return; 
 
	if (test_and_clear_bit(RS_EVENT_WRITE_WAKEUP, &info->event)) { 
		if ((tty->flags & (1 << TTY_DO_WRITE_WAKEUP)) && 
		    tty->ldisc.write_wakeup) 
			(tty->ldisc.write_wakeup)(tty); 
		wake_up_interruptible(&tty->write_wait); 
#ifdef SERIAL_HAVE_POLL_WAIT 
		wake_up_interruptible(&tty->poll_wait); 
#endif 
	} 
} 
 
/* 
 * This subroutine is called when the RS_TIMER goes off.  It is used 
 * by the serial driver to handle ports that do not have an interrupt 
 * (irq=0).  This doesn't work very well for 16450's, but gives barely 
 * passable results for a 16550A.  (Although at the expense of much 
 * CPU overhead). 
 */ 
static void rs_timer(unsigned long dummy) 
{ 
#ifndef UARTPOLLING  
	static unsigned long last_strobe; 
	struct async_struct *info; 
	unsigned int	i; 
	unsigned long flags; 
 
	if ((jiffies - last_strobe) >= RS_STROBE_TIME) { 
		for (i=0; i < NR_IRQS; i++) { 
			info = IRQ_ports[i]; 
			if (!info) 
				continue; 
			save_flags(flags); cli(); 
			rs_interrupt_single(i, NULL, NULL); 
			restore_flags(flags); 
		} 
	} 
	last_strobe = jiffies; 
	mod_timer(&serial_timer, jiffies + RS_STROBE_TIME); 
 
	if (IRQ_ports[0]) { 
		save_flags(flags); cli(); 
		rs_interrupt_single(0, NULL, NULL); 
		restore_flags(flags); 
 
		mod_timer(&serial_timer, jiffies + IRQ_timeout[0]); 
	} 
#endif 
} 
/* 
 * --------------------------------------------------------------- 
 * Low level utility subroutines for the serial driver:  routines to 
 * figure out the appropriate timeout for an interrupt chain, routines 
 * to initialize and startup a serial port, and routines to shutdown a 
 * serial port.  Useful stuff like that. 
 * --------------------------------------------------------------- 
 */ 
 
/* 
 * This routine figures out the correct timeout for a particular IRQ. 
 * It uses the smallest timeout of all of the serial ports in a 
 * particular interrupt chain.  Now only used for IRQ 0.... 
 */ 
static void figure_IRQ_timeout(int irq) 
{ 
	struct	async_struct	*info; 
	int	timeout = 60*HZ;	// 60 seconds === a long time :-) 
 
	info = IRQ_ports[irq]; 
	if (!info) { 
		IRQ_timeout[irq] = 60*HZ; 
		return; 
	} 
	while (info) { 
		if (info->timeout < timeout) 
			timeout = info->timeout; 
		info = info->next_port; 
	} 
	if (!irq) 
		timeout = timeout / 2; 
	IRQ_timeout[irq] = (timeout > 3) ? timeout-2 : 1; 
} 
 
static int startup(struct async_struct * info) 
{ 
	unsigned long flags; 
	int	retval=0; 
	void (*handler)(int, void *, struct pt_regs *); 
	struct serial_state *state= info->state; 
	unsigned long page; 
 
	page = get_zeroed_page(GFP_KERNEL); 
	if (!page) 
		return -ENOMEM; 
 
	save_flags(flags); cli(); 
 
	if (info->flags & ASYNC_INITIALIZED) { 
		free_page(page); 
		goto errout; 
	} 
 
	if (!CONFIGURED_SERIAL_PORT(state) || !state->type) { 
		if (info->tty) 
			set_bit(TTY_IO_ERROR, &info->tty->flags); 
		free_page(page); 
		goto errout; 
	} 
	if (info->xmit.buf) 
		free_page(page); 
	else 
		info->xmit.buf = (unsigned char *) page; 
 
#ifdef SERIAL_DEBUG_OPEN 
	printk("starting up ttys%d (irq %d)...", info->line, state->irq); 
#endif 
	switch(info->line) 
	{ 
	    case 0: 
	    case 1: 
		CLEAR_PEND_INT(state->irq);   /* clear serial UART rx interrupt */ 
		CLEAR_PEND_INT(state->irq-4); /* clear serial UART tx interrupt */ 
		break; 
						      /**/ 
	    /*case 2:                                   */ 
	    /*case 3:                                   */ 
		/*disable_uart_tx_interrupt(info->line);*/ 
		/*enable_uart_tx_interrupt(info->line); */ 
		/*disable_uart_rx_interrupt(info->line);*/ 
		/*enable_uart_rx_interrupt(info->line); */ 
		/*break;                                */ 
	    default: 
		break; 
	} 
 
	// 
	// Allocate the IRQ if necessary 
	// 
	if (state->irq && (!IRQ_ports[state->irq] || 
			  !IRQ_ports[state->irq]->next_port)) { 
		if (IRQ_ports[state->irq]) { 
			retval = -EBUSY; 
			goto errout; 
		} else 
		{ 
			if(info->line==0||info->line==1) 
			{ 
			    handler = rs_interrupt_single; 
			    // Mac Wang register rx and tx seperatly 
			    retval = request_irq(state->irq, handler, SA_SHIRQ, 
						 "serial Rx", &IRQ_ports[state->irq]); 
 
			    retval = request_irq(state->irq - 4, handler, SA_SHIRQ, 
						 "serial Tx", &IRQ_ports[state->irq-4]); 
 
			} 
			/* 
			else 
			{ 
			    handler = rs_83977_interrupt; 
			    printk("rs_83977_interrupt:%d\n",state->irq); 
			    retval = request_irq(state->irq, handler, SA_SHIRQ, 
						 "83977 serial Tx && Rx", &IRQ_ports[state->irq]); 
			} 
			*/ 
		} 
 
 
		if (retval) { 
			if (capable(CAP_SYS_ADMIN)) { 
				if (info->tty) 
					set_bit(TTY_IO_ERROR, 
						&info->tty->flags); 
				retval = 0; 
			} 
			goto errout; 
		} 
	} 
 
	// 
	// Insert serial port into IRQ chain. 
	// 
	info->prev_port = 0; 
	info->next_port = IRQ_ports[state->irq]; 
	if (info->next_port) 
		info->next_port->prev_port = info; 
	if(info->line==0||info->line==1) 
	{ 
	    IRQ_ports[state->irq] = info; 
	    figure_IRQ_timeout(state->irq); 
 
	    IRQ_ports[state->irq - 4] = info; 
	    figure_IRQ_timeout(state->irq - 4); 
	} 
	/* 
	else 
	{ 
	    IRQ_ports[state->irq] = info; 
	    figure_IRQ_timeout(state->irq); 
	} 
	*/ 
	// 
	// Now, initialize the UART  
	// 
/* 
	serial_outp(info, UART_LCR, UART_LCR_WLEN8);	// reset DLAB 
 
	info->MCR = 0; 
	if (info->tty->termios->c_cflag & CBAUD) 
		info->MCR = UART_MCR_DTR | UART_MCR_RTS; 
	{ 
		if (state->irq != 0) 
			info->MCR |= UART_MCR_OUT2; 
	} 
	info->MCR |= ALPHA_KLUDGE_MCR; 		// Don't ask 
	serial_outp(info, UART_MCR, info->MCR); 
*/ 
#ifdef CONFIG_SERIAL_SAMSUNG_IRDA 
	if (info->line == 1){ 
		serial_outp(info, UART_LCR, UART_LCR_WLEN8|UART_LCR_IRDA); 
	} 
	else 
		serial_outp(info, UART_LCR, UART_LCR_WLEN8); // UART0 don't support IRDA mode 
#else 
	if(info->line==0||info->line==1) 
		serial_outp(info, UART_LCR, UART_LCR_WLEN8); 
	/*else                                                   */ 
	/*        serial_outp(info, EX_UART_LCR, UART_LCR_WLEN8);*/ 
	     
#endif 
 
	/* Enable the I/O pins*/ 
	if (info->line == 1) 
	{ 
		CSR_WRITE(rPCONC,((3 << 20)|(3 << 22)|(3 << 24)| (3 << 26) | (3<<28)|(3 <<30))); // Enable UART1's Pin 
		CSR_WRITE(rPUPC,CSR_READ(rPUPC)|0xfc00); // set pull up 
#ifdef CONFIG_SERIAL_SAMSUNG_IRDA 
		CSR_WRITE(rPCONE,CSR_READ(rPCONE)|1);    // Enable UART0's Pin 
		CSR_WRITE(rPDATE,CSR_READ(rPDATE)&0xFE); // select enable IRDA 
#endif 
	} 
	else 
	{ 
		if(info->line==0) 
		{ 
			CSR_WRITE(rPCONE,CSR_READ(rPCONE)|(2 << 2) | (2 << 4));		 // Enable UART0's Pin 
		} 
	} 
 
//	serial_outp(info, UART_GCR, UART_GCR_RX_INT|UART_GCR_TX_INT|UART_GCR_RX_STAT_INT); 
	if(info->line==0||info->line==1) 
	{  
		serial_outp(info, UART_GCR, UART_GCR_RX_INT|UART_GCR_TX_INT); 
	} 
	/*else                                                                          */ 
	/*{                                                                             */ 
		/*serial_outp(info, EX_UART_GCR, EX_UART_GCR_RX_INT|EX_UART_GCR_TX_INT);*/ 
	/*}                                                                             */ 
 
	// 
	// Finally, enable interrupts 
	// 
/* 
	info->IER = UART_IER_MSI | UART_IER_RLSI | UART_IER_RDI; 
	serial_outp(info, UART_IER, info->IER);	// enable interrupts 
*/ 
	enable_uart_rx_interrupt(info->line) ; 
	 
 
	// 
	// And clear the interrupt registers again for luck. 
	// 
	CLEAR_PEND_INT(state->irq);   /* clear serial UART rx interrupt */ 
	if(info->line==0||info->line==1) 
	{ 
	    CLEAR_PEND_INT(state->irq-4); /* clear serial UART tx interrupt */ 
	} 
 
	if (info->tty) 
		clear_bit(TTY_IO_ERROR, &info->tty->flags); 
	info->xmit.head = info->xmit.tail = 0; 
 
	// 
	// Set up serial timers... 
	// 
	mod_timer(&serial_timer, jiffies + 2*HZ/100); 
 
 	// 
	// Set up the tty->alt_speed kludge 
	// 
#if (LINUX_VERSION_CODE >= 131394) // Linux 2.1.66 
	if (info->tty) { 
		if ((info->flags & ASYNC_SPD_MASK) == ASYNC_SPD_HI) 
			info->tty->alt_speed = 57600; 
		if ((info->flags & ASYNC_SPD_MASK) == ASYNC_SPD_VHI) 
			info->tty->alt_speed = 115200; 
		if ((info->flags & ASYNC_SPD_MASK) == ASYNC_SPD_SHI) 
			info->tty->alt_speed = 230400; 
		if ((info->flags & ASYNC_SPD_MASK) == ASYNC_SPD_WARP) 
			info->tty->alt_speed = 460800; 
	} 
#endif 
	 
	// 
	// and set the speed of the serial port 
	// 
	change_speed(info, 0); 
 
	info->flags |= ASYNC_INITIALIZED; 
	restore_flags(flags); 
	return 0; 
	 
errout: 
	restore_flags(flags); 
	return retval; 
} 
 
/* 
 * This routine will shutdown a serial port; interrupts are disabled, and 
 * DTR is dropped if the hangup on close termio flag is on. 
 */ 
static void shutdown(struct async_struct * info) 
{ 
	unsigned long	flags; 
	struct serial_state *state; 
	int		retval; 
 
	if (!(info->flags & ASYNC_INITIALIZED)) 
		return; 
 
	state = info->state; 
 
#ifdef SERIAL_DEBUG_OPEN 
	printk("Shutting down serial port %d (irq %d)....", info->line, 
	       state->irq); 
#endif 
	 
	save_flags(flags); cli(); // Disable interrupts 
 
	// 
	// clear delta_msr_wait queue to avoid mem leaks: we may free the irq 
	// here so the queue might never be waken up 
	// 
	wake_up_interruptible(&info->delta_msr_wait); 
	 
	// 
	// First unlink the serial port from the IRQ chain... 
	// 
	if (info->next_port) 
		info->next_port->prev_port = info->prev_port; 
	if (info->prev_port) 
		info->prev_port->next_port = info->next_port; 
	else 
		IRQ_ports[state->irq] = info->next_port; 
	figure_IRQ_timeout(state->irq); 
	 
	// 
	// Free the IRQ, if necessary 
	// 
	// rx 
	if (state->irq && (!IRQ_ports[state->irq] || 
			  !IRQ_ports[state->irq]->next_port)) { 
		if (IRQ_ports[state->irq]) { 
			free_irq(state->irq, &IRQ_ports[state->irq]); 
			retval = request_irq(state->irq, rs_interrupt_single, 
					     SA_SHIRQ, "serial", 
					     &IRQ_ports[state->irq]); 
			 
			if (retval) 
				printk("serial shutdown: request_irq: error %d" 
				       "  Couldn't reacquire IRQ.\n", retval); 
		} else 
			free_irq(state->irq, &IRQ_ports[state->irq]); 
	} 
	// tx 
 
	if (state->irq - 4 && (!IRQ_ports[state->irq - 4] || 
			  !IRQ_ports[state->irq - 4]->next_port)) { 
		free_irq(state->irq-4, &IRQ_ports[state->irq-4]); 
	} 
 
	if (info->xmit.buf) { 
		unsigned long pg = (unsigned long) info->xmit.buf; 
		info->xmit.buf = 0; 
		free_page(pg); 
	} 
 
/* 
	info->IER = 0; 
	serial_outp(info, UART_IER, 0x00);	// disable all intrs 
*/ 
	disable_uart_rx_interrupt(info->line); 
	disable_uart_tx_interrupt(info->line); 
	if(info->line==0||info->line==1) 
		(void)uart_in(info, UART_RX);    // read data port to reset things 
	/*else                                                                         */ 
		/*(void)uart_in(info, EX_UART_RX);    // read data port to reset things*/ 
 
	 
	if (info->tty) 
		set_bit(TTY_IO_ERROR, &info->tty->flags); 
 
	info->flags &= ~ASYNC_INITIALIZED; 
	restore_flags(flags); 
} 
 
#if (LINUX_VERSION_CODE < 131394) /* Linux 2.1.66 */ 
static int baud_table[] = { 
	0, 50, 75, 110, 134, 150, 200, 300, 
	600, 1200, 1800, 2400, 4800, 9600, 19200, 
	38400, 57600, 115200, 230400, 460800, 0 }; 
#endif 
 
/* 
 * This routine is called to set the UART divisor registers to match 
 * the specified baud rate for a serial port. 
 */ 
static void change_speed(struct async_struct *info, 
			 struct termios *old_termios) 
{ 
	int	quot = 0, baud_base, baud; 
//	unsigned cflag, cval, fcr = 0; 
	unsigned cflag, cval; 
	int	bits; 
	unsigned long	flags; 
 
	if (!info->tty || !info->tty->termios) 
		return; 
	cflag = info->tty->termios->c_cflag; 
	if (!CONFIGURED_SERIAL_PORT(info)) 
		return; 
 
	// byte size and parity 
	switch (cflag & CSIZE) { 
	      case CS5: cval = 0x00; bits = 7; break; 
	      case CS6: cval = 0x01; bits = 8; break; 
	      case CS7: cval = 0x02; bits = 9; break; 
	      case CS8: cval = 0x03; bits = 10; break; 
	      // Never happens, but GCC is too dumb to figure it out 
	      default:  cval = 0x00; bits = 7; break; 
	      } 
	if (cflag & CSTOPB) { 
		cval |= 0x04; 
		bits++; 
	} 
	if (cflag & PARENB) { 
		bits++; 
	} 
	if (cflag & PARODD) 
		cval |= UART_LCR_OPAR; 
	else if (cflag & PARENB) 
		cval |= UART_LCR_EPAR; 
	else 
		cval |= UART_LCR_NPAR; 
 
#ifdef CONFIG_SERIAL_SAMSUNG_IRDA 
	if (info->line == 1){ 
		cval |= UART_LCR_IRDA; 
	} 
#endif 
 
	// Determine divisor based on baud rate 
	baud = tty_get_baud_rate(info->tty); 
	if (!baud) 
		baud = 9600;	// B0 transition handled in rs_set_termios 
 
	baud_base = info->state->baud_base; 
 
	quot = 1;	//tricky 
 
	// If the quotient is zero refuse the change 
	if (!quot && old_termios) { 
		info->tty->termios->c_cflag &= ~CBAUD; 
		info->tty->termios->c_cflag |= (old_termios->c_cflag & CBAUD); 
		baud = tty_get_baud_rate(info->tty); 
		if (!baud) 
			baud = 9600; 
	} 
	// As a last resort, if the quotient is zero, default to 9600 bps 
	if (!quot) 
		quot = baud_base / 9600; 
	// 
	// Work around a bug in the Oxford Semiconductor 952 rev B 
	// chip which causes it to seriously miscalculate baud rates 
	// when DLL is 0. 
	// 
	info->quot = quot; 
	info->timeout = ((info->xmit_fifo_size*HZ*bits*quot) / baud_base); 
	info->timeout += HZ/50;		// Add .02 seconds of slop 
 
 
	// 
	// Set up parity check flag 
	// 
#define RELEVANT_IFLAG(iflag) (iflag & (IGNBRK|BRKINT|IGNPAR|PARMRK|INPCK)) 
 
	info->read_status_mask = UART_LSR_OE | UART_LSR_THRE | UART_LSR_DR; 
	if (I_INPCK(info->tty)) 
		info->read_status_mask |= UART_LSR_FE | UART_LSR_PE; 
	if (I_BRKINT(info->tty) || I_PARMRK(info->tty)) 
		info->read_status_mask |= UART_LSR_BI; 
	 
	// 
	// Characters to ignore 
	// 
	info->ignore_status_mask = 0; 
	if (I_IGNPAR(info->tty)) 
		info->ignore_status_mask |= UART_LSR_PE | UART_LSR_FE; 
	if (I_IGNBRK(info->tty)) { 
		info->ignore_status_mask |= UART_LSR_BI; 
		// 
		// If we're ignore parity and break indicators, ignore  
		// overruns too.  (For real raw support). 
		// 
		if (I_IGNPAR(info->tty)) 
			info->ignore_status_mask |= UART_LSR_OE; 
	} 
	save_flags(flags); cli(); 
	info->LCR = cval;				// Save LCR 
	if(info->line==0||info->line==1) 
	{ 
		serial_outp(info, UART_BDR, baudrate_div(baud)); 
		serial_outp(info, UART_LCR, cval); 
	} 
	/*else                                                             */ 
	/*{                                                                */ 
		/*serial_outp(info, EX_UART_LCR, 0x80);	//set line ctrol   */ 
		/*serial_outp(info, EX_UART_BDR, 115200%baud);             */ 
		/*serial_outp(info, EX_UART_BDR+1, 115200/baud);           */ 
		/*serial_outp(info, EX_UART_LCR, cval);	//re set line ctrol*/ 
	/*}                                                                */ 
	restore_flags(flags); 
} 
 
static void rs_put_char(struct tty_struct *tty, unsigned char ch) 
{ 
	struct async_struct *info = (struct async_struct *)tty->driver_data; 
	unsigned long flags; 
 
	if (serial_paranoia_check(info, tty->device, "rs_put_char")) 
		return; 
 
	if (!tty || !info->xmit.buf) 
		return; 
 
	save_flags(flags); cli(); 
	if (CIRC_SPACE(info->xmit.head, 
		       info->xmit.tail, 
		       SERIAL_XMIT_SIZE) == 0) { 
		restore_flags(flags); 
		return; 
	} 
#ifdef UARTPOLLING 
	wait_for_xmitr(info); 
	/*	Send the character out. */ 
	if(info->line==0||info->line==1) 
		uart_out(info, UART_TX, ch); 
	/*else                                   */ 
		/*uart_out(info, EX_UART_TX, ch);*/ 
	     
#else 
	info->xmit.buf[info->xmit.head] = ch; 
	info->xmit.head = (info->xmit.head + 1) & (SERIAL_XMIT_SIZE-1); 
#endif 
	restore_flags(flags); 
} 
 
static void rs_flush_chars(struct tty_struct *tty) 
{ 
	struct async_struct *info = (struct async_struct *)tty->driver_data; 
	unsigned long flags; 
				 
	if (serial_paranoia_check(info, tty->device, "rs_flush_chars")) 
		return; 
 
	if (info->xmit.head == info->xmit.tail 
	    || tty->stopped 
	    || tty->hw_stopped 
	    || !info->xmit.buf) 
		return; 
 
	save_flags(flags); cli(); 
 
	enable_uart_tx_interrupt(info->line); 
	restore_flags(flags); 
} 
 
static int rs_write(struct tty_struct * tty, int from_user, 
		    const unsigned char *buf, int count) 
{ 
	int	c, ret = 0; 
	struct async_struct *info = (struct async_struct *)tty->driver_data; 
	unsigned long flags; 
				 
	if (serial_paranoia_check(info, tty->device, "rs_write")) 
		return 0; 
 
	if (!tty || !info->xmit.buf || !tmp_buf) 
		return 0; 
 
	save_flags(flags); 
	if (from_user) { 
		down(&tmp_buf_sem); 
		while (1) { 
			int c1; 
			c = CIRC_SPACE_TO_END(info->xmit.head, info->xmit.tail, SERIAL_XMIT_SIZE); 
			if (count < c) 
				c = count; 
			if (c <= 0) 
				break; 
 
			c -= copy_from_user(tmp_buf, buf, c); 
			if (!c) { 
				if (!ret) 
					ret = -EFAULT; 
				break; 
			} 
			cli(); 
			c1 = CIRC_SPACE_TO_END(info->xmit.head, info->xmit.tail, SERIAL_XMIT_SIZE); 
			if (c1 < c) 
				c = c1; 
			memcpy(info->xmit.buf + info->xmit.head, tmp_buf, c); 
			info->xmit.head = ((info->xmit.head + c) & (SERIAL_XMIT_SIZE-1)); 
			restore_flags(flags); 
			buf += c; 
			count -= c; 
			ret += c; 
		} 
		up(&tmp_buf_sem); 
	} else { 
		cli(); 
		while (1) { 
			c = CIRC_SPACE_TO_END(info->xmit.head, info->xmit.tail, SERIAL_XMIT_SIZE); 
			if (count < c) 
				c = count; 
			if (c <= 0) { 
				break; 
			} 
			memcpy(info->xmit.buf + info->xmit.head, buf, c); 
			info->xmit.head = ((info->xmit.head + c) & (SERIAL_XMIT_SIZE-1)); 
			buf += c; 
			count -= c; 
			ret += c; 
		} 
		restore_flags(flags); 
	} 
	if (info->xmit.head != info->xmit.tail && !tty->stopped && !tty->hw_stopped)  
	{ 
#ifdef UARTPOLLING 
		cli(); 
        count = ret; 
        do { 
        	wait_for_xmitr(info); 
        	/*	Send the character out. */ 
		if(info->line==0||info->line==1) 
			uart_out(info, UART_TX, info->xmit.buf[info->xmit.tail]); 
		/*else                                                                */ 
			/*uart_out(info, EX_UART_TX, info->xmit.buf[info->xmit.tail]);*/ 
		     
        	info->xmit.tail = (info->xmit.tail + 1) & (SERIAL_XMIT_SIZE-1); 
        	info->state->icount.tx++; 
        	if (info->xmit.head == info->xmit.tail) 
        		break; 
        } while (--count > 0); 
    	restore_flags(flags); 
#else 
		enable_uart_tx_interrupt(info->line); 
#endif 
	} 
	return ret; 
} 
 
static int rs_write_room(struct tty_struct *tty) 
{ 
	struct async_struct *info = (struct async_struct *)tty->driver_data; 
 
	if (serial_paranoia_check(info, tty->device, "rs_write_room")) 
		return 0; 
	return CIRC_SPACE(info->xmit.head, info->xmit.tail, SERIAL_XMIT_SIZE); 
} 
 
static int rs_chars_in_buffer(struct tty_struct *tty) 
{ 
	struct async_struct *info = (struct async_struct *)tty->driver_data; 
				 
	if (serial_paranoia_check(info, tty->device, "rs_chars_in_buffer")) 
		return 0; 
	return CIRC_CNT(info->xmit.head, info->xmit.tail, SERIAL_XMIT_SIZE); 
} 
 
static void rs_flush_buffer(struct tty_struct *tty) 
{ 
	struct async_struct *info = (struct async_struct *)tty->driver_data; 
	unsigned long flags; 
	 
	if (serial_paranoia_check(info, tty->device, "rs_flush_buffer")) 
		return; 
	save_flags(flags); cli(); 
	info->xmit.head = info->xmit.tail = 0; 
	restore_flags(flags); 
	wake_up_interruptible(&tty->write_wait); 
#ifdef SERIAL_HAVE_POLL_WAIT 
	wake_up_interruptible(&tty->poll_wait); 
#endif 
	if ((tty->flags & (1 << TTY_DO_WRITE_WAKEUP)) && 
	    tty->ldisc.write_wakeup) 
		(tty->ldisc.write_wakeup)(tty); 
} 
 
/* 
 * This function is used to send a high-priority XON/XOFF character to 
 * the device 
 */ 
static void rs_send_xchar(struct tty_struct *tty, char ch) 
{ 
	struct async_struct *info = (struct async_struct *)tty->driver_data; 
 
	if (serial_paranoia_check(info, tty->device, "rs_send_char")) 
		return; 
 
	info->x_char = ch; 
	if (ch) { 
		// Make sure transmit interrupts are on 
		enable_uart_tx_interrupt(info->line); 
	} 
} 
 
/* 
 * ------------------------------------------------------------ 
 * rs_throttle() 
 *  
 * This routine is called by the upper-layer tty layer to signal that 
 * incoming characters should be throttled. 
 * ------------------------------------------------------------ 
 */ 
static void rs_throttle(struct tty_struct * tty) 
{ 
	struct async_struct *info = (struct async_struct *)tty->driver_data; 
	unsigned long flags; 
#ifdef SERIAL_DEBUG_THROTTLE 
	char	buf[64]; 
	 
	printk("throttle %s: %d....\n", tty_name(tty, buf), 
	       tty->ldisc.chars_in_buffer(tty)); 
#endif 
 
	if (serial_paranoia_check(info, tty->device, "rs_throttle")) 
		return; 
	 
	if (I_IXOFF(tty)) 
		rs_send_xchar(tty, STOP_CHAR(tty)); 
/* 
	if (tty->termios->c_cflag & CRTSCTS) 
		info->MCR &= ~UART_MCR_RTS; 
*/ 
	save_flags(flags); cli(); 
 
	restore_flags(flags); 
} 
 
static void rs_unthrottle(struct tty_struct * tty) 
{ 
	struct async_struct *info = (struct async_struct *)tty->driver_data; 
	unsigned long flags; 
#ifdef SERIAL_DEBUG_THROTTLE 
	char	buf[64]; 
	 
	printk("unthrottle %s: %d....\n", tty_name(tty, buf), 
	       tty->ldisc.chars_in_buffer(tty)); 
#endif 
 
	if (serial_paranoia_check(info, tty->device, "rs_unthrottle")) 
		return; 
	 
	if (I_IXOFF(tty)) { 
		if (info->x_char) 
			info->x_char = 0; 
		else 
			rs_send_xchar(tty, START_CHAR(tty)); 
	} 
 
	save_flags(flags); cli(); 
 
	restore_flags(flags); 
} 
 
/* 
 * ------------------------------------------------------------ 
 * rs_ioctl() and friends 
 * ------------------------------------------------------------ 
 */ 
static int get_serial_info(struct async_struct * info, 
			   struct serial_struct * retinfo) 
{ 
	struct serial_struct tmp; 
	struct serial_state *state = info->state; 
    
	if (!retinfo) 
		return -EFAULT; 
	memset(&tmp, 0, sizeof(tmp)); 
	tmp.type = state->type; 
	tmp.line = state->line; 
	tmp.port = state->port; 
	if (HIGH_BITS_OFFSET) 
		tmp.port_high = state->port >> HIGH_BITS_OFFSET; 
	else 
		tmp.port_high = 0; 
	tmp.irq = state->irq; 
	tmp.flags = state->flags; 
	tmp.xmit_fifo_size = state->xmit_fifo_size; 
	tmp.baud_base = state->baud_base; 
	tmp.close_delay = state->close_delay; 
	tmp.closing_wait = state->closing_wait; 
	tmp.custom_divisor = state->custom_divisor; 
	tmp.hub6 = state->hub6; 
	tmp.io_type = state->io_type; 
	if (copy_to_user(retinfo,&tmp,sizeof(*retinfo))) 
		return -EFAULT; 
	return 0; 
} 
 
static int set_serial_info(struct async_struct * info, 
			   struct serial_struct * new_info) 
{ 
	struct serial_struct new_serial; 
 	struct serial_state old_state, *state; 
	unsigned int		i,change_irq,change_port; 
	int 			retval = 0; 
	unsigned long		new_port; 
 
	if (copy_from_user(&new_serial,new_info,sizeof(new_serial))) 
		return -EFAULT; 
	state = info->state; 
	old_state = *state; 
 
	new_port = new_serial.port; 
	if (HIGH_BITS_OFFSET) 
		new_port += (unsigned long) new_serial.port_high << HIGH_BITS_OFFSET; 
 
	change_irq = new_serial.irq != state->irq; 
	change_port = (new_port != ((int) state->port)) || 
		(new_serial.hub6 != state->hub6); 
   
	if (!capable(CAP_SYS_ADMIN)) { 
		if (change_irq || change_port || 
		    (new_serial.baud_base != state->baud_base) || 
		    (new_serial.type != state->type) || 
		    (new_serial.close_delay != state->close_delay) || 
		    (new_serial.xmit_fifo_size != state->xmit_fifo_size) || 
		    ((new_serial.flags & ~ASYNC_USR_MASK) != 
		     (state->flags & ~ASYNC_USR_MASK))) 
			return -EPERM; 
		state->flags = ((state->flags & ~ASYNC_USR_MASK) | 
			       (new_serial.flags & ASYNC_USR_MASK)); 
		info->flags = ((info->flags & ~ASYNC_USR_MASK) | 
			       (new_serial.flags & ASYNC_USR_MASK)); 
		state->custom_divisor = new_serial.custom_divisor; 
		goto check_and_exit; 
	} 
 
	new_serial.irq = irq_cannonicalize(new_serial.irq); 
 
	if ((new_serial.irq >= NR_IRQS) || (new_serial.irq < 0) ||  
	    (new_serial.baud_base < 9600)|| (new_serial.type < PORT_UNKNOWN) || 
	    (new_serial.type > PORT_MAX) || (new_serial.type == PORT_CIRRUS) || 
	    (new_serial.type == PORT_STARTECH)) { 
		return -EINVAL; 
	} 
 
	if ((new_serial.type != state->type) || 
	    (new_serial.xmit_fifo_size <= 0)) 
		new_serial.xmit_fifo_size = 
			uart_config[new_serial.type].dfl_xmit_fifo_size; 
 
	// Make sure address is not already in use 
	if (new_serial.type) { 
		for (i = 0 ; i < NR_PORTS; i++) 
			if ((state != &rs_table[i]) && 
			    (rs_table[i].port == new_port) && 
			    rs_table[i].type) 
				return -EADDRINUSE; 
	} 
 
	if ((change_port || change_irq) && (state->count > 1)) 
		return -EBUSY; 
 
	// 
	// OK, past this point, all the error checking has been done. 
	// At this point, we start making changes..... 
	// 
 
	state->baud_base = new_serial.baud_base; 
	state->flags = ((state->flags & ~ASYNC_FLAGS) | 
			(new_serial.flags & ASYNC_FLAGS)); 
	info->flags = ((state->flags & ~ASYNC_INTERNAL_FLAGS) | 
		       (info->flags & ASYNC_INTERNAL_FLAGS)); 
	state->custom_divisor = new_serial.custom_divisor; 
	state->close_delay = new_serial.close_delay * HZ/100; 
	state->closing_wait = new_serial.closing_wait * HZ/100; 
#if (LINUX_VERSION_CODE > 0x20100) 
	info->tty->low_latency = (info->flags & ASYNC_LOW_LATENCY) ? 1 : 0; 
#endif 
	info->xmit_fifo_size = state->xmit_fifo_size = 
		new_serial.xmit_fifo_size; 
 
	if ((state->type != PORT_UNKNOWN) && state->port) { 
		release_region(state->port,8); 
	} 
	state->type = new_serial.type; 
	if (change_port || change_irq) { 
		// 
		// We need to shutdown the serial port at the old 
		// port/irq combination. 
		// 
		shutdown(info); 
		state->irq = new_serial.irq; 
		info->port = state->port = new_port; 
		info->hub6 = state->hub6 = new_serial.hub6; 
		if (info->hub6) 
			info->io_type = state->io_type = SERIAL_IO_HUB6; 
		else if (info->io_type == SERIAL_IO_HUB6) 
			info->io_type = state->io_type = SERIAL_IO_PORT; 
	} 
	if ((state->type != PORT_UNKNOWN) && state->port) { 
			request_region(state->port,8,"serial(set)"); 
	} 
 
	 
check_and_exit: 
	if (!state->port || !state->type) 
		return 0; 
	if (info->flags & ASYNC_INITIALIZED) { 
		if (((old_state.flags & ASYNC_SPD_MASK) != 
		     (state->flags & ASYNC_SPD_MASK)) || 
		    (old_state.custom_divisor != state->custom_divisor)) { 
#if (LINUX_VERSION_CODE >= 131394) // Linux 2.1.66 
			if ((state->flags & ASYNC_SPD_MASK) == ASYNC_SPD_HI) 
				info->tty->alt_speed = 57600; 
			if ((state->flags & ASYNC_SPD_MASK) == ASYNC_SPD_VHI) 
				info->tty->alt_speed = 115200; 
			if ((state->flags & ASYNC_SPD_MASK) == ASYNC_SPD_SHI) 
				info->tty->alt_speed = 230400; 
			if ((state->flags & ASYNC_SPD_MASK) == ASYNC_SPD_WARP) 
				info->tty->alt_speed = 460800; 
#endif 
			change_speed(info, 0); 
		} 
	} else 
		retval = startup(info); 
	return retval; 
} 
 
 
/* 
 * get_lsr_info - get line status register info 
 * 
 * Purpose: Let user call ioctl() to get info when the UART physically 
 * 	    is emptied.  On bus types like RS485, the transmitter must 
 * 	    release the bus after transmitting. This must be done when 
 * 	    the transmit shift register is empty, not be done when the 
 * 	    transmit holding register is empty.  This functionality 
 * 	    allows an RS485 driver to be written in user space.  
 */ 
static int get_lsr_info(struct async_struct * info, unsigned int *value) 
{ 
	unsigned char status; 
	unsigned int result; 
	unsigned long flags; 
 
	save_flags(flags); cli(); 
	if(info->line==0||info->line==1) 
	    status = serial_in(info, UART_LSR); 
	/*else                                      */ 
	    /*status = serial_in(info, EX_UART_LSR);*/ 
	     
	restore_flags(flags); 
	result = ((status & UART_LSR_TEMT) ? TIOCSER_TEMT : 0); 
 
	// 
	// If we're about to load something into the transmit 
	// register, we'll pretend the transmitter isn't empty to 
	// avoid a race condition (depending on when the transmit 
	// interrupt happens). 
	// 
	if (info->x_char ||  
	    ((CIRC_CNT(info->xmit.head, info->xmit.tail, 
		       SERIAL_XMIT_SIZE) > 0) && 
	     !info->tty->stopped && !info->tty->hw_stopped)) 
		result &= TIOCSER_TEMT; 
 
	if (copy_to_user(value, &result, sizeof(int))) 
		return -EFAULT; 
	return 0; 
} 
 
 
static int do_autoconfig(struct async_struct * info) 
{ 
	int irq, retval; 
	 
	if (!capable(CAP_SYS_ADMIN)) 
		return -EPERM; 
	 
	if (info->state->count > 1) 
		return -EBUSY; 
	 
	shutdown(info); 
 
	autoconfig(info->state); 
	if ((info->state->flags & ASYNC_AUTO_IRQ) && 
	    (info->state->port != 0) && 
	    (info->state->type != PORT_UNKNOWN)) { 
		irq = detect_uart_irq(info->state); 
		if (irq > 0) 
			info->state->irq = irq; 
	} 
 
	retval = startup(info); 
	if (retval) 
		return retval; 
	return 0; 
} 
 
/* 
 * rs_break() --- routine which turns the break handling on or off 
 */ 
#if (LINUX_VERSION_CODE < 131394) /* Linux 2.1.66 */ 
 
#else 
static void rs_break(struct tty_struct *tty, int break_state) 
{ 
	struct async_struct * info = (struct async_struct *)tty->driver_data; 
	unsigned long flags, tmp; 
	 
	if (serial_paranoia_check(info, tty->device, "rs_break")) 
		return; 
 
	if (!CONFIGURED_SERIAL_PORT(info)) 
		return; 
	save_flags(flags); cli(); 
/* 
	if (break_state == -1) 
		info->LCR |= UART_LCR_SBC; 
	else 
		info->LCR &= ~UART_LCR_SBC; 
	serial_out(info, UART_LCR, info->LCR); 
*/ 
	if(info->line==0||info->line==1) 
		tmp = serial_in(info, UART_GCR); 
	/*else                                      */ 
		/*tmp = serial_in(info,EX_UART_GCR);*/ 
	if (break_state == -1) 
		tmp |= UART_LCR_SBC; 
	else 
		tmp &= ~UART_LCR_SBC; 
	if(info->line==0||info->line==1) 
		serial_out(info, UART_GCR, tmp); 
	restore_flags(flags); 
} 
#endif 
 
 
static int rs_ioctl(struct tty_struct *tty, struct file * file, 
		    unsigned int cmd, unsigned long arg) 
{ 
	struct async_struct * info = (struct async_struct *)tty->driver_data; 
	struct async_icount cprev, cnow;	// kernel counter temps 
	struct serial_icounter_struct icount; 
	unsigned long flags; 
#if (LINUX_VERSION_CODE < 131394) // Linux 2.1.66 
	int retval, tmp; 
#endif 
	 
	if (serial_paranoia_check(info, tty->device, "rs_ioctl")) 
		return -ENODEV; 
 
	if ((cmd != TIOCGSERIAL) && (cmd != TIOCSSERIAL) && 
	    (cmd != TIOCSERCONFIG) && (cmd != TIOCSERGSTRUCT) && 
	    (cmd != TIOCMIWAIT) && (cmd != TIOCGICOUNT)) { 
		if (tty->flags & (1 << TTY_IO_ERROR)) 
		    return -EIO; 
	} 
	 
	switch (cmd) { 
#if (LINUX_VERSION_CODE < 131394) // Linux 2.1.66 
		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 (signal_pending(current)) 
				return -EINTR; 
			if (!arg) { 
				send_break(info, HZ/4);	// 1/4 second 
				if (signal_pending(current)) 
					return -EINTR; 
			} 
			return 0; 
		case TCSBRKP:	// support for POSIX tcsendbreak() 
			retval = tty_check_change(tty); 
			if (retval) 
				return retval; 
			tty_wait_until_sent(tty, 0); 
			if (signal_pending(current)) 
				return -EINTR; 
			send_break(info, arg ? arg*(HZ/10) : HZ/4); 
			if (signal_pending(current)) 
				return -EINTR; 
			return 0; 
		case TIOCGSOFTCAR: 
			tmp = C_CLOCAL(tty) ? 1 : 0; 
			if (copy_to_user((void *)arg, &tmp, sizeof(int))) 
				return -EFAULT; 
			return 0; 
		case TIOCSSOFTCAR: 
			if (copy_from_user(&tmp, (void *)arg, sizeof(int))) 
				return -EFAULT; 
 
			tty->termios->c_cflag = 
				((tty->termios->c_cflag & ~CLOCAL) | 
				 (tmp ? CLOCAL : 0)); 
			return 0; 
#endif 
 
		case TIOCGSERIAL: 
			return get_serial_info(info, 
					       (struct serial_struct *) arg); 
		case TIOCSSERIAL: 
			return set_serial_info(info, 
					       (struct serial_struct *) arg); 
		case TIOCSERCONFIG: 
			return do_autoconfig(info); 
 
		case TIOCSERGETLSR: // Get line status register 
			return get_lsr_info(info, (unsigned int *) arg); 
 
		case TIOCSERGSTRUCT: 
			if (copy_to_user((struct async_struct *) arg, 
					 info, sizeof(struct async_struct))) 
				return -EFAULT; 
			return 0; 
				 
			 
		// 
		// Wait for any of the 4 modem inputs (DCD,RI,DSR,CTS) to change 
		// - mask passed in arg for lines of interest 
 		//   (use |'ed TIOCM_RNG/DSR/CD/CTS for masking) 
		// Caller should use TIOCGICOUNT to see which one it was 
		// 
		case TIOCMIWAIT: 
			save_flags(flags); cli(); 
			// note the counters on entry 
			cprev = info->state->icount; 
			restore_flags(flags); 
			// Force modem status interrupts on 
			while (1) { 
				interruptible_sleep_on(&info->delta_msr_wait); 
				// see if a signal did it 
				if (signal_pending(current)) 
					return -ERESTARTSYS; 
				save_flags(flags); cli(); 
				cnow = info->state->icount; // atomic copy 
				restore_flags(flags); 
				if (cnow.rng == cprev.rng && cnow.dsr == cprev.dsr &&  
				    cnow.dcd == cprev.dcd && cnow.cts == cprev.cts) 
					return -EIO; // no change => error 
				if ( ((arg & TIOCM_RNG) && (cnow.rng != cprev.rng)) || 
				     ((arg & TIOCM_DSR) && (cnow.dsr != cprev.dsr)) || 
				     ((arg & TIOCM_CD)  && (cnow.dcd != cprev.dcd)) || 
				     ((arg & TIOCM_CTS) && (cnow.cts != cprev.cts)) ) { 
					return 0; 
				} 
				cprev = cnow; 
			} 
			// NOTREACHED 
 
		//  
		// Get counter of input serial line interrupts (DCD,RI,DSR,CTS) 
		// Return: write counters to the user passed counter struct 
		// NB: both 1->0 and 0->1 transitions are counted except for 
		//     RI where only 0->1 is counted. 
		// 
		case TIOCGICOUNT: 
			save_flags(flags); cli(); 
			cnow = info->state->icount; 
			restore_flags(flags); 
			icount.cts = cnow.cts; 
			icount.dsr = cnow.dsr; 
			icount.rng = cnow.rng; 
			icount.dcd = cnow.dcd; 
			icount.rx = cnow.rx; 
			icount.tx = cnow.tx; 
			icount.frame = cnow.frame; 
			icount.overrun = cnow.overrun; 
			icount.parity = cnow.parity; 
			icount.brk = cnow.brk; 
			icount.buf_overrun = cnow.buf_overrun; 
			 
			if (copy_to_user((void *)arg, &icount, sizeof(icount))) 
				return -EFAULT; 
			return 0; 
		case TIOCSERGWILD: 
		case TIOCSERSWILD: 
			// "setserial -W" is called in Debian boot 
			printk ("TIOCSER?WILD ioctl obsolete, ignored.\n"); 
			return 0; 
 
		default: 
			return -ENOIOCTLCMD; 
		} 
	return 0; 
} 
 
static void rs_set_termios(struct tty_struct *tty, struct termios *old_termios) 
{ 
	struct async_struct *info = (struct async_struct *)tty->driver_data; 
//	unsigned long flags; 
	unsigned int cflag = tty->termios->c_cflag; 
	 
	if (   (cflag == old_termios->c_cflag) 
	    && (   RELEVANT_IFLAG(tty->termios->c_iflag)  
		== RELEVANT_IFLAG(old_termios->c_iflag))) 
	  return; 
	change_speed(info, old_termios); 
 
	// Handle turning off CRTSCTS 
	if ((old_termios->c_cflag & CRTSCTS) && 
	    !(tty->termios->c_cflag & CRTSCTS)) { 
		tty->hw_stopped = 0; 
		rs_start(tty); 
	} 
} 
 
/* 
 * ------------------------------------------------------------ 
 * rs_close() 
 *  
 * This routine is called when the serial port gets closed.  First, we 
 * wait for the last remaining data to be sent.  Then, we unlink its 
 * async structure from the interrupt chain if necessary, and we free 
 * that IRQ if nothing is left in the chain. 
 * ------------------------------------------------------------ 
 */ 
static void rs_close(struct tty_struct *tty, struct file * filp) 
{ 
	struct async_struct * info = (struct async_struct *)tty->driver_data; 
	struct serial_state *state; 
	unsigned long flags; 
 
	if (!info || serial_paranoia_check(info, tty->device, "rs_close")) 
		return; 
 
	state = info->state; 
	 
	save_flags(flags); cli(); 
	 
	if (tty_hung_up_p(filp)) { 
		DBG_CNT("before DEC-hung"); 
		MOD_DEC_USE_COUNT; 
		restore_flags(flags); 
		return; 
	} 
	 
#ifdef SERIAL_DEBUG_OPEN 
	printk("rs_close ttys%d, count = %d\n", info->line, state->count); 
#endif 
	if ((tty->count == 1) && (state->count != 1)) { 
		// 
		// Uh, oh.  tty->count is 1, which means that the tty 
		// structure will be freed.  state->count should always 
		// be one in these conditions.  If it's greater than 
		// one, we've got real problems, since it means the 
		// serial port won't be shutdown. 
		// 
		printk("rs_close: bad serial port count; tty->count is 1, " 
		       "state->count is %d\n", state->count); 
		state->count = 1; 
	} 
	if (--state->count < 0) { 
		printk("rs_close: bad serial port count for ttys%d: %d\n", 
		       info->line, state->count); 
		state->count = 0; 
	} 
	if (state->count) { 
		DBG_CNT("before DEC-2"); 
		MOD_DEC_USE_COUNT; 
		restore_flags(flags); 
		return; 
	} 
	info->flags |= ASYNC_CLOSING; 
	restore_flags(flags); 
	// 
	// Save the termios structure, since this port may have 
	// separate termios for callout and dialin. 
	// 
	if (info->flags & ASYNC_NORMAL_ACTIVE) 
		info->state->normal_termios = *tty->termios; 
	if (info->flags & ASYNC_CALLOUT_ACTIVE) 
		info->state->callout_termios = *tty->termios; 
	// 
	// 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 (info->closing_wait != ASYNC_CLOSING_WAIT_NONE) 
		tty_wait_until_sent(tty, info->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. 
	// 
	//info->IER &= ~UART_IER_RLSI; 
	info->read_status_mask &= ~UART_LSR_DR; 
	if (info->flags & ASYNC_INITIALIZED) { 
		//serial_out(info, UART_IER, info->IER); 
		enable_uart_rx_interrupt(info->line); 
		// 
		// Before we drop DTR, make sure the UART transmitter 
		// has completely drained; this is especially 
		// important if there is a transmit FIFO! 
		// 
		rs_wait_until_sent(tty, info->timeout); 
	} 
	shutdown(info); 
	if (tty->driver.flush_buffer) 
		tty->driver.flush_buffer(tty); 
	if (tty->ldisc.flush_buffer) 
		tty->ldisc.flush_buffer(tty); 
	tty->closing = 0; 
	info->event = 0; 
	info->tty = 0; 
	if (info->blocked_open) { 
		if (info->close_delay) { 
			set_current_state(TASK_INTERRUPTIBLE); 
			schedule_timeout(info->close_delay); 
		} 
		wake_up_interruptible(&info->open_wait); 
	} 
	info->flags &= ~(ASYNC_NORMAL_ACTIVE|ASYNC_CALLOUT_ACTIVE| 
			 ASYNC_CLOSING); 
	wake_up_interruptible(&info->close_wait); 
	MOD_DEC_USE_COUNT; 
} 
 
/* 
 * rs_wait_until_sent() --- wait until the transmitter is empty 
 */ 
static void rs_wait_until_sent(struct tty_struct *tty, int timeout) 
{ 
	struct async_struct * info = (struct async_struct *)tty->driver_data; 
	unsigned long orig_jiffies, char_time; 
	int tmp; 
	int lsr; 
	 
	if (serial_paranoia_check(info, tty->device, "rs_wait_until_sent")) 
		return; 
 
	if (info->state->type == PORT_UNKNOWN) 
		return; 
 
	if (info->xmit_fifo_size == 0) 
		return; // Just in case....  
 
	orig_jiffies = jiffies; 
	// 
	// Set the check interval to be 1/5 of the estimated time to 
	// send a single character, and make it at least 1.  The check 
	// interval should also be less than the timeout. 
	//  
	// Note: we have to use pretty tight timings here to satisfy 
	// the NIST-PCTS. 
	// 
	char_time = (info->timeout - HZ/50) / info->xmit_fifo_size; 
	char_time = char_time / 5; 
	if (char_time == 0) 
		char_time = 1; 
	if (timeout && timeout < char_time) 
		char_time = timeout; 
	// 
	// If the transmitter hasn't cleared in twice the approximate 
	// amount of time to send the entire FIFO, it probably won't 
	// ever clear.  This assumes the UART isn't doing flow 
	// control, which is currently the case.  Hence, if it ever 
	// takes longer than info->timeout, this is probably due to a 
	// UART bug of some kind.  So, we clamp the timeout parameter at 
	// 2*info->timeout. 
	// 
	if (!timeout || timeout > 2*info->timeout) 
		timeout = 2*info->timeout; 
#ifdef SERIAL_DEBUG_RS_WAIT_UNTIL_SENT 
	printk("In rs_wait_until_sent(%d) check=%lu...", timeout, char_time); 
	printk("jiff=%lu...", jiffies); 
#endif 
	if(info->line==0||info->line==1) 
	{ 
		tmp=UART_LSR; 
	} 
	/*else                    */ 
	/*{                       */ 
		/*tmp=EX_UART_LSR;*/ 
	/*}                       */ 
	while (!((lsr = serial_inp(info, tmp)) & UART_LSR_TEMT)) { 
#ifdef SERIAL_DEBUG_RS_WAIT_UNTIL_SENT 
		printk("lsr = %d (jiff=%lu)...", lsr, jiffies); 
#endif 
		set_current_state(TASK_INTERRUPTIBLE); 
		schedule_timeout(char_time); 
		if (signal_pending(current)) 
			break; 
		if (timeout && time_after(jiffies, orig_jiffies + timeout)) 
			break; 
	} 
#ifdef SERIAL_DEBUG_RS_WAIT_UNTIL_SENT 
	printk("lsr = %d (jiff=%lu)...done\n", lsr, jiffies); 
#endif 
} 
 
/* 
 * rs_hangup() --- called by tty_hangup() when a hangup is signaled. 
 */ 
static void rs_hangup(struct tty_struct *tty) 
{ 
	struct async_struct * info = (struct async_struct *)tty->driver_data; 
	struct serial_state *state = info->state; 
	 
	if (serial_paranoia_check(info, tty->device, "rs_hangup")) 
		return; 
 
	state = info->state; 
	 
	rs_flush_buffer(tty); 
	if (info->flags & ASYNC_CLOSING) 
		return; 
	shutdown(info); 
	info->event = 0; 
	state->count = 0; 
	info->flags &= ~(ASYNC_NORMAL_ACTIVE|ASYNC_CALLOUT_ACTIVE); 
	info->tty = 0; 
	wake_up_interruptible(&info->open_wait); 
} 
 
/* 
 * ------------------------------------------------------------ 
 * rs_open() and friends 
 * ------------------------------------------------------------ 
 */ 
static int block_til_ready(struct tty_struct *tty, struct file * filp, 
			   struct async_struct *info) 
{ 
	DECLARE_WAITQUEUE(wait, current); 
	struct serial_state *state = info->state; 
	int		retval; 
	int		do_clocal = 0, extra_count = 0; 
	unsigned long	flags; 
 
	// 
	// 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) || 
	    (info->flags & ASYNC_CLOSING)) { 
		if (info->flags & ASYNC_CLOSING) 
			interruptible_sleep_on(&info->close_wait); 
#ifdef SERIAL_DO_RESTART 
		return ((info->flags & ASYNC_HUP_NOTIFY) ? 
			-EAGAIN : -ERESTARTSYS); 
#else 
		return -EAGAIN; 
#endif 
	} 
 
	// 
	// If this is a callout device, then just make sure the normal 
	// device isn't being used. 
	// 
	if (tty->driver.subtype == SERIAL_TYPE_CALLOUT) { 
		if (info->flags & ASYNC_NORMAL_ACTIVE) 
			return -EBUSY; 
		if ((info->flags & ASYNC_CALLOUT_ACTIVE) && 
		    (info->flags & ASYNC_SESSION_LOCKOUT) && 
		    (info->session != current->session)) 
		    return -EBUSY; 
		if ((info->flags & ASYNC_CALLOUT_ACTIVE) && 
		    (info->flags & ASYNC_PGRP_LOCKOUT) && 
		    (info->pgrp != current->pgrp)) 
		    return -EBUSY; 
		info->flags |= ASYNC_CALLOUT_ACTIVE; 
		return 0; 
	} 
	 
	// 
	// 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))) { 
		if (info->flags & ASYNC_CALLOUT_ACTIVE) 
			return -EBUSY; 
		info->flags |= ASYNC_NORMAL_ACTIVE; 
		return 0; 
	} 
 
	if (info->flags & ASYNC_CALLOUT_ACTIVE) { 
		if (state->normal_termios.c_cflag & CLOCAL) 
			do_clocal = 1; 
	} else { 
		if (tty->termios->c_cflag & CLOCAL) 
			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, state->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(&info->open_wait, &wait); 
#ifdef SERIAL_DEBUG_OPEN 
	printk("block_til_ready before block: ttys%d, count = %d\n", 
	       state->line, state->count); 
#endif 
	save_flags(flags); cli(); 
	if (!tty_hung_up_p(filp)) { 
		extra_count = 1; 
		state->count--; 
	} 
	restore_flags(flags); 
	info->blocked_open++; 
	while (1) { 
		save_flags(flags); cli(); 
/* 
		if (!(info->flags & ASYNC_CALLOUT_ACTIVE) && 
		    (tty->termios->c_cflag & CBAUD)) 
			serial_out(info, UART_MCR, 
				   serial_inp(info, UART_MCR) | 
				   (UART_MCR_DTR | UART_MCR_RTS)); 
*/ 
		restore_flags(flags); 
		set_current_state(TASK_INTERRUPTIBLE); 
		if (tty_hung_up_p(filp) || 
		    !(info->flags & ASYNC_INITIALIZED)) { 
#ifdef SERIAL_DO_RESTART 
			if (info->flags & ASYNC_HUP_NOTIFY) 
				retval = -EAGAIN; 
			else 
				retval = -ERESTARTSYS;	 
#else 
			retval = -EAGAIN; 
#endif 
			break; 
		} 
/* 
		if (!(info->flags & ASYNC_CALLOUT_ACTIVE) && 
		    !(info->flags & ASYNC_CLOSING) && 
		    (do_clocal || (serial_in(info, UART_MSR) & 
				   UART_MSR_DCD))) 
			break; 
*/ 
		break; 
		if (signal_pending(current)) { 
			retval = -ERESTARTSYS; 
			break; 
		} 
#ifdef SERIAL_DEBUG_OPEN 
		printk("block_til_ready blocking: ttys%d, count = %d\n", 
		       info->line, state->count); 
#endif 
		schedule(); 
	} 
	set_current_state(TASK_RUNNING); 
	remove_wait_queue(&info->open_wait, &wait); 
	if (extra_count) 
		state->count++; 
	info->blocked_open--; 
#ifdef SERIAL_DEBUG_OPEN 
	printk("block_til_ready after blocking: ttys%d, count = %d\n", 
	       info->line, state->count); 
#endif 
	if (retval) 
		return retval; 
	info->flags |= ASYNC_NORMAL_ACTIVE; 
	return 0; 
} 
 
static int get_async_struct(int line, struct async_struct **ret_info) 
{ 
	struct async_struct *info; 
	struct serial_state *sstate; 
 
	sstate = rs_table + line; 
	sstate->count++; 
	if (sstate->info) { 
		*ret_info = sstate->info; 
		return 0; 
	} 
	info = kmalloc(sizeof(struct async_struct), GFP_KERNEL); 
	if (!info) { 
		sstate->count--; 
		return -ENOMEM; 
	} 
	memset(info, 0, sizeof(struct async_struct)); 
	init_waitqueue_head(&info->open_wait); 
	init_waitqueue_head(&info->close_wait); 
	init_waitqueue_head(&info->delta_msr_wait); 
	info->magic = SERIAL_MAGIC; 
	info->port = sstate->port; 
	info->flags = sstate->flags; 
	info->io_type = sstate->io_type; 
	info->iomem_base = sstate->iomem_base; 
	info->iomem_reg_shift = sstate->iomem_reg_shift; 
	info->xmit_fifo_size = sstate->xmit_fifo_size; 
	info->line = line; 
	info->tqueue.routine = do_softint; 
	info->tqueue.data = info; 
	info->state = sstate; 
	if (sstate->info) { 
		kfree(info); 
		*ret_info = sstate->info; 
		return 0; 
	} 
	*ret_info = sstate->info = info; 
	return 0; 
} 
 
/* 
 * This routine is called whenever a serial port is opened.  It 
 * enables interrupts for a serial port, linking in its async structure into 
 * the IRQ chain.   It also performs the serial-specific 
 * initialization for the tty structure. 
 */ 
static int rs_open(struct tty_struct *tty, struct file * filp) 
{ 
	struct async_struct	*info; 
	int 			retval, line; 
	unsigned long		page; 
 
	MOD_INC_USE_COUNT; 
	line = MINOR(tty->device) - tty->driver.minor_start; 
	if ((line < 0) || (line >= NR_PORTS)) { 
		MOD_DEC_USE_COUNT; 
		return -ENODEV; 
	} 
	retval = get_async_struct(line, &info); 
	if (retval) { 
		MOD_DEC_USE_COUNT; 
		return retval; 
	} 
	tty->driver_data = info; 
	info->tty = tty; 
	if (serial_paranoia_check(info, tty->device, "rs_open")) { 
		MOD_DEC_USE_COUNT;		 
		return -ENODEV; 
	} 
 
#ifdef SERIAL_DEBUG_OPEN 
	printk("rs_open %s%d, count = %d\n", tty->driver.name, info->line, 
	       info->state->count); 
#endif 
#if (LINUX_VERSION_CODE > 0x20100) 
	info->tty->low_latency = (info->flags & ASYNC_LOW_LATENCY) ? 1 : 0; 
#endif 
 
	// 
	//	This relies on lock_kernel() stuff so wants tidying for 2.5 
	// 
	if (!tmp_buf) { 
		page = get_zeroed_page(GFP_KERNEL); 
		if (!page) { 
			MOD_DEC_USE_COUNT; 
			return -ENOMEM; 
		} 
		if (tmp_buf) 
			free_page(page); 
		else 
			tmp_buf = (unsigned char *) page; 
	} 
 
	// 
	// If the port is the middle of closing, bail out now 
	// 
	if (tty_hung_up_p(filp) || 
	    (info->flags & ASYNC_CLOSING)) { 
		if (info->flags & ASYNC_CLOSING) 
			interruptible_sleep_on(&info->close_wait); 
		MOD_DEC_USE_COUNT; 
#ifdef SERIAL_DO_RESTART 
		return ((info->flags & ASYNC_HUP_NOTIFY) ? 
			-EAGAIN : -ERESTARTSYS); 
#else 
		return -EAGAIN; 
#endif 
	} 
 
	// 
	// Start up serial port 
	// 
	retval = startup(info); 
	if (retval) { 
		MOD_DEC_USE_COUNT; 
		return retval; 
	} 
 
	retval = block_til_ready(tty, filp, info); 
	if (retval) { 
#ifdef SERIAL_DEBUG_OPEN 
		printk("rs_open returning after block_til_ready with %d\n", 
		       retval); 
#endif 
		MOD_DEC_USE_COUNT; 
		return retval; 
	} 
 
	if ((info->state->count == 1) && 
	    (info->flags & ASYNC_SPLIT_TERMIOS)) { 
		if (tty->driver.subtype == SERIAL_TYPE_NORMAL) 
			*tty->termios = info->state->normal_termios; 
		else  
			*tty->termios = info->state->callout_termios; 
		change_speed(info, 0); 
	} 
#ifdef CONFIG_SERIAL_SAMSUNG_CONSOLE 
	if (sercons.cflag && sercons.index == line) { 
		tty->termios->c_cflag = sercons.cflag; 
		sercons.cflag = 0; 
		change_speed(info, 0); 
	} 
#endif 
	info->session = current->session; 
	info->pgrp = current->pgrp; 
 
#ifdef SERIAL_DEBUG_OPEN 
	printk("rs_open ttys%d successful...", info->line); 
#endif 
	return 0; 
} 
 
/* 
 * /proc fs routines.... 
 */ 
static _INLINE_ int line_info(char *buf, struct serial_state *state) 
{ 
	struct async_struct *info = state->info, scr_info; 
//	char	stat_buf[30], control, status; 
	char	stat_buf[30]; 
	int	ret; 
	//unsigned long flags; 
 
	ret = sprintf(buf, "%d: uart:%s port:%lX irq:%d", 
		      state->line, uart_config[state->type].name,  
		      state->port, state->irq); 
 
	if (!state->port || (state->type == PORT_UNKNOWN)) { 
		ret += sprintf(buf+ret, "\n"); 
		return ret; 
	} 
 
	// 
	// Figure out the current RS-232 lines 
	// 
	if (!info) { 
		info = &scr_info;	// This is just for serial_{in,out} 
 
		info->magic = SERIAL_MAGIC; 
		info->port = state->port; 
		info->flags = state->flags; 
		info->hub6 = state->hub6; 
		info->io_type = state->io_type; 
		info->iomem_base = state->iomem_base; 
		info->iomem_reg_shift = state->iomem_reg_shift; 
		info->quot = 0; 
		info->tty = 0; 
	} 
	stat_buf[0] = 0; 
	stat_buf[1] = 0; 
	if (info->quot) { 
		ret += sprintf(buf+ret, " baud:%d", 
			       state->baud_base / info->quot); 
	} 
 
	ret += sprintf(buf+ret, " tx:%d rx:%d", 
		      state->icount.tx, state->icount.rx); 
 
	if (state->icount.frame) 
		ret += sprintf(buf+ret, " fe:%d", state->icount.frame); 
	 
	if (state->icount.parity) 
		ret += sprintf(buf+ret, " pe:%d", state->icount.parity); 
	 
	if (state->icount.brk) 
		ret += sprintf(buf+ret, " brk:%d", state->icount.brk);	 
 
	if (state->icount.overrun) 
		ret += sprintf(buf+ret, " oe:%d", state->icount.overrun); 
 
	// 
	// Last thing is the RS-232 status lines 
	// 
	ret += sprintf(buf+ret, " %s\n", stat_buf+1); 
	return ret; 
} 
 
int rs_read_proc(char *page, char **start, off_t off, int count, 
		 int *eof, void *data) 
{ 
	int i, len = 0, l; 
	off_t	begin = 0; 
 
	len += sprintf(page, "serinfo:1.0 driver:%s%s revision:%s\n", 
		       serial_version, LOCAL_VERSTRING, serial_revdate); 
	for (i = 0; i < NR_PORTS && len < 4000; i++) { 
		l = line_info(page + len, &rs_table[i]); 
		len += l; 
		if (len+begin > off+count) 
			goto done; 
		if (len+begin < off) { 
			begin += len; 
			len = 0; 
		} 
	} 
	*eof = 1; 
done: 
	if (off >= len+begin) 
		return 0; 
	*start = page + (off-begin); 
	return ((count < begin+len-off) ? count : begin+len-off); 
} 
 
 
/* 
 * This routine prints out the appropriate serial driver version 
 * number, and identifies which options were configured into this 
 * driver. 
 */ 
static char serial_options[] __initdata = 
#ifdef CONFIG_SERIAL_DETECT_IRQ 
       " DETECT_IRQ" 
#define SERIAL_OPT 
#endif 
#ifdef SERIAL_OPT 
       " enabled\n"; 
#else 
       " no serial options enabled\n"; 
#endif 
#undef SERIAL_OPT 
 
static _INLINE_ void show_serial_version(void) 
{ 
 	printk(KERN_INFO "%s version %s%s (%s) with%s", serial_name, 
	       serial_version, LOCAL_VERSTRING, serial_revdate, 
	       serial_options); 
} 
 
/* 
 * This routine detect the IRQ of a serial port by clearing OUT2 when 
 * no UART interrupt are requested (IER = 0) (*GPL*). This seems to work at 
 * each time, as long as no other device permanently request the IRQ. 
 * If no IRQ is detected, or multiple IRQ appear, this function returns 0. 
 * The variable "state" and the field "state->port" should not be null. 
 */ 
static unsigned detect_uart_irq (struct serial_state * state) 
{ 
	return 0; 
} 
 
/* 
 * This routine is called by rs_init() to initialize a specific serial 
 * port.  It determines what type of UART chip this serial port is 
 * using: 8250, 16450, 16550, 16550A.  The important question is 
 * whether or not this UART is a 16550A or not, since this will 
 * determine whether or not we can use its FIFO features or not. 
 */ 
static void autoconfig(struct serial_state * state) 
{ 
	struct async_struct *info, scr_info; 
	unsigned long flags; 
 
	state->type = PORT_UNKNOWN; 
 
#ifdef SERIAL_DEBUG_AUTOCONF 
	printk("Testing ttyS%d (0x%04lx, 0x%04x)...\n", state->line, 
	       state->port, (unsigned) state->iomem_base); 
#endif 
	 
	if (!CONFIGURED_SERIAL_PORT(state)) 
		return; 
		 
	info = &scr_info;	// This is just for serial_{in,out} 
 
	info->magic = SERIAL_MAGIC; 
	info->state = state; 
	info->line = state->line; // Qinwei 02.09.17 
	info->port = state->port; 
	info->flags = state->flags; 
	info->io_type = state->io_type; 
	info->iomem_base = state->iomem_base; 
	info->iomem_reg_shift = state->iomem_reg_shift; 
 
	save_flags(flags); cli(); 
	 
	state->type = PORT_S3C4510B; 
	state->xmit_fifo_size =	uart_config[state->type].dfl_xmit_fifo_size; 
 
	if (state->type == PORT_UNKNOWN) { 
		restore_flags(flags); 
		return; 
	} 
 
	if (info->port) { 
 
			request_region(info->port,8,"serial(auto)"); 
	} 
 
	// 
	// Reset the UART. 
	// 
 
#ifdef CONFIG_SERIAL_SAMSUNG_IRDA 
	if (info->line == 1){ 
		serial_outp(info, UART_LCR, UART_LCR_WLEN8|UART_LCR_IRDA); 
        CSR_WRITE(rPCONC,((3 << 20)| (3 << 22)|(3 << 24)| (3 << 26) | (3<<28)|(3 <<30))); // Enable UART1's Pin 
        CSR_WRITE(rPUPC,CSR_READ(rPUPC)|0xfc00); // set pull up 
        CSR_WRITE(rPCONE,CSR_READ(rPCONE)|1);    // Enable UART0's Pin 
		CSR_WRITE(rPDATE,CSR_READ(rPDATE)&0xFE); // select enable IRDA 
	} 
	else 
		serial_outp(info, UART_LCR, UART_LCR_WLEN8); // UART0 don't support IRDA mode 
#else 
	if(info->line==0||info->line==1) 
	{ 
		serial_outp(info, UART_LCR, UART_LCR_WLEN8); 
	} 
	/*else                                                   */ 
	/*{                                                      */ 
		/*serial_outp(info, EX_UART_LCR, UART_LCR_WLEN8);*/ 
	/*}                                                      */ 
#endif 
	 
	if(info->line==0||info->line==1) 
	{ 
		serial_outp(info, UART_GCR, UART_GCR_RX_INT|UART_GCR_TX_INT); 
	} 
	/*else                                                                          */ 
	/*{                                                                             */ 
		/*serial_outp(info, EX_UART_GCR, EX_UART_GCR_RX_INT|EX_UART_GCR_TX_INT);*/ 
	/*}                                                                             */ 
 
	if(info->line==0||info->line==1) 
		(void)uart_in(info, UART_RX); 
	/*else                                    */ 
		/*(void)uart_in(info, EX_UART_RX);*/ 
 
	 
	restore_flags(flags); 
} 
 
 
int register_serial(struct serial_struct *req); 
void unregister_serial(int line); 
 
#if (LINUX_VERSION_CODE > 0x20100) 
EXPORT_SYMBOL(register_serial); 
EXPORT_SYMBOL(unregister_serial); 
#else 
static struct symbol_table serial_syms = { 
#include  
	X(register_serial), 
	X(unregister_serial), 
#include  
}; 
#endif 
 
/* 
 * The serial driver boot-time initialization code! 
 */ 
static int __init rs_init(void) 
{ 
	int i; 
	struct serial_state * state; 
 
	init_bh(SERIAL_BH, do_serial_bh); 
	init_timer(&serial_timer); 
	serial_timer.function = rs_timer; 
	mod_timer(&serial_timer, jiffies + RS_STROBE_TIME); 
 
	for (i = 0; i < NR_IRQS; i++) { 
		IRQ_ports[i] = 0; 
		IRQ_timeout[i] = 0; 
	} 
#ifdef CONFIG_SERIAL_SAMSUNG_CONSOLE 
	// 
	//	The interrupt of the serial console port 
	//	can't be shared. 
	// 
	if (sercons.flags & CON_CONSDEV) { 
		for(i = 0; i < NR_PORTS; i++) 
			if (i != sercons.index && 
			    rs_table[i].irq == rs_table[sercons.index].irq) 
				rs_table[i].irq = 0; 
	} 
#endif 
	show_serial_version(); 
 
	// Initialize the tty_driver structure 
	 
	memset(&serial_driver, 0, sizeof(struct tty_driver)); 
	/*serial_driver.magic = TTY_DRIVE_MAGIC;*/ 
	serial_driver.magic = SERIAL_MAGIC; 
#if (LINUX_VERSION_CODE > 0x20100) 
	serial_driver.driver_name = "serial"; 
#endif 
#if (LINUX_VERSION_CODE > 0x2032D && defined(CONFIG_DEVFS_FS)) 
	serial_driver.name = "tts/%d"; 
#else 
	serial_driver.name = "ttyS"; 
#endif 
	serial_driver.major = TTY_MAJOR; 
	serial_driver.minor_start = 64 + SERIAL_DEV_OFFSET; 
	serial_driver.num = NR_PORTS; 
	serial_driver.type = TTY_DRIVER_TYPE_SERIAL; 
	serial_driver.subtype = SERIAL_TYPE_NORMAL; 
	serial_driver.init_termios = tty_std_termios; 
	serial_driver.init_termios.c_cflag = 
		B115200 | CS8 | CREAD | HUPCL | CLOCAL; 
	serial_driver.flags = TTY_DRIVER_REAL_RAW | TTY_DRIVER_NO_DEVFS; 
	serial_driver.refcount = &serial_refcount; 
	serial_driver.table = serial_table; 
	serial_driver.termios = serial_termios; 
	serial_driver.termios_locked = serial_termios_locked; 
 
	serial_driver.open = rs_open; 
	serial_driver.close = rs_close; 
	serial_driver.write = rs_write; 
	serial_driver.put_char = rs_put_char; 
	serial_driver.flush_chars = rs_flush_chars; 
	serial_driver.write_room = rs_write_room; 
	serial_driver.chars_in_buffer = rs_chars_in_buffer; 
	serial_driver.flush_buffer = rs_flush_buffer; 
	serial_driver.ioctl = rs_ioctl; 
	serial_driver.throttle = rs_throttle; 
	serial_driver.unthrottle = rs_unthrottle; 
	serial_driver.set_termios = rs_set_termios; 
	serial_driver.stop = rs_stop; 
	serial_driver.start = rs_start; 
	serial_driver.hangup = rs_hangup; 
#if (LINUX_VERSION_CODE >= 131394) // Linux 2.1.66 
	serial_driver.break_ctl = rs_break; 
#endif 
#if (LINUX_VERSION_CODE >= 131343) 
	serial_driver.send_xchar = rs_send_xchar; 
	serial_driver.wait_until_sent = rs_wait_until_sent; 
	serial_driver.read_proc = rs_read_proc; 
#endif 
	 
	// 
	// The callout device is just like normal device except for 
	// major number and the subtype code. 
	// 
	callout_driver = serial_driver; 
#if (LINUX_VERSION_CODE > 0x2032D && defined(CONFIG_DEVFS_FS)) 
	callout_driver.name = "cua/%d"; 
#else 
	callout_driver.name = "cua"; 
#endif 
	callout_driver.major = TTYAUX_MAJOR; 
	callout_driver.subtype = SERIAL_TYPE_CALLOUT; 
#if (LINUX_VERSION_CODE >= 131343) 
	callout_driver.read_proc = 0; 
	callout_driver.proc_entry = 0; 
#endif 
 
	if (tty_register_driver(&serial_driver)) 
		panic("Couldn't register serial driver\n"); 
	if (tty_register_driver(&callout_driver)) 
		panic("Couldn't register callout driver\n"); 
	 
	for (i = 0, state = rs_table; i < NR_PORTS; i++,state++) { 
		state->magic = SSTATE_MAGIC; 
		state->line = i; 
		state->type = PORT_UNKNOWN; 
		state->custom_divisor = 0; 
		state->close_delay = 5*HZ/10; 
		state->closing_wait = 30*HZ; 
		state->callout_termios = callout_driver.init_termios; 
		state->normal_termios = serial_driver.init_termios; 
		state->icount.cts = state->icount.dsr =  
			state->icount.rng = state->icount.dcd = 0; 
		state->icount.rx = state->icount.tx = 0; 
		state->icount.frame = state->icount.parity = 0; 
		state->icount.overrun = state->icount.brk = 0; 
		state->irq = irq_cannonicalize(state->irq); 
		if (state->hub6) 
			state->io_type = SERIAL_IO_HUB6; 
		if (state->port && check_region(state->port,8)) 
			continue; 
#ifdef CONFIG_MCA			 
		if ((state->flags & ASYNC_BOOT_ONLYMCA) && !MCA_bus) 
			continue; 
#endif			 
		if (state->flags & ASYNC_BOOT_AUTOCONF) 
			autoconfig(state); 
	} 
	for (i = 0, state = rs_table; i < NR_PORTS; i++,state++) { 
		if (state->type == PORT_UNKNOWN) 
			continue; 
		if (   (state->flags & ASYNC_BOOT_AUTOCONF) 
		    && (state->flags & ASYNC_AUTO_IRQ) 
		    && (state->port != 0)) 
			state->irq = detect_uart_irq(state); 
		printk(KERN_INFO "ttyS%02d%s at 0x%04lx (irq = %d) is a %s\n", 
		       state->line + SERIAL_DEV_OFFSET, 
		       (state->flags & ASYNC_FOURPORT) ? " FourPort" : "", 
		       state->port, state->irq, 
		       uart_config[state->type].name); 
		tty_register_devfs(&serial_driver, 0, 
				   serial_driver.minor_start + state->line); 
		tty_register_devfs(&callout_driver, 0, 
				   callout_driver.minor_start + state->line); 
	} 
	return 0; 
} 
 
static void __exit rs_fini(void)  
{ 
	unsigned long flags; 
	int e1, e2; 
	int i; 
	struct async_struct *info; 
 
	// printk("Unloading %s: version %s\n", serial_name, serial_version); 
	del_timer_sync(&serial_timer); 
	save_flags(flags); cli(); 
        remove_bh(SERIAL_BH); 
	if ((e1 = tty_unregister_driver(&serial_driver))) 
		printk("serial: failed to unregister serial driver (%d)\n", 
		       e1); 
	if ((e2 = tty_unregister_driver(&callout_driver))) 
		printk("serial: failed to unregister callout driver (%d)\n",  
		       e2); 
	restore_flags(flags); 
 
	for (i = 0; i < NR_PORTS; i++) { 
		if ((info = rs_table[i].info)) { 
			rs_table[i].info = NULL; 
			kfree(info); 
		} 
		if ((rs_table[i].type != PORT_UNKNOWN) && rs_table[i].port) { 
				release_region(rs_table[i].port, 8); 
		} 
	} 
	if (tmp_buf) { 
		unsigned long pg = (unsigned long) tmp_buf; 
		tmp_buf = NULL; 
		free_page(pg); 
	} 
} 
 
module_init(rs_init); 
module_exit(rs_fini); 
MODULE_DESCRIPTION("Standard/generic (dumb) serial driver"); 
MODULE_AUTHOR("Theodore Ts'o "); 
MODULE_LICENSE("GPL"); 
 
#ifndef CONFIG_SERIAL_SAMSUNG_CONSOLE 
 
#define BOTH_EMPTY (UART_LSR_TEMT | UART_LSR_THRE) 
 
/* 
 *	Wait for transmitter & holding register to empty 
 */ 
static _INLINE_ void wait_for_xmitr(struct async_struct *info) 
{ 
	unsigned int status, tmout = 1000000; 
 
	do { 
		if(info->line==0||info->line==1) 
			status = serial_in(info, UART_LSR); 
		else 
			status = serial_in(info, EX_UART_LSR); 
 
		if (--tmout == 0) 
			break; 
	} while((status & BOTH_EMPTY) != BOTH_EMPTY); 
} 
#endif 
/* 
 * ------------------------------------------------------------ 
 * Serial console driver 
 * ------------------------------------------------------------ 
 */ 
#ifdef CONFIG_SERIAL_SAMSUNG_CONSOLE 
 
#define BOTH_EMPTY (UART_LSR_TEMT | UART_LSR_THRE) 
 
static struct async_struct async_sercons; 
 
/* 
 *	Wait for transmitter & holding register to empty 
 */ 
static _INLINE_ void wait_for_xmitr(struct async_struct *info) 
{ 
	unsigned int status, tmout = 1000000; 
 
	do { 
		if(info->line==0||info->line==1) 
			status = serial_in(info, UART_LSR); 
		else 
			status = serial_in(info, EX_UART_LSR); 
 
		if (status & UART_LSR_BI) 
			lsr_break_flag = UART_LSR_BI; 
 
		if (--tmout == 0) 
			break; 
	} while((status & BOTH_EMPTY) != BOTH_EMPTY); 
} 
 
 
/* 
 *	Print a string to the serial port trying not to disturb 
 *	any possible real use of the port... 
 * 
 *	The console must be locked when we get here. 
 */ 
static void serial_console_write(struct console *co, const char *s, 
				unsigned count) 
{ 
	static struct async_struct *info = &async_sercons; 
	unsigned i; 
 
	/* 
	 *	First save the IER then disable the interrupts 
	 */ 
 
	/* 
	 *	Now, do each character 
	 */ 
	for (i = 0; i < count; i++, s++) { 
		wait_for_xmitr(info); 
 
		/* 
		 *	Send the character out. 
		 *	If a LF, also do CR... 
		 */ 
		if(info->line==0||info->line==1) 
			uart_out(info, UART_TX, *s); 
		/*else                                   */ 
			/*uart_out(info, EX_UART_TX, *s);*/ 
		if (*s == 10) { 
			wait_for_xmitr(info); 
			if(info->line==0||info->line==1) 
				uart_out(info, UART_TX, 13); 
			/*else                                   */ 
				/*uart_out(info, EX_UART_TX, 13);*/ 
		} 
	} 
 
	/* 
	 *	Finally, Wait for transmitter & holding register to empty 
	 * 	and restore the IER 
	 */ 
	wait_for_xmitr(info); 
} 
 
/* 
 *	Receive character from the serial port 
 */ 
static int serial_console_wait_key(struct console *co) 
{ 
	static struct async_struct *info; 
	int c; 
	int tmp; 
	info = &async_sercons; 
 
	/* 
	 *	First save the IER then disable the interrupts so 
	 *	that the real driver for the port does not get the 
	 *	character. 
	 */ 
  
	if(info->line==0||info->line==1) 
	    tmp=UART_LSR; 
	/*else                */ 
	    /*tmp=EX_UART_LSR;*/ 
	while ((serial_in(info, tmp) & UART_LSR_DR) == 0); 
	if(info->line==0||info->line==1) 
		c = uart_in(info, UART_RX); 
	/*else                                  */ 
		/*c = uart_in(info, EX_UART_RX);*/ 
 
	return c; 
} 
 
static kdev_t serial_console_device(struct console *c) 
{ 
	return MKDEV(TTY_MAJOR, 64 + c->index); 
} 
 
/* 
 *	Setup initial baud/bits/parity/flow control. We do two things here: 
 *	- construct a cflag setting for the first rs_open() 
 *	- initialize the serial port 
 *	Return non-zero if we didn't find a serial port. 
 */ 
static int __init serial_console_setup(struct console *co, char *options) 
{ 
	static struct async_struct *info; 
	struct serial_state *state; 
	unsigned cval; 
	int	baud = 115200; 
	int	bits = 8; 
	int	parity = 'n'; 
	int	doflow = 0; 
	int	cflag = CREAD | HUPCL | CLOCAL; 
	int	quot = 0; 
	char	*s; 
 
	if (options) { 
		baud = simple_strtoul(options, NULL, 10); 
		s = options; 
		while(*s >= '0' && *s <= '9') 
			s++; 
		if (*s) parity = *s++; 
		if (*s) bits   = *s++ - '0'; 
		if (*s) doflow = (*s++ == 'r'); 
	} 
 
	/* 
	 *	Now construct a cflag setting. 
	 */ 
	switch(baud) { 
		case 1200: 
			cflag |= B1200; 
			break; 
		case 2400: 
			cflag |= B2400; 
			break; 
		case 4800: 
			cflag |= B4800; 
			break; 
		case 19200: 
			cflag |= B19200; 
			break; 
		case 38400: 
			cflag |= B38400; 
			break; 
		case 57600: 
			cflag |= B57600; 
			break; 
		case 115200: 
			cflag |= B115200; 
			break; 
		case 9600: 
		default: 
			cflag |= B9600; 
			/* 
			 * Set this to a sane value to prevent a divide error 
			 */ 
			baud  = 9600; 
			break; 
	} 
	switch(bits) { 
		case 7: 
			cflag |= CS7; 
			break; 
		default: 
		case 8: 
			cflag |= CS8; 
			break; 
	} 
	switch(parity) { 
		case 'o': case 'O': 
			cflag |= PARODD; 
			break; 
		case 'e': case 'E': 
			cflag |= PARENB; 
			break; 
	} 
	co->cflag = cflag; 
 
	/* 
	 *	Divisor, bytesize and parity 
	 */ 
	state = rs_table + co->index; 
	if (doflow) 
		state->flags |= ASYNC_CONS_FLOW; 
	info = &async_sercons; 
	info->magic = SERIAL_MAGIC; 
	info->state = state; 
	info->port = state->port; 
	info->flags = state->flags; 
	info->io_type = state->io_type; 
	info->iomem_base = state->iomem_base; 
	info->iomem_reg_shift = state->iomem_reg_shift; 
	quot = state->baud_base / baud; 
	cval = cflag & (CSIZE | CSTOPB); 
	cval >>= 4; 
 
	if (cflag & PARENB) 
	{ 
		cval |= UART_LCR_PARITY; 
		if (!(cflag & PARODD)) 
			cval |= UART_LCR_EPAR; 
	} 
 
    /* set up uart's IO Pins*/ 
    if (info->line){ 
        CSR_WRITE(rPCONC,((3 << 20)| (3 << 22)|(3 << 24)| (3 << 26) | (3<<28)|(3 <<30))); // Enable UART1's Pin 
        CSR_WRITE(rPUPC,CSR_READ(rPUPC)|0xfc00); // set pull up 
#ifdef CONFIG_SERIAL_SAMSUNG_IRDA 
        CSR_WRITE(rPCONE,CSR_READ(rPCONE)|1);    // Enable UART0's Pin 
		CSR_WRITE(rPDATE,CSR_READ(rPDATE)&0xFE); // select enable IRDA 
#endif 
    } 
    else{ 
		CSR_WRITE(rPCONE,CSR_READ(rPCONE)|(2 << 2) | (2 << 4)); 
	} 
 
	/* 
	 *	Disable UART interrupts, set DTR and RTS high 
	 *	and set speed. 
	 */ 
	if(info->line==0||info->line==1) 
	{ 
		serial_out(info, UART_BDR, baudrate_div(baud)); 
		serial_out(info, UART_LCR, cval); 
		serial_out(info, UART_GCR, UART_GCR_RX_INT|UART_GCR_TX_INT); 
	} 
	/*else                                                                          */ 
	/*{                                                                             */ 
		/*serial_outp(info, EX_UART_LCR, 0x80);	//set line ctrol                */ 
		/*serial_outp(info, EX_UART_BDR, 115200%baud);                          */ 
		/*serial_outp(info, EX_UART_BDR+1, 115200/baud);                        */ 
		/*serial_outp(info, EX_UART_LCR, cval);	//re set line ctrol             */ 
		/*serial_outp(info, EX_UART_GCR, EX_UART_GCR_RX_INT|EX_UART_GCR_TX_INT);*/ 
										      /**/ 
	/*}                                                                             */ 
	return 0; 
} 
 
static struct console sercons = { 
	name:		"ttyS", 
	write:		serial_console_write, 
	device:		serial_console_device, 
	wait_key:	serial_console_wait_key, 
	setup:		serial_console_setup, 
	flags:		CON_PRINTBUFFER, 
	index:		-1, 
}; 
 
/* 
 *	Register console. 
 */ 
void __init samsung_console_init(void) 
{ 
	register_console(&sercons); 
} 
#endif 
 
/* 
  Local variables: 
  compile-command: "gcc -D__KERNEL__ -I../../include -Wall -Wstrict-prototypes -O2 -fomit-frame-pointer -fno-strict-aliasing -pipe -fno-strength-reduce -march=i586 -DMODULE -DMODVERSIONS -include ../../include/linux/modversions.h   -DEXPORT_SYMTAB -c serial.c" 
  End: 
*/