[PATCH 2/2] serial 16c950 : add preliminary support for 9-bit receive and transmit

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



From c706030ca2374fc3219010fcd479e0b5de36cabf Mon Sep 17 00:00:00 2001
From: pinkasfeld joseph <jpinkasfeld@xxxxxxxxxxxxxxxxxxxxxx>
Date: Wed, 18 Nov 2009 16:35:16 +0100
Subject: [PATCH 2/2] serial 16c950 : add preliminary support for 9-bit receive and transmit
 2 bytes must be written or read for transmiting or receiving 9-bit data
 first byte LSB is ninth bit, second byte is normal data

Signed-off-by: pinkasfeld joseph <jpinkasfeld@xxxxxxxxxxxxxxxxxxxxxx>
---
 drivers/serial/8250.c |   54 ++++++++++++++++++++++++++++++++++++++++++++----
 1 files changed, 49 insertions(+), 5 deletions(-)

diff --git a/drivers/serial/8250.c b/drivers/serial/8250.c
index 737b4c9..1b4143f 100644
--- a/drivers/serial/8250.c
+++ b/drivers/serial/8250.c
@@ -134,6 +134,7 @@ struct uart_8250_port {
 	unsigned short		bugs;		/* port bugs */
 	unsigned int		tx_loadsz;	/* transmit fifo load size */
 	unsigned char		acr;
+	unsigned char		nmr;
 	unsigned char		ier;
 	unsigned char		lcr;
 	unsigned char		mcr;
@@ -835,6 +836,7 @@ static void autoconfig_has_efr(struct uart_8250_port *up)
 	 * recommended for new designs).
 	 */
 	up->acr = 0;
+	up->nmr = 0;
 	serial_out(up, UART_LCR, 0xBF);
 	serial_out(up, UART_EFR, UART_EFR_ECB);
 	serial_out(up, UART_LCR, 0x00);
@@ -1385,8 +1387,8 @@ static void
 receive_chars(struct uart_8250_port *up, unsigned int *status)
 {
 	struct tty_struct *tty = up->port.state->port.tty;
-	unsigned char ch, lsr = *status;
-	int max_count = 256;
+	unsigned char ch, ch9, lsr = *status;
+	int error_break = 0, max_count = 256;
 	char flag;
 
 	do {
@@ -1408,7 +1410,20 @@ receive_chars(struct uart_8250_port *up, unsigned int *status)
 		lsr |= up->lsr_saved_flags;
 		up->lsr_saved_flags = 0;
 
-		if (unlikely(lsr & UART_LSR_BRK_ERROR_BITS)) {
+		/*
+		 * Due to sharing of bit 2 in LSR depending on 9 bits mode
+		 * bit 2 rise Parity Error or 9th bit read
+		 */
+		error_break = 0 ;
+		if (up->nmr & UART_NMR_9BENB) {
+			if (lsr & UART_LSR_NBM_BRK_ERROR_BITS)
+				error_break = 1;
+		} else {
+			if (lsr & UART_LSR_BRK_ERROR_BITS)
+				error_break = 1;
+		}
+
+		if (unlikely(error_break)) {
 			/*
 			 * For statistics only
 			 */
@@ -1423,7 +1438,8 @@ receive_chars(struct uart_8250_port *up, unsigned int *status)
 				 */
 				if (uart_handle_break(&up->port))
 					goto ignore_char;
-			} else if (lsr & UART_LSR_PE)
+			} else if ((lsr & UART_LSR_PE) &&
+				!(up->nmr & UART_NMR_9BENB))
 				up->port.icount.parity++;
 			else if (lsr & UART_LSR_FE)
 				up->port.icount.frame++;
@@ -1438,7 +1454,8 @@ receive_chars(struct uart_8250_port *up, unsigned int *status)
 			if (lsr & UART_LSR_BI) {
 				DEBUG_INTR("handling break....");
 				flag = TTY_BREAK;
-			} else if (lsr & UART_LSR_PE)
+			} else if ((lsr & UART_LSR_PE) &&
+				!(up->nmr & UART_NMR_9BENB))
 				flag = TTY_PARITY;
 			else if (lsr & UART_LSR_FE)
 				flag = TTY_FRAME;
@@ -1446,6 +1463,16 @@ receive_chars(struct uart_8250_port *up, unsigned int *status)
 		if (uart_handle_sysrq_char(&up->port, ch))
 			goto ignore_char;
 
+		/*
+		 * In 9bit mode Parity error is replace by 9th bit
+		 */
+		if (up->nmr & UART_NMR_9BENB) {
+			if (lsr & UART_LSR_9TH_BIT)
+				ch9 = 1;
+			else
+				ch9 = 0;
+			uart_insert_char(&up->port, lsr, UART_LSR_OE, ch9, flag);
+		}
 		uart_insert_char(&up->port, lsr, UART_LSR_OE, ch, flag);
 
 ignore_char:
@@ -1479,6 +1506,13 @@ static void transmit_chars(struct uart_8250_port *up)
 
 	count = up->tx_loadsz;
 	do {
+		/* first byte contain 9th bit for 9 bit mode */
+		if ((up->nmr  & UART_NMR_9BENB)  && (count > 1) ){
+			serial_out(up, UART_SCR, 0x01 & xmit->buf[xmit->tail]);
+			xmit->tail = (xmit->tail + 1) & (UART_XMIT_SIZE - 1);
+			if (uart_circ_empty(xmit))
+				break;
+		}
 		serial_out(up, UART_TX, xmit->buf[xmit->tail]);
 		xmit->tail = (xmit->tail + 1) & (UART_XMIT_SIZE - 1);
 		up->port.icount.tx++;
@@ -1956,6 +1990,7 @@ static int serial8250_startup(struct uart_port *port)
 	if (up->port.type == PORT_16C950) {
 		/* Wake up and initialize UART */
 		up->acr = 0;
+		up->nmr = 0;
 		serial_outp(up, UART_LCR, 0xBF);
 		serial_outp(up, UART_EFR, UART_EFR_ECB);
 		serial_outp(up, UART_IER, 0);
@@ -2412,6 +2447,15 @@ serial8250_set_termios(struct uart_port *port, struct ktermios *termios,
 	/* Don't rewrite B0 */
 	if (tty_termios_baud_rate(termios))
 		tty_termios_encode_baud_rate(termios, baud, baud);
+
+	/*
+	 * 9-bit mode control flag on 16c950
+	 */
+	if (termios->c_cflag & CNBENB) {
+		up->nmr |= UART_NMR_9BENB;
+		serial_icr_write(up, UART_NMR, up->nmr);
+	}
+
 }
 
 static void
-- 
1.6.5.2


[Index of Archives]     [Kernel Newbies]     [Security]     [Netfilter]     [Bugtraq]     [Linux PPP]     [Linux FS]     [Yosemite News]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Samba]     [Video 4 Linux]     [Linmodem]     [Device Mapper]     [Linux Kernel for ARM]

  Powered by Linux