Commit d358788f authored by Russell King's avatar Russell King Committed by Russell King

[SERIAL] kernel console should send CRLF not LFCR

Glen Turner reported that writing LFCR rather than the more
traditional CRLF causes issues with some terminals.

Since this aflicts many serial drivers, extract the common code
to a library function (uart_console_write) and arrange for each
driver to supply a "putchar" function.
Signed-off-by: default avatarRussell King <rmk+kernel@arm.linux.org.uk>
parent 7705a879
...@@ -375,23 +375,18 @@ static void serial21285_setup_ports(void) ...@@ -375,23 +375,18 @@ static void serial21285_setup_ports(void)
} }
#ifdef CONFIG_SERIAL_21285_CONSOLE #ifdef CONFIG_SERIAL_21285_CONSOLE
static void serial21285_console_putchar(struct uart_port *port, int ch)
{
while (*CSR_UARTFLG & 0x20)
barrier();
*CSR_UARTDR = ch;
}
static void static void
serial21285_console_write(struct console *co, const char *s, serial21285_console_write(struct console *co, const char *s,
unsigned int count) unsigned int count)
{ {
int i; uart_console_write(&serial21285_port, s, count, serial21285_console_putchar);
for (i = 0; i < count; i++) {
while (*CSR_UARTFLG & 0x20)
barrier();
*CSR_UARTDR = s[i];
if (s[i] == '\n') {
while (*CSR_UARTFLG & 0x20)
barrier();
*CSR_UARTDR = '\r';
}
}
} }
static void __init static void __init
......
...@@ -2182,6 +2182,14 @@ static inline void wait_for_xmitr(struct uart_8250_port *up, int bits) ...@@ -2182,6 +2182,14 @@ static inline void wait_for_xmitr(struct uart_8250_port *up, int bits)
} }
} }
static void serial8250_console_putchar(struct uart_port *port, int ch)
{
struct uart_8250_port *up = (struct uart_8250_port *)port;
wait_for_xmitr(up, UART_LSR_THRE);
serial_out(up, UART_TX, ch);
}
/* /*
* Print a string to the serial port trying not to disturb * Print a string to the serial port trying not to disturb
* any possible real use of the port... * any possible real use of the port...
...@@ -2193,7 +2201,6 @@ serial8250_console_write(struct console *co, const char *s, unsigned int count) ...@@ -2193,7 +2201,6 @@ serial8250_console_write(struct console *co, const char *s, unsigned int count)
{ {
struct uart_8250_port *up = &serial8250_ports[co->index]; struct uart_8250_port *up = &serial8250_ports[co->index];
unsigned int ier; unsigned int ier;
int i;
touch_nmi_watchdog(); touch_nmi_watchdog();
...@@ -2207,22 +2214,7 @@ serial8250_console_write(struct console *co, const char *s, unsigned int count) ...@@ -2207,22 +2214,7 @@ serial8250_console_write(struct console *co, const char *s, unsigned int count)
else else
serial_out(up, UART_IER, 0); serial_out(up, UART_IER, 0);
/* uart_console_write(&up->port, s, count, serial8250_console_putchar);
* Now, do each character
*/
for (i = 0; i < count; i++, s++) {
wait_for_xmitr(up, UART_LSR_THRE);
/*
* Send the character out.
* If a LF, also do CR...
*/
serial_out(up, UART_TX, *s);
if (*s == 10) {
wait_for_xmitr(up, UART_LSR_THRE);
serial_out(up, UART_TX, 13);
}
}
/* /*
* Finally, wait for transmitter to become empty * Finally, wait for transmitter to become empty
......
...@@ -74,7 +74,7 @@ static void __init wait_for_xmitr(struct uart_port *port) ...@@ -74,7 +74,7 @@ static void __init wait_for_xmitr(struct uart_port *port)
} }
} }
static void __init putc(struct uart_port *port, unsigned char c) static void __init putc(struct uart_port *port, int c)
{ {
wait_for_xmitr(port); wait_for_xmitr(port);
serial_out(port, UART_TX, c); serial_out(port, UART_TX, c);
...@@ -89,12 +89,7 @@ static void __init early_uart_write(struct console *console, const char *s, unsi ...@@ -89,12 +89,7 @@ static void __init early_uart_write(struct console *console, const char *s, unsi
ier = serial_in(port, UART_IER); ier = serial_in(port, UART_IER);
serial_out(port, UART_IER, 0); serial_out(port, UART_IER, 0);
while (*s && count-- > 0) { uart_console_write(port, s, count, putc);
putc(port, *s);
if (*s == '\n')
putc(port, '\r');
s++;
}
/* Wait for transmitter to become empty and restore the IER */ /* Wait for transmitter to become empty and restore the IER */
wait_for_xmitr(port); wait_for_xmitr(port);
......
...@@ -591,12 +591,18 @@ static struct uart_amba_port amba_ports[UART_NR] = { ...@@ -591,12 +591,18 @@ static struct uart_amba_port amba_ports[UART_NR] = {
#ifdef CONFIG_SERIAL_AMBA_PL010_CONSOLE #ifdef CONFIG_SERIAL_AMBA_PL010_CONSOLE
static void pl010_console_putchar(struct uart_port *port, int ch)
{
while (!UART_TX_READY(UART_GET_FR(port)))
barrier();
UART_PUT_CHAR(port, ch);
}
static void static void
pl010_console_write(struct console *co, const char *s, unsigned int count) pl010_console_write(struct console *co, const char *s, unsigned int count)
{ {
struct uart_port *port = &amba_ports[co->index].port; struct uart_port *port = &amba_ports[co->index].port;
unsigned int status, old_cr; unsigned int status, old_cr;
int i;
/* /*
* First save the CR then disable the interrupts * First save the CR then disable the interrupts
...@@ -604,21 +610,7 @@ pl010_console_write(struct console *co, const char *s, unsigned int count) ...@@ -604,21 +610,7 @@ pl010_console_write(struct console *co, const char *s, unsigned int count)
old_cr = UART_GET_CR(port); old_cr = UART_GET_CR(port);
UART_PUT_CR(port, UART01x_CR_UARTEN); UART_PUT_CR(port, UART01x_CR_UARTEN);
/* uart_console_write(port, s, count, pl010_console_putchar);
* Now, do each character
*/
for (i = 0; i < count; i++) {
do {
status = UART_GET_FR(port);
} while (!UART_TX_READY(status));
UART_PUT_CHAR(port, s[i]);
if (s[i] == '\n') {
do {
status = UART_GET_FR(port);
} while (!UART_TX_READY(status));
UART_PUT_CHAR(port, '\r');
}
}
/* /*
* Finally, wait for transmitter to become empty * Finally, wait for transmitter to become empty
......
...@@ -587,14 +587,12 @@ static struct uart_amba_port *amba_ports[UART_NR]; ...@@ -587,14 +587,12 @@ static struct uart_amba_port *amba_ports[UART_NR];
#ifdef CONFIG_SERIAL_AMBA_PL011_CONSOLE #ifdef CONFIG_SERIAL_AMBA_PL011_CONSOLE
static inline void static void pl011_console_putchar(struct uart_port *port, int ch)
pl011_console_write_char(struct uart_amba_port *uap, char ch)
{ {
unsigned int status; struct uart_amba_port *uap = (struct uart_amba_port *)port;
do { while (readw(uap->port.membase + UART01x_FR) & UART01x_FR_TXFF)
status = readw(uap->port.membase + UART01x_FR); barrier();
} while (status & UART01x_FR_TXFF);
writew(ch, uap->port.membase + UART01x_DR); writew(ch, uap->port.membase + UART01x_DR);
} }
...@@ -603,7 +601,6 @@ pl011_console_write(struct console *co, const char *s, unsigned int count) ...@@ -603,7 +601,6 @@ pl011_console_write(struct console *co, const char *s, unsigned int count)
{ {
struct uart_amba_port *uap = amba_ports[co->index]; struct uart_amba_port *uap = amba_ports[co->index];
unsigned int status, old_cr, new_cr; unsigned int status, old_cr, new_cr;
int i;
clk_enable(uap->clk); clk_enable(uap->clk);
...@@ -615,14 +612,7 @@ pl011_console_write(struct console *co, const char *s, unsigned int count) ...@@ -615,14 +612,7 @@ pl011_console_write(struct console *co, const char *s, unsigned int count)
new_cr |= UART01x_CR_UARTEN | UART011_CR_TXE; new_cr |= UART01x_CR_UARTEN | UART011_CR_TXE;
writew(new_cr, uap->port.membase + UART011_CR); writew(new_cr, uap->port.membase + UART011_CR);
/* uart_console_write(&uap->port, s, count, pl011_console_putchar);
* Now, do each character
*/
for (i = 0; i < count; i++) {
pl011_console_write_char(uap, s[i]);
if (s[i] == '\n')
pl011_console_write_char(uap, '\r');
}
/* /*
* Finally, wait for transmitter to become empty * Finally, wait for transmitter to become empty
......
...@@ -711,6 +711,12 @@ void __init at91_register_uart(int idx, int port) ...@@ -711,6 +711,12 @@ void __init at91_register_uart(int idx, int port)
} }
#ifdef CONFIG_SERIAL_AT91_CONSOLE #ifdef CONFIG_SERIAL_AT91_CONSOLE
static void at91_console_putchar(struct uart_port *port, int ch)
{
while (!(UART_GET_CSR(port) & AT91_US_TXRDY))
barrier();
UART_PUT_CHAR(port, ch);
}
/* /*
* Interrupts are disabled on entering * Interrupts are disabled on entering
...@@ -718,7 +724,7 @@ void __init at91_register_uart(int idx, int port) ...@@ -718,7 +724,7 @@ void __init at91_register_uart(int idx, int port)
static void at91_console_write(struct console *co, const char *s, u_int count) static void at91_console_write(struct console *co, const char *s, u_int count)
{ {
struct uart_port *port = at91_ports + co->index; struct uart_port *port = at91_ports + co->index;
unsigned int status, i, imr; unsigned int status, imr;
/* /*
* First, save IMR and then disable interrupts * First, save IMR and then disable interrupts
...@@ -726,21 +732,7 @@ static void at91_console_write(struct console *co, const char *s, u_int count) ...@@ -726,21 +732,7 @@ static void at91_console_write(struct console *co, const char *s, u_int count)
imr = UART_GET_IMR(port); /* get interrupt mask */ imr = UART_GET_IMR(port); /* get interrupt mask */
UART_PUT_IDR(port, AT91_US_RXRDY | AT91_US_TXRDY); UART_PUT_IDR(port, AT91_US_RXRDY | AT91_US_TXRDY);
/* uart_console_write(port, s, count, at91_console_putchar);
* Now, do each character
*/
for (i = 0; i < count; i++) {
do {
status = UART_GET_CSR(port);
} while (!(status & AT91_US_TXRDY));
UART_PUT_CHAR(port, s[i]);
if (s[i] == '\n') {
do {
status = UART_GET_CSR(port);
} while (!(status & AT91_US_TXRDY));
UART_PUT_CHAR(port, '\r');
}
}
/* /*
* Finally, wait for transmitter to become empty * Finally, wait for transmitter to become empty
......
...@@ -1121,6 +1121,14 @@ static inline void wait_for_xmitr(struct uart_8250_port *up) ...@@ -1121,6 +1121,14 @@ static inline void wait_for_xmitr(struct uart_8250_port *up)
} }
} }
static void au1x00_console_putchar(struct uart_port *port, int ch)
{
struct uart_8250_port *up = (struct uart_8250_port *)port;
wait_for_xmitr(up);
serial_out(up, UART_TX, ch);
}
/* /*
* Print a string to the serial port trying not to disturb * Print a string to the serial port trying not to disturb
* any possible real use of the port... * any possible real use of the port...
...@@ -1132,7 +1140,6 @@ serial8250_console_write(struct console *co, const char *s, unsigned int count) ...@@ -1132,7 +1140,6 @@ serial8250_console_write(struct console *co, const char *s, unsigned int count)
{ {
struct uart_8250_port *up = &serial8250_ports[co->index]; struct uart_8250_port *up = &serial8250_ports[co->index];
unsigned int ier; unsigned int ier;
int i;
/* /*
* First save the UER then disable the interrupts * First save the UER then disable the interrupts
...@@ -1140,22 +1147,7 @@ serial8250_console_write(struct console *co, const char *s, unsigned int count) ...@@ -1140,22 +1147,7 @@ serial8250_console_write(struct console *co, const char *s, unsigned int count)
ier = serial_in(up, UART_IER); ier = serial_in(up, UART_IER);
serial_out(up, UART_IER, 0); serial_out(up, UART_IER, 0);
/* uart_console_write(&up->port, s, count, au1x00_console_putchar);
* Now, do each character
*/
for (i = 0; i < count; i++, s++) {
wait_for_xmitr(up);
/*
* Send the character out.
* If a LF, also do CR...
*/
serial_out(up, UART_TX, *s);
if (*s == 10) {
wait_for_xmitr(up);
serial_out(up, UART_TX, 13);
}
}
/* /*
* Finally, wait for transmitter to become empty * Finally, wait for transmitter to become empty
......
...@@ -424,6 +424,13 @@ static struct uart_port clps711x_ports[UART_NR] = { ...@@ -424,6 +424,13 @@ static struct uart_port clps711x_ports[UART_NR] = {
}; };
#ifdef CONFIG_SERIAL_CLPS711X_CONSOLE #ifdef CONFIG_SERIAL_CLPS711X_CONSOLE
static void clps711xuart_console_putchar(struct uart_port *port, int ch)
{
while (clps_readl(SYSFLG(port)) & SYSFLG_UTXFF)
barrier();
clps_writel(ch, UARTDR(port));
}
/* /*
* Print a string to the serial port trying not to disturb * Print a string to the serial port trying not to disturb
* any possible real use of the port... * any possible real use of the port...
...@@ -438,7 +445,6 @@ clps711xuart_console_write(struct console *co, const char *s, ...@@ -438,7 +445,6 @@ clps711xuart_console_write(struct console *co, const char *s,
{ {
struct uart_port *port = clps711x_ports + co->index; struct uart_port *port = clps711x_ports + co->index;
unsigned int status, syscon; unsigned int status, syscon;
int i;
/* /*
* Ensure that the port is enabled. * Ensure that the port is enabled.
...@@ -446,21 +452,7 @@ clps711xuart_console_write(struct console *co, const char *s, ...@@ -446,21 +452,7 @@ clps711xuart_console_write(struct console *co, const char *s,
syscon = clps_readl(SYSCON(port)); syscon = clps_readl(SYSCON(port));
clps_writel(syscon | SYSCON_UARTEN, SYSCON(port)); clps_writel(syscon | SYSCON_UARTEN, SYSCON(port));
/* uart_console_write(port, s, count, clps711xuart_console_putchar);
* Now, do each character
*/
for (i = 0; i < count; i++) {
do {
status = clps_readl(SYSFLG(port));
} while (status & SYSFLG_UTXFF);
clps_writel(s[i], UARTDR(port));
if (s[i] == '\n') {
do {
status = clps_readl(SYSFLG(port));
} while (status & SYSFLG_UTXFF);
clps_writel('\r', UARTDR(port));
}
}
/* /*
* Finally, wait for transmitter to become empty * Finally, wait for transmitter to become empty
......
...@@ -674,11 +674,12 @@ static void dz_reset(struct dz_port *dport) ...@@ -674,11 +674,12 @@ static void dz_reset(struct dz_port *dport)
} }
#ifdef CONFIG_SERIAL_DZ_CONSOLE #ifdef CONFIG_SERIAL_DZ_CONSOLE
static void dz_console_put_char(struct dz_port *dport, unsigned char ch) static void dz_console_putchar(struct uart_port *port, int ch)
{ {
struct dz_port *dport = (struct dz_port *)uport;
unsigned long flags; unsigned long flags;
int loops = 2500; int loops = 2500;
unsigned short tmp = ch; unsigned short tmp = (unsigned char)ch;
/* this code sends stuff out to serial device - spinning its /* this code sends stuff out to serial device - spinning its
wheels and waiting. */ wheels and waiting. */
...@@ -694,6 +695,7 @@ static void dz_console_put_char(struct dz_port *dport, unsigned char ch) ...@@ -694,6 +695,7 @@ static void dz_console_put_char(struct dz_port *dport, unsigned char ch)
spin_unlock_irqrestore(&dport->port.lock, flags); spin_unlock_irqrestore(&dport->port.lock, flags);
} }
/* /*
* ------------------------------------------------------------------- * -------------------------------------------------------------------
* dz_console_print () * dz_console_print ()
...@@ -710,11 +712,7 @@ static void dz_console_print(struct console *cons, ...@@ -710,11 +712,7 @@ static void dz_console_print(struct console *cons,
#ifdef DEBUG_DZ #ifdef DEBUG_DZ
prom_printf((char *) str); prom_printf((char *) str);
#endif #endif
while (count--) { uart_console_write(&dport->port, str, count, dz_console_putchar);
if (*str == '\n')
dz_console_put_char(dport, '\r');
dz_console_put_char(dport, *str++);
}
} }
static int __init dz_console_setup(struct console *co, char *options) static int __init dz_console_setup(struct console *co, char *options)
......
...@@ -743,6 +743,13 @@ static void __init imx_init_ports(void) ...@@ -743,6 +743,13 @@ static void __init imx_init_ports(void)
} }
#ifdef CONFIG_SERIAL_IMX_CONSOLE #ifdef CONFIG_SERIAL_IMX_CONSOLE
static void imx_console_putchar(struct uart_port *port, int ch)
{
struct imx_port *sport = (struct imx_port *)port;
while ((UTS((u32)sport->port.membase) & UTS_TXFULL))
barrier();
URTX0((u32)sport->port.membase) = ch;
}
/* /*
* Interrupts are disabled on entering * Interrupts are disabled on entering
...@@ -751,7 +758,7 @@ static void ...@@ -751,7 +758,7 @@ static void
imx_console_write(struct console *co, const char *s, unsigned int count) imx_console_write(struct console *co, const char *s, unsigned int count)
{ {
struct imx_port *sport = &imx_ports[co->index]; struct imx_port *sport = &imx_ports[co->index];
unsigned int old_ucr1, old_ucr2, i; unsigned int old_ucr1, old_ucr2;
/* /*
* First, save UCR1/2 and then disable interrupts * First, save UCR1/2 and then disable interrupts
...@@ -764,22 +771,7 @@ imx_console_write(struct console *co, const char *s, unsigned int count) ...@@ -764,22 +771,7 @@ imx_console_write(struct console *co, const char *s, unsigned int count)
& ~(UCR1_TXMPTYEN | UCR1_RRDYEN | UCR1_RTSDEN); & ~(UCR1_TXMPTYEN | UCR1_RRDYEN | UCR1_RTSDEN);
UCR2((u32)sport->port.membase) = old_ucr2 | UCR2_TXEN; UCR2((u32)sport->port.membase) = old_ucr2 | UCR2_TXEN;
/* uart_console_write(&sport->port, s, count, imx_console_putchar);
* Now, do each character
*/
for (i = 0; i < count; i++) {
while ((UTS((u32)sport->port.membase) & UTS_TXFULL))
barrier();
URTX0((u32)sport->port.membase) = s[i];
if (s[i] == '\n') {
while ((UTS((u32)sport->port.membase) & UTS_TXFULL))
barrier();
URTX0((u32)sport->port.membase) = '\r';
}
}
/* /*
* Finally, wait for transmitter to become empty * Finally, wait for transmitter to become empty
......
...@@ -967,8 +967,9 @@ static struct zilog_layout * __init get_zs(int chip) ...@@ -967,8 +967,9 @@ static struct zilog_layout * __init get_zs(int chip)
#define ZS_PUT_CHAR_MAX_DELAY 2000 /* 10 ms */ #define ZS_PUT_CHAR_MAX_DELAY 2000 /* 10 ms */
#ifdef CONFIG_SERIAL_IP22_ZILOG_CONSOLE #ifdef CONFIG_SERIAL_IP22_ZILOG_CONSOLE
static void ip22zilog_put_char(struct zilog_channel *channel, unsigned char ch) static void ip22zilog_put_char(struct uart_port *port, int ch)
{ {
struct zilog_channel *channel = ZILOG_CHANNEL_FROM_PORT(port);
int loops = ZS_PUT_CHAR_MAX_DELAY; int loops = ZS_PUT_CHAR_MAX_DELAY;
/* This is a timed polling loop so do not switch the explicit /* This is a timed polling loop so do not switch the explicit
...@@ -992,16 +993,10 @@ static void ...@@ -992,16 +993,10 @@ static void
ip22zilog_console_write(struct console *con, const char *s, unsigned int count) ip22zilog_console_write(struct console *con, const char *s, unsigned int count)
{ {
struct uart_ip22zilog_port *up = &ip22zilog_port_table[con->index]; struct uart_ip22zilog_port *up = &ip22zilog_port_table[con->index];
struct zilog_channel *channel = ZILOG_CHANNEL_FROM_PORT(&up->port);
unsigned long flags; unsigned long flags;
int i;
spin_lock_irqsave(&up->port.lock, flags); spin_lock_irqsave(&up->port.lock, flags);
for (i = 0; i < count; i++, s++) { uart_console_write(&up->port, s, count, ip22zilog_put_char);
ip22zilog_put_char(channel, *s);
if (*s == 10)
ip22zilog_put_char(channel, 13);
}
udelay(2); udelay(2);
spin_unlock_irqrestore(&up->port.lock, flags); spin_unlock_irqrestore(&up->port.lock, flags);
} }
......
...@@ -1039,6 +1039,14 @@ static inline void wait_for_xmitr(struct uart_sio_port *up) ...@@ -1039,6 +1039,14 @@ static inline void wait_for_xmitr(struct uart_sio_port *up)
} }
} }
static void m32r_sio_console_putchar(struct uart_port *port, int ch)
{
struct uart_sio_port *up = (struct uart_sio_port *)port;
wait_for_xmitr(up);
sio_out(up, SIOTXB, ch);
}
/* /*
* Print a string to the serial port trying not to disturb * Print a string to the serial port trying not to disturb
* any possible real use of the port... * any possible real use of the port...
...@@ -1058,23 +1066,7 @@ static void m32r_sio_console_write(struct console *co, const char *s, ...@@ -1058,23 +1066,7 @@ static void m32r_sio_console_write(struct console *co, const char *s,
ier = sio_in(up, SIOTRCR); ier = sio_in(up, SIOTRCR);
sio_out(up, SIOTRCR, 0); sio_out(up, SIOTRCR, 0);
/* uart_console_write(&up->port, s, count, m32r_sio_console_putchar);
* Now, do each character
*/
for (i = 0; i < count; i++, s++) {
wait_for_xmitr(up);
/*
* Send the character out.
* If a LF, also do CR...
*/
sio_out(up, SIOTXB, *s);
if (*s == 10) {
wait_for_xmitr(up);
sio_out(up, SIOTXB, 13);
}
}
/* /*
* Finally, wait for transmitter to become empty * Finally, wait for transmitter to become empty
......
...@@ -603,15 +603,14 @@ mpc52xx_console_write(struct console *co, const char *s, unsigned int count) ...@@ -603,15 +603,14 @@ mpc52xx_console_write(struct console *co, const char *s, unsigned int count)
udelay(1); udelay(1);
/* Write all the chars */ /* Write all the chars */
for ( i=0 ; i<count ; i++ ) { for (i = 0; i < count; i++, s++) {
/* Send the char */
out_8(&psc->mpc52xx_psc_buffer_8, *s);
/* Line return handling */ /* Line return handling */
if ( *s++ == '\n' ) if (*s == '\n')
out_8(&psc->mpc52xx_psc_buffer_8, '\r'); out_8(&psc->mpc52xx_psc_buffer_8, '\r');
/* Send the char */
out_8(&psc->mpc52xx_psc_buffer_8, *s);
/* Wait the TX buffer to be empty */ /* Wait the TX buffer to be empty */
j = 20000; /* Maximum wait */ j = 20000; /* Maximum wait */
while (!(in_be16(&psc->mpc52xx_psc_status) & while (!(in_be16(&psc->mpc52xx_psc_status) &
......
...@@ -1916,6 +1916,16 @@ static void __exit exit_pmz(void) ...@@ -1916,6 +1916,16 @@ static void __exit exit_pmz(void)
#ifdef CONFIG_SERIAL_PMACZILOG_CONSOLE #ifdef CONFIG_SERIAL_PMACZILOG_CONSOLE
static void pmz_console_putchar(struct uart_port *port, int ch)
{
struct uart_pmac_port *uap = (struct uart_pmac_port *)port;
/* Wait for the transmit buffer to empty. */
while ((read_zsreg(uap, R0) & Tx_BUF_EMP) == 0)
udelay(5);
write_zsdata(uap, ch);
}
/* /*
* Print a string to the serial port trying not to disturb * Print a string to the serial port trying not to disturb
* any possible real use of the port... * any possible real use of the port...
...@@ -1924,7 +1934,6 @@ static void pmz_console_write(struct console *con, const char *s, unsigned int c ...@@ -1924,7 +1934,6 @@ static void pmz_console_write(struct console *con, const char *s, unsigned int c
{ {
struct uart_pmac_port *uap = &pmz_ports[con->index]; struct uart_pmac_port *uap = &pmz_ports[con->index];
unsigned long flags; unsigned long flags;
int i;
if (ZS_IS_ASLEEP(uap)) if (ZS_IS_ASLEEP(uap))
return; return;
...@@ -1934,17 +1943,7 @@ static void pmz_console_write(struct console *con, const char *s, unsigned int c ...@@ -1934,17 +1943,7 @@ static void pmz_console_write(struct console *con, const char *s, unsigned int c
write_zsreg(uap, R1, uap->curregs[1] & ~TxINT_ENAB); write_zsreg(uap, R1, uap->curregs[1] & ~TxINT_ENAB);
write_zsreg(uap, R5, uap->curregs[5] | TxENABLE | RTS | DTR); write_zsreg(uap, R5, uap->curregs[5] | TxENABLE | RTS | DTR);
for (i = 0; i < count; i++) { uart_console_write(&uap->port, s, count, pmz_console_putchar);
/* Wait for the transmit buffer to empty. */
while ((read_zsreg(uap, R0) & Tx_BUF_EMP) == 0)
udelay(5);
write_zsdata(uap, s[i]);
if (s[i] == 10) {
while ((read_zsreg(uap, R0) & Tx_BUF_EMP) == 0)
udelay(5);
write_zsdata(uap, R13);
}
}
/* Restore the values in the registers. */ /* Restore the values in the registers. */
write_zsreg(uap, R1, uap->curregs[1]); write_zsreg(uap, R1, uap->curregs[1]);
......
...@@ -619,6 +619,14 @@ static inline void wait_for_xmitr(struct uart_pxa_port *up) ...@@ -619,6 +619,14 @@ static inline void wait_for_xmitr(struct uart_pxa_port *up)
} }
} }
static void serial_pxa_console_putchar(struct uart_port *port, int ch)
{
struct uart_pxa_port *up = (struct uart_pxa_port *)port;
wait_for_xmitr(up);
serial_out(up, UART_TX, ch);
}
/* /*
* Print a string to the serial port trying not to disturb * Print a string to the serial port trying not to disturb
* any possible real use of the port... * any possible real use of the port...
...@@ -630,7 +638,6 @@ serial_pxa_console_write(struct console *co, const char *s, unsigned int count) ...@@ -630,7 +638,6 @@ serial_pxa_console_write(struct console *co, const char *s, unsigned int count)
{ {
struct uart_pxa_port *up = &serial_pxa_ports[co->index]; struct uart_pxa_port *up = &serial_pxa_ports[co->index];
unsigned int ier; unsigned int ier;
int i;
/* /*
* First save the IER then disable the interrupts * First save the IER then disable the interrupts
...@@ -638,22 +645,7 @@ serial_pxa_console_write(struct console *co, const char *s, unsigned int count) ...@@ -638,22 +645,7 @@ serial_pxa_console_write(struct console *co, const char *s, unsigned int count)
ier = serial_in(up, UART_IER); ier = serial_in(up, UART_IER);
serial_out(up, UART_IER, UART_IER_UUE); serial_out(up, UART_IER, UART_IER_UUE);
/* uart_console_write(&up->port, s, count, serial_pxa_console_putchar);
* Now, do each character
*/
for (i = 0; i < count; i++, s++) {
wait_for_xmitr(up);
/*
* Send the character out.
* If a LF, also do CR...
*/
serial_out(up, UART_TX, *s);
if (*s == 10) {
wait_for_xmitr(up);
serial_out(up, UART_TX, 13);
}
}
/* /*
* Finally, wait for transmitter to become empty * Finally, wait for transmitter to become empty
......
...@@ -1584,25 +1584,19 @@ s3c24xx_serial_console_txrdy(struct uart_port *port, unsigned int ufcon) ...@@ -1584,25 +1584,19 @@ s3c24xx_serial_console_txrdy(struct uart_port *port, unsigned int ufcon)
} }
static void static void
s3c24xx_serial_console_write(struct console *co, const char *s, s3c24xx_serial_console_putchar(struct uart_port *port, int ch)
unsigned int count)
{ {
int i;
unsigned int ufcon = rd_regl(cons_uart, S3C2410_UFCON); unsigned int ufcon = rd_regl(cons_uart, S3C2410_UFCON);
while (!s3c24xx_serial_console_txrdy(port, ufcon))
barrier();
wr_regb(cons_uart, S3C2410_UTXH, ch);
}
for (i = 0; i < count; i++) { static void
while (!s3c24xx_serial_console_txrdy(cons_uart, ufcon)) s3c24xx_serial_console_write(struct console *co, const char *s,
barrier(); unsigned int count)
{
wr_regb(cons_uart, S3C2410_UTXH, s[i]); uart_console_write(cons_uart, s, count, s3c24xx_serial_console_putchar);
if (s[i] == '\n') {
while (!s3c24xx_serial_console_txrdy(cons_uart, ufcon))
barrier();
wr_regb(cons_uart, S3C2410_UTXH, '\r');
}
}
} }
static void __init static void __init
......
...@@ -689,6 +689,14 @@ void __init sa1100_register_uart(int idx, int port) ...@@ -689,6 +689,14 @@ void __init sa1100_register_uart(int idx, int port)
#ifdef CONFIG_SERIAL_SA1100_CONSOLE #ifdef CONFIG_SERIAL_SA1100_CONSOLE
static void sa1100_console_putchar(struct uart_port *port, int ch)
{
struct sa1100_port *sport = (struct sa1100_port *)port;
while (!(UART_GET_UTSR1(sport) & UTSR1_TNF))
barrier();
UART_PUT_CHAR(sport, ch);
}
/* /*
* Interrupts are disabled on entering * Interrupts are disabled on entering
...@@ -697,7 +705,7 @@ static void ...@@ -697,7 +705,7 @@ static void
sa1100_console_write(struct console *co, const char *s, unsigned int count) sa1100_console_write(struct console *co, const char *s, unsigned int count)
{ {
struct sa1100_port *sport = &sa1100_ports[co->index]; struct sa1100_port *sport = &sa1100_ports[co->index];
unsigned int old_utcr3, status, i; unsigned int old_utcr3, status;
/* /*
* First, save UTCR3 and then disable interrupts * First, save UTCR3 and then disable interrupts
...@@ -706,21 +714,7 @@ sa1100_console_write(struct console *co, const char *s, unsigned int count) ...@@ -706,21 +714,7 @@ sa1100_console_write(struct console *co, const char *s, unsigned int count)
UART_PUT_UTCR3(sport, (old_utcr3 & ~(UTCR3_RIE | UTCR3_TIE)) | UART_PUT_UTCR3(sport, (old_utcr3 & ~(UTCR3_RIE | UTCR3_TIE)) |
UTCR3_TXE); UTCR3_TXE);
/* uart_console_write(&sport->port, s, count, sa1100_console_putchar);
* Now, do each character
*/
for (i = 0; i < count; i++) {
do {
status = UART_GET_UTSR1(sport);
} while (!(status & UTSR1_TNF));
UART_PUT_CHAR(sport, s[i]);
if (s[i] == '\n') {
do {
status = UART_GET_UTSR1(sport);
} while (!(status & UTSR1_TNF));
UART_PUT_CHAR(sport, '\r');
}
}
/* /*
* Finally, wait for transmitter to become empty * Finally, wait for transmitter to become empty
......
...@@ -1754,6 +1754,27 @@ static int uart_read_proc(char *page, char **start, off_t off, ...@@ -1754,6 +1754,27 @@ static int uart_read_proc(char *page, char **start, off_t off,
#endif #endif
#ifdef CONFIG_SERIAL_CORE_CONSOLE #ifdef CONFIG_SERIAL_CORE_CONSOLE
/*
* uart_console_write - write a console message to a serial port
* @port: the port to write the message
* @s: array of characters
* @count: number of characters in string to write
* @write: function to write character to port
*/
void uart_console_write(struct uart_port *port, const char *s,
unsigned int count,
void (*putchar)(struct uart_port *, int))
{
unsigned int i;
for (i = 0; i < count; i++, s++) {
if (*s == '\n')
putchar(port, '\r');
putchar(port, *s);
}
}
EXPORT_SYMBOL_GPL(uart_console_write);
/* /*
* Check whether an invalid uart number has been specified, and * Check whether an invalid uart number has been specified, and
* if so, search for the first available port that does have * if so, search for the first available port that does have
......
...@@ -543,6 +543,12 @@ static struct uart_port_lh7a40x lh7a40x_ports[DEV_NR] = { ...@@ -543,6 +543,12 @@ static struct uart_port_lh7a40x lh7a40x_ports[DEV_NR] = {
#else #else
# define LH7A40X_CONSOLE &lh7a40x_console # define LH7A40X_CONSOLE &lh7a40x_console
static void lh7a40xuart_console_putchar(struct uart_port *port, int ch)
{
while (UR(port, UART_R_STATUS) & nTxRdy)
;
UR(port, UART_R_DATA) = ch;
}
static void lh7a40xuart_console_write (struct console* co, static void lh7a40xuart_console_write (struct console* co,
const char* s, const char* s,
...@@ -556,16 +562,7 @@ static void lh7a40xuart_console_write (struct console* co, ...@@ -556,16 +562,7 @@ static void lh7a40xuart_console_write (struct console* co,
UR (port, UART_R_INTEN) = 0; /* Disable all interrupts */ UR (port, UART_R_INTEN) = 0; /* Disable all interrupts */
BIT_SET (port, UART_R_CON, UARTEN | SIRDIS); /* Enable UART */ BIT_SET (port, UART_R_CON, UARTEN | SIRDIS); /* Enable UART */
for (; count-- > 0; ++s) { uart_console_write(port, s, count, lh7a40xuart_console_putchar);
while (UR (port, UART_R_STATUS) & nTxRdy)
;
UR (port, UART_R_DATA) = *s;
if (*s == '\n') {
while ((UR (port, UART_R_STATUS) & TxBusy))
;
UR (port, UART_R_DATA) = '\r';
}
}
/* Wait until all characters are sent */ /* Wait until all characters are sent */
while (UR (port, UART_R_STATUS) & TxBusy) while (UR (port, UART_R_STATUS) & TxBusy)
......
...@@ -854,6 +854,14 @@ static inline void wait_for_xmitr(struct uart_txx9_port *up) ...@@ -854,6 +854,14 @@ static inline void wait_for_xmitr(struct uart_txx9_port *up)
} }
} }
static void serial_txx9_console_putchar(struct uart_port *port, int ch)
{
struct uart_txx9_port *up = (struct uart_txx9_port *)port;
wait_for_xmitr(up);
sio_out(up, TXX9_SITFIFO, ch);
}
/* /*
* Print a string to the serial port trying not to disturb * Print a string to the serial port trying not to disturb
* any possible real use of the port... * any possible real use of the port...
...@@ -865,7 +873,6 @@ serial_txx9_console_write(struct console *co, const char *s, unsigned int count) ...@@ -865,7 +873,6 @@ serial_txx9_console_write(struct console *co, const char *s, unsigned int count)
{ {
struct uart_txx9_port *up = &serial_txx9_ports[co->index]; struct uart_txx9_port *up = &serial_txx9_ports[co->index];
unsigned int ier, flcr; unsigned int ier, flcr;
int i;
/* /*
* First save the UER then disable the interrupts * First save the UER then disable the interrupts
...@@ -879,22 +886,7 @@ serial_txx9_console_write(struct console *co, const char *s, unsigned int count) ...@@ -879,22 +886,7 @@ serial_txx9_console_write(struct console *co, const char *s, unsigned int count)
if (!(up->port.flags & UPF_CONS_FLOW) && (flcr & TXX9_SIFLCR_TES)) if (!(up->port.flags & UPF_CONS_FLOW) && (flcr & TXX9_SIFLCR_TES))
sio_out(up, TXX9_SIFLCR, flcr & ~TXX9_SIFLCR_TES); sio_out(up, TXX9_SIFLCR, flcr & ~TXX9_SIFLCR_TES);
/* uart_console_write(&up->port, s, count, serial_txx9_console_putchar);
* Now, do each character
*/
for (i = 0; i < count; i++, s++) {
wait_for_xmitr(up);
/*
* Send the character out.
* If a LF, also do CR...
*/
sio_out(up, TXX9_SITFIFO, *s);
if (*s == 10) {
wait_for_xmitr(up);
sio_out(up, TXX9_SITFIFO, 13);
}
}
/* /*
* Finally, wait for transmitter to become empty * Finally, wait for transmitter to become empty
......
...@@ -861,8 +861,9 @@ static int num_channels; ...@@ -861,8 +861,9 @@ static int num_channels;
#ifdef CONFIG_SERIAL_SUNSAB_CONSOLE #ifdef CONFIG_SERIAL_SUNSAB_CONSOLE
static __inline__ void sunsab_console_putchar(struct uart_sunsab_port *up, char c) static void sunsab_console_putchar(struct uart_port *port, int c)
{ {
struct uart_sunsab_port *up = (struct uart_sunsab_port *)port;
unsigned long flags; unsigned long flags;
spin_lock_irqsave(&up->port.lock, flags); spin_lock_irqsave(&up->port.lock, flags);
...@@ -876,13 +877,8 @@ static __inline__ void sunsab_console_putchar(struct uart_sunsab_port *up, char ...@@ -876,13 +877,8 @@ static __inline__ void sunsab_console_putchar(struct uart_sunsab_port *up, char
static void sunsab_console_write(struct console *con, const char *s, unsigned n) static void sunsab_console_write(struct console *con, const char *s, unsigned n)
{ {
struct uart_sunsab_port *up = &sunsab_ports[con->index]; struct uart_sunsab_port *up = &sunsab_ports[con->index];
int i;
for (i = 0; i < n; i++) { uart_console_write(&up->port, s, n, sunsab_console_putchar);
if (*s == '\n')
sunsab_console_putchar(up, '\r');
sunsab_console_putchar(up, *s++);
}
sunsab_tec_wait(up); sunsab_tec_wait(up);
} }
......
...@@ -1376,6 +1376,14 @@ static __inline__ void wait_for_xmitr(struct uart_sunsu_port *up) ...@@ -1376,6 +1376,14 @@ static __inline__ void wait_for_xmitr(struct uart_sunsu_port *up)
} }
} }
static void sunsu_console_putchar(struct uart_port *port, int ch)
{
struct uart_sunsu_port *up = (struct uart_sunsu_port *)port;
wait_for_xmitr(up);
serial_out(up, UART_TX, ch);
}
/* /*
* Print a string to the serial port trying not to disturb * Print a string to the serial port trying not to disturb
* any possible real use of the port... * any possible real use of the port...
...@@ -1385,7 +1393,6 @@ static void sunsu_console_write(struct console *co, const char *s, ...@@ -1385,7 +1393,6 @@ static void sunsu_console_write(struct console *co, const char *s,
{ {
struct uart_sunsu_port *up = &sunsu_ports[co->index]; struct uart_sunsu_port *up = &sunsu_ports[co->index];
unsigned int ier; unsigned int ier;
int i;
/* /*
* First save the UER then disable the interrupts * First save the UER then disable the interrupts
...@@ -1393,22 +1400,7 @@ static void sunsu_console_write(struct console *co, const char *s, ...@@ -1393,22 +1400,7 @@ static void sunsu_console_write(struct console *co, const char *s,
ier = serial_in(up, UART_IER); ier = serial_in(up, UART_IER);
serial_out(up, UART_IER, 0); serial_out(up, UART_IER, 0);
/* uart_console_write(&up->port, s, count, sunsu_console_putchar);
* Now, do each character
*/
for (i = 0; i < count; i++, s++) {
wait_for_xmitr(up);
/*
* Send the character out.
* If a LF, also do CR...
*/
serial_out(up, UART_TX, *s);
if (*s == 10) {
wait_for_xmitr(up);
serial_out(up, UART_TX, 13);
}
}
/* /*
* Finally, wait for transmitter to become empty * Finally, wait for transmitter to become empty
......
...@@ -1252,8 +1252,9 @@ static struct zilog_layout __iomem * __init get_zs(int chip, int node) ...@@ -1252,8 +1252,9 @@ static struct zilog_layout __iomem * __init get_zs(int chip, int node)
#define ZS_PUT_CHAR_MAX_DELAY 2000 /* 10 ms */ #define ZS_PUT_CHAR_MAX_DELAY 2000 /* 10 ms */
static void sunzilog_put_char(struct zilog_channel __iomem *channel, unsigned char ch) static void sunzilog_putchar(struct uart_port *port, int ch)
{ {
struct zilog_channel *channel = ZILOG_CHANNEL_FROM_PORT(port);
int loops = ZS_PUT_CHAR_MAX_DELAY; int loops = ZS_PUT_CHAR_MAX_DELAY;
/* This is a timed polling loop so do not switch the explicit /* This is a timed polling loop so do not switch the explicit
...@@ -1284,7 +1285,7 @@ static int sunzilog_serio_write(struct serio *serio, unsigned char ch) ...@@ -1284,7 +1285,7 @@ static int sunzilog_serio_write(struct serio *serio, unsigned char ch)
spin_lock_irqsave(&sunzilog_serio_lock, flags); spin_lock_irqsave(&sunzilog_serio_lock, flags);
sunzilog_put_char(ZILOG_CHANNEL_FROM_PORT(&up->port), ch); sunzilog_putchar(&up->port, ch);
spin_unlock_irqrestore(&sunzilog_serio_lock, flags); spin_unlock_irqrestore(&sunzilog_serio_lock, flags);
...@@ -1325,16 +1326,10 @@ static void ...@@ -1325,16 +1326,10 @@ static void
sunzilog_console_write(struct console *con, const char *s, unsigned int count) sunzilog_console_write(struct console *con, const char *s, unsigned int count)
{ {
struct uart_sunzilog_port *up = &sunzilog_port_table[con->index]; struct uart_sunzilog_port *up = &sunzilog_port_table[con->index];
struct zilog_channel *channel = ZILOG_CHANNEL_FROM_PORT(&up->port);
unsigned long flags; unsigned long flags;
int i;
spin_lock_irqsave(&up->port.lock, flags); spin_lock_irqsave(&up->port.lock, flags);
for (i = 0; i < count; i++, s++) { uart_console_write(&up->port, s, count, sunzilog_putchar);
sunzilog_put_char(channel, *s);
if (*s == 10)
sunzilog_put_char(channel, 13);
}
udelay(2); udelay(2);
spin_unlock_irqrestore(&up->port.lock, flags); spin_unlock_irqrestore(&up->port.lock, flags);
} }
......
...@@ -821,25 +821,23 @@ static void wait_for_xmitr(struct uart_port *port) ...@@ -821,25 +821,23 @@ static void wait_for_xmitr(struct uart_port *port)
} }
} }
static void siu_console_putchar(struct uart_port *port, int ch)
{
wait_for_xmitr(port);
siu_write(port, UART_TX, ch);
}
static void siu_console_write(struct console *con, const char *s, unsigned count) static void siu_console_write(struct console *con, const char *s, unsigned count)
{ {
struct uart_port *port; struct uart_port *port;
uint8_t ier; uint8_t ier;
unsigned i;
port = &siu_uart_ports[con->index]; port = &siu_uart_ports[con->index];
ier = siu_read(port, UART_IER); ier = siu_read(port, UART_IER);
siu_write(port, UART_IER, 0); siu_write(port, UART_IER, 0);
for (i = 0; i < count && *s != '\0'; i++, s++) { uart_console_write(port, s, count, siu_console_putchar);
wait_for_xmitr(port);
siu_write(port, UART_TX, *s);
if (*s == '\n') {
wait_for_xmitr(port);
siu_write(port, UART_TX, '\r');
}
}
wait_for_xmitr(port); wait_for_xmitr(port);
siu_write(port, UART_IER, ier); siu_write(port, UART_IER, ier);
......
...@@ -366,6 +366,9 @@ void uart_parse_options(char *options, int *baud, int *parity, int *bits, ...@@ -366,6 +366,9 @@ void uart_parse_options(char *options, int *baud, int *parity, int *bits,
int uart_set_options(struct uart_port *port, struct console *co, int baud, int uart_set_options(struct uart_port *port, struct console *co, int baud,
int parity, int bits, int flow); int parity, int bits, int flow);
struct tty_driver *uart_console_device(struct console *co, int *index); struct tty_driver *uart_console_device(struct console *co, int *index);
void uart_console_write(struct uart_port *port, const char *s,
unsigned int count,
void (*putchar)(struct uart_port *, int));
/* /*
* Port/driver registration/removal * Port/driver registration/removal
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment