mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2024-11-16 16:54:20 +08:00
aa4148cfc7
Also fixes all serial drivers. Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
549 lines
15 KiB
C
549 lines
15 KiB
C
/*
|
||
* drivers/serial/v850e_uart.c -- Serial I/O using V850E on-chip UART or UARTB
|
||
*
|
||
* Copyright (C) 2001,02,03 NEC Electronics Corporation
|
||
* Copyright (C) 2001,02,03 Miles Bader <miles@gnu.org>
|
||
*
|
||
* This file is subject to the terms and conditions of the GNU General
|
||
* Public License. See the file COPYING in the main directory of this
|
||
* archive for more details.
|
||
*
|
||
* Written by Miles Bader <miles@gnu.org>
|
||
*/
|
||
|
||
/* This driver supports both the original V850E UART interface (called
|
||
merely `UART' in the docs) and the newer `UARTB' interface, which is
|
||
roughly a superset of the first one. The selection is made at
|
||
configure time -- if CONFIG_V850E_UARTB is defined, then UARTB is
|
||
presumed, otherwise the old UART -- as these are on-CPU UARTS, a system
|
||
can never have both.
|
||
|
||
The UARTB interface also has a 16-entry FIFO mode, which is not
|
||
yet supported by this driver. */
|
||
|
||
#include <linux/kernel.h>
|
||
#include <linux/init.h>
|
||
#include <linux/module.h>
|
||
#include <linux/console.h>
|
||
#include <linux/tty.h>
|
||
#include <linux/tty_flip.h>
|
||
#include <linux/serial.h>
|
||
#include <linux/serial_core.h>
|
||
|
||
#include <asm/v850e_uart.h>
|
||
|
||
/* Initial UART state. This may be overridden by machine-dependent headers. */
|
||
#ifndef V850E_UART_INIT_BAUD
|
||
#define V850E_UART_INIT_BAUD 115200
|
||
#endif
|
||
#ifndef V850E_UART_INIT_CFLAGS
|
||
#define V850E_UART_INIT_CFLAGS (B115200 | CS8 | CREAD)
|
||
#endif
|
||
|
||
/* A string used for prefixing printed descriptions; since the same UART
|
||
macro is actually used on other chips than the V850E. This must be a
|
||
constant string. */
|
||
#ifndef V850E_UART_CHIP_NAME
|
||
#define V850E_UART_CHIP_NAME "V850E"
|
||
#endif
|
||
|
||
#define V850E_UART_MINOR_BASE 64 /* First tty minor number */
|
||
|
||
|
||
/* Low-level UART functions. */
|
||
|
||
/* Configure and turn on uart channel CHAN, using the termios `control
|
||
modes' bits in CFLAGS, and a baud-rate of BAUD. */
|
||
void v850e_uart_configure (unsigned chan, unsigned cflags, unsigned baud)
|
||
{
|
||
int flags;
|
||
v850e_uart_speed_t old_speed;
|
||
v850e_uart_config_t old_config;
|
||
v850e_uart_speed_t new_speed = v850e_uart_calc_speed (baud);
|
||
v850e_uart_config_t new_config = v850e_uart_calc_config (cflags);
|
||
|
||
/* Disable interrupts while we're twiddling the hardware. */
|
||
local_irq_save (flags);
|
||
|
||
#ifdef V850E_UART_PRE_CONFIGURE
|
||
V850E_UART_PRE_CONFIGURE (chan, cflags, baud);
|
||
#endif
|
||
|
||
old_config = V850E_UART_CONFIG (chan);
|
||
old_speed = v850e_uart_speed (chan);
|
||
|
||
if (! v850e_uart_speed_eq (old_speed, new_speed)) {
|
||
/* The baud rate has changed. First, disable the UART. */
|
||
V850E_UART_CONFIG (chan) = V850E_UART_CONFIG_FINI;
|
||
old_config = 0; /* Force the uart to be re-initialized. */
|
||
|
||
/* Reprogram the baud-rate generator. */
|
||
v850e_uart_set_speed (chan, new_speed);
|
||
}
|
||
|
||
if (! (old_config & V850E_UART_CONFIG_ENABLED)) {
|
||
/* If we are using the uart for the first time, start by
|
||
enabling it, which must be done before turning on any
|
||
other bits. */
|
||
V850E_UART_CONFIG (chan) = V850E_UART_CONFIG_INIT;
|
||
/* See the initial state. */
|
||
old_config = V850E_UART_CONFIG (chan);
|
||
}
|
||
|
||
if (new_config != old_config) {
|
||
/* Which of the TXE/RXE bits we'll temporarily turn off
|
||
before changing other control bits. */
|
||
unsigned temp_disable = 0;
|
||
/* Which of the TXE/RXE bits will be enabled. */
|
||
unsigned enable = 0;
|
||
unsigned changed_bits = new_config ^ old_config;
|
||
|
||
/* Which of RX/TX will be enabled in the new configuration. */
|
||
if (new_config & V850E_UART_CONFIG_RX_BITS)
|
||
enable |= (new_config & V850E_UART_CONFIG_RX_ENABLE);
|
||
if (new_config & V850E_UART_CONFIG_TX_BITS)
|
||
enable |= (new_config & V850E_UART_CONFIG_TX_ENABLE);
|
||
|
||
/* Figure out which of RX/TX needs to be disabled; note
|
||
that this will only happen if they're not already
|
||
disabled. */
|
||
if (changed_bits & V850E_UART_CONFIG_RX_BITS)
|
||
temp_disable
|
||
|= (old_config & V850E_UART_CONFIG_RX_ENABLE);
|
||
if (changed_bits & V850E_UART_CONFIG_TX_BITS)
|
||
temp_disable
|
||
|= (old_config & V850E_UART_CONFIG_TX_ENABLE);
|
||
|
||
/* We have to turn off RX and/or TX mode before changing
|
||
any associated control bits. */
|
||
if (temp_disable)
|
||
V850E_UART_CONFIG (chan) = old_config & ~temp_disable;
|
||
|
||
/* Write the new control bits, while RX/TX are disabled. */
|
||
if (changed_bits & ~enable)
|
||
V850E_UART_CONFIG (chan) = new_config & ~enable;
|
||
|
||
v850e_uart_config_delay (new_config, new_speed);
|
||
|
||
/* Write the final version, with enable bits turned on. */
|
||
V850E_UART_CONFIG (chan) = new_config;
|
||
}
|
||
|
||
local_irq_restore (flags);
|
||
}
|
||
|
||
|
||
/* Low-level console. */
|
||
|
||
#ifdef CONFIG_V850E_UART_CONSOLE
|
||
|
||
static void v850e_uart_cons_write (struct console *co,
|
||
const char *s, unsigned count)
|
||
{
|
||
if (count > 0) {
|
||
unsigned chan = co->index;
|
||
unsigned irq = V850E_UART_TX_IRQ (chan);
|
||
int irq_was_enabled, irq_was_pending, flags;
|
||
|
||
/* We don't want to get `transmission completed'
|
||
interrupts, since we're busy-waiting, so we disable them
|
||
while sending (we don't disable interrupts entirely
|
||
because sending over a serial line is really slow). We
|
||
save the status of the tx interrupt and restore it when
|
||
we're done so that using printk doesn't interfere with
|
||
normal serial transmission (other than interleaving the
|
||
output, of course!). This should work correctly even if
|
||
this function is interrupted and the interrupt printks
|
||
something. */
|
||
|
||
/* Disable interrupts while fiddling with tx interrupt. */
|
||
local_irq_save (flags);
|
||
/* Get current tx interrupt status. */
|
||
irq_was_enabled = v850e_intc_irq_enabled (irq);
|
||
irq_was_pending = v850e_intc_irq_pending (irq);
|
||
/* Disable tx interrupt if necessary. */
|
||
if (irq_was_enabled)
|
||
v850e_intc_disable_irq (irq);
|
||
/* Turn interrupts back on. */
|
||
local_irq_restore (flags);
|
||
|
||
/* Send characters. */
|
||
while (count > 0) {
|
||
int ch = *s++;
|
||
|
||
if (ch == '\n') {
|
||
/* We don't have the benefit of a tty
|
||
driver, so translate NL into CR LF. */
|
||
v850e_uart_wait_for_xmit_ok (chan);
|
||
v850e_uart_putc (chan, '\r');
|
||
}
|
||
|
||
v850e_uart_wait_for_xmit_ok (chan);
|
||
v850e_uart_putc (chan, ch);
|
||
|
||
count--;
|
||
}
|
||
|
||
/* Restore saved tx interrupt status. */
|
||
if (irq_was_enabled) {
|
||
/* Wait for the last character we sent to be
|
||
completely transmitted (as we'll get an
|
||
interrupt interrupt at that point). */
|
||
v850e_uart_wait_for_xmit_done (chan);
|
||
/* Clear pending interrupts received due
|
||
to our transmission, unless there was already
|
||
one pending, in which case we want the
|
||
handler to be called. */
|
||
if (! irq_was_pending)
|
||
v850e_intc_clear_pending_irq (irq);
|
||
/* ... and then turn back on handling. */
|
||
v850e_intc_enable_irq (irq);
|
||
}
|
||
}
|
||
}
|
||
|
||
extern struct uart_driver v850e_uart_driver;
|
||
static struct console v850e_uart_cons =
|
||
{
|
||
.name = "ttyS",
|
||
.write = v850e_uart_cons_write,
|
||
.device = uart_console_device,
|
||
.flags = CON_PRINTBUFFER,
|
||
.cflag = V850E_UART_INIT_CFLAGS,
|
||
.index = -1,
|
||
.data = &v850e_uart_driver,
|
||
};
|
||
|
||
void v850e_uart_cons_init (unsigned chan)
|
||
{
|
||
v850e_uart_configure (chan, V850E_UART_INIT_CFLAGS,
|
||
V850E_UART_INIT_BAUD);
|
||
v850e_uart_cons.index = chan;
|
||
register_console (&v850e_uart_cons);
|
||
printk ("Console: %s on-chip UART channel %d\n",
|
||
V850E_UART_CHIP_NAME, chan);
|
||
}
|
||
|
||
/* This is what the init code actually calls. */
|
||
static int v850e_uart_console_init (void)
|
||
{
|
||
v850e_uart_cons_init (V850E_UART_CONSOLE_CHANNEL);
|
||
return 0;
|
||
}
|
||
console_initcall(v850e_uart_console_init);
|
||
|
||
#define V850E_UART_CONSOLE &v850e_uart_cons
|
||
|
||
#else /* !CONFIG_V850E_UART_CONSOLE */
|
||
#define V850E_UART_CONSOLE 0
|
||
#endif /* CONFIG_V850E_UART_CONSOLE */
|
||
|
||
/* TX/RX interrupt handlers. */
|
||
|
||
static void v850e_uart_stop_tx (struct uart_port *port);
|
||
|
||
void v850e_uart_tx (struct uart_port *port)
|
||
{
|
||
struct circ_buf *xmit = &port->info->xmit;
|
||
int stopped = uart_tx_stopped (port);
|
||
|
||
if (v850e_uart_xmit_ok (port->line)) {
|
||
int tx_ch;
|
||
|
||
if (port->x_char) {
|
||
tx_ch = port->x_char;
|
||
port->x_char = 0;
|
||
} else if (!uart_circ_empty (xmit) && !stopped) {
|
||
tx_ch = xmit->buf[xmit->tail];
|
||
xmit->tail = (xmit->tail + 1) & (UART_XMIT_SIZE - 1);
|
||
} else
|
||
goto no_xmit;
|
||
|
||
v850e_uart_putc (port->line, tx_ch);
|
||
port->icount.tx++;
|
||
|
||
if (uart_circ_chars_pending (xmit) < WAKEUP_CHARS)
|
||
uart_write_wakeup (port);
|
||
}
|
||
|
||
no_xmit:
|
||
if (uart_circ_empty (xmit) || stopped)
|
||
v850e_uart_stop_tx (port, stopped);
|
||
}
|
||
|
||
static irqreturn_t v850e_uart_tx_irq(int irq, void *data, struct pt_regs *regs)
|
||
{
|
||
struct uart_port *port = data;
|
||
v850e_uart_tx (port);
|
||
return IRQ_HANDLED;
|
||
}
|
||
|
||
static irqreturn_t v850e_uart_rx_irq(int irq, void *data, struct pt_regs *regs)
|
||
{
|
||
struct uart_port *port = data;
|
||
unsigned ch_stat = TTY_NORMAL;
|
||
unsigned ch = v850e_uart_getc (port->line);
|
||
unsigned err = v850e_uart_err (port->line);
|
||
|
||
if (err) {
|
||
if (err & V850E_UART_ERR_OVERRUN) {
|
||
ch_stat = TTY_OVERRUN;
|
||
port->icount.overrun++;
|
||
} else if (err & V850E_UART_ERR_FRAME) {
|
||
ch_stat = TTY_FRAME;
|
||
port->icount.frame++;
|
||
} else if (err & V850E_UART_ERR_PARITY) {
|
||
ch_stat = TTY_PARITY;
|
||
port->icount.parity++;
|
||
}
|
||
}
|
||
|
||
port->icount.rx++;
|
||
|
||
tty_insert_flip_char (port->info->tty, ch, ch_stat);
|
||
tty_schedule_flip (port->info->tty);
|
||
|
||
return IRQ_HANDLED;
|
||
}
|
||
|
||
|
||
/* Control functions for the serial framework. */
|
||
|
||
static void v850e_uart_nop (struct uart_port *port) { }
|
||
static int v850e_uart_success (struct uart_port *port) { return 0; }
|
||
|
||
static unsigned v850e_uart_tx_empty (struct uart_port *port)
|
||
{
|
||
return TIOCSER_TEMT; /* Can't detect. */
|
||
}
|
||
|
||
static void v850e_uart_set_mctrl (struct uart_port *port, unsigned mctrl)
|
||
{
|
||
#ifdef V850E_UART_SET_RTS
|
||
V850E_UART_SET_RTS (port->line, (mctrl & TIOCM_RTS));
|
||
#endif
|
||
}
|
||
|
||
static unsigned v850e_uart_get_mctrl (struct uart_port *port)
|
||
{
|
||
/* We don't support DCD or DSR, so consider them permanently active. */
|
||
int mctrl = TIOCM_CAR | TIOCM_DSR;
|
||
|
||
/* We may support CTS. */
|
||
#ifdef V850E_UART_CTS
|
||
mctrl |= V850E_UART_CTS(port->line) ? TIOCM_CTS : 0;
|
||
#else
|
||
mctrl |= TIOCM_CTS;
|
||
#endif
|
||
|
||
return mctrl;
|
||
}
|
||
|
||
static void v850e_uart_start_tx (struct uart_port *port)
|
||
{
|
||
v850e_intc_disable_irq (V850E_UART_TX_IRQ (port->line));
|
||
v850e_uart_tx (port);
|
||
v850e_intc_enable_irq (V850E_UART_TX_IRQ (port->line));
|
||
}
|
||
|
||
static void v850e_uart_stop_tx (struct uart_port *port)
|
||
{
|
||
v850e_intc_disable_irq (V850E_UART_TX_IRQ (port->line));
|
||
}
|
||
|
||
static void v850e_uart_start_rx (struct uart_port *port)
|
||
{
|
||
v850e_intc_enable_irq (V850E_UART_RX_IRQ (port->line));
|
||
}
|
||
|
||
static void v850e_uart_stop_rx (struct uart_port *port)
|
||
{
|
||
v850e_intc_disable_irq (V850E_UART_RX_IRQ (port->line));
|
||
}
|
||
|
||
static void v850e_uart_break_ctl (struct uart_port *port, int break_ctl)
|
||
{
|
||
/* Umm, do this later. */
|
||
}
|
||
|
||
static int v850e_uart_startup (struct uart_port *port)
|
||
{
|
||
int err;
|
||
|
||
/* Alloc RX irq. */
|
||
err = request_irq (V850E_UART_RX_IRQ (port->line), v850e_uart_rx_irq,
|
||
SA_INTERRUPT, "v850e_uart", port);
|
||
if (err)
|
||
return err;
|
||
|
||
/* Alloc TX irq. */
|
||
err = request_irq (V850E_UART_TX_IRQ (port->line), v850e_uart_tx_irq,
|
||
SA_INTERRUPT, "v850e_uart", port);
|
||
if (err) {
|
||
free_irq (V850E_UART_RX_IRQ (port->line), port);
|
||
return err;
|
||
}
|
||
|
||
v850e_uart_start_rx (port);
|
||
|
||
return 0;
|
||
}
|
||
|
||
static void v850e_uart_shutdown (struct uart_port *port)
|
||
{
|
||
/* Disable port interrupts. */
|
||
free_irq (V850E_UART_TX_IRQ (port->line), port);
|
||
free_irq (V850E_UART_RX_IRQ (port->line), port);
|
||
|
||
/* Turn off xmit/recv enable bits. */
|
||
V850E_UART_CONFIG (port->line)
|
||
&= ~(V850E_UART_CONFIG_TX_ENABLE
|
||
| V850E_UART_CONFIG_RX_ENABLE);
|
||
/* Then reset the channel. */
|
||
V850E_UART_CONFIG (port->line) = 0;
|
||
}
|
||
|
||
static void
|
||
v850e_uart_set_termios (struct uart_port *port, struct termios *termios,
|
||
struct termios *old)
|
||
{
|
||
unsigned cflags = termios->c_cflag;
|
||
|
||
/* Restrict flags to legal values. */
|
||
if ((cflags & CSIZE) != CS7 && (cflags & CSIZE) != CS8)
|
||
/* The new value of CSIZE is invalid, use the old value. */
|
||
cflags = (cflags & ~CSIZE)
|
||
| (old ? (old->c_cflag & CSIZE) : CS8);
|
||
|
||
termios->c_cflag = cflags;
|
||
|
||
v850e_uart_configure (port->line, cflags,
|
||
uart_get_baud_rate (port, termios, old,
|
||
v850e_uart_min_baud(),
|
||
v850e_uart_max_baud()));
|
||
}
|
||
|
||
static const char *v850e_uart_type (struct uart_port *port)
|
||
{
|
||
return port->type == PORT_V850E_UART ? "v850e_uart" : 0;
|
||
}
|
||
|
||
static void v850e_uart_config_port (struct uart_port *port, int flags)
|
||
{
|
||
if (flags & UART_CONFIG_TYPE)
|
||
port->type = PORT_V850E_UART;
|
||
}
|
||
|
||
static int
|
||
v850e_uart_verify_port (struct uart_port *port, struct serial_struct *ser)
|
||
{
|
||
if (ser->type != PORT_UNKNOWN && ser->type != PORT_V850E_UART)
|
||
return -EINVAL;
|
||
if (ser->irq != V850E_UART_TX_IRQ (port->line))
|
||
return -EINVAL;
|
||
return 0;
|
||
}
|
||
|
||
static struct uart_ops v850e_uart_ops = {
|
||
.tx_empty = v850e_uart_tx_empty,
|
||
.get_mctrl = v850e_uart_get_mctrl,
|
||
.set_mctrl = v850e_uart_set_mctrl,
|
||
.start_tx = v850e_uart_start_tx,
|
||
.stop_tx = v850e_uart_stop_tx,
|
||
.stop_rx = v850e_uart_stop_rx,
|
||
.enable_ms = v850e_uart_nop,
|
||
.break_ctl = v850e_uart_break_ctl,
|
||
.startup = v850e_uart_startup,
|
||
.shutdown = v850e_uart_shutdown,
|
||
.set_termios = v850e_uart_set_termios,
|
||
.type = v850e_uart_type,
|
||
.release_port = v850e_uart_nop,
|
||
.request_port = v850e_uart_success,
|
||
.config_port = v850e_uart_config_port,
|
||
.verify_port = v850e_uart_verify_port,
|
||
};
|
||
|
||
/* Initialization and cleanup. */
|
||
|
||
static struct uart_driver v850e_uart_driver = {
|
||
.owner = THIS_MODULE,
|
||
.driver_name = "v850e_uart",
|
||
.dev_name = "ttyS",
|
||
.major = TTY_MAJOR,
|
||
.minor = V850E_UART_MINOR_BASE,
|
||
.nr = V850E_UART_NUM_CHANNELS,
|
||
.cons = V850E_UART_CONSOLE,
|
||
};
|
||
|
||
|
||
static struct uart_port v850e_uart_ports[V850E_UART_NUM_CHANNELS];
|
||
|
||
static int __init v850e_uart_init (void)
|
||
{
|
||
int rval;
|
||
|
||
printk (KERN_INFO "%s on-chip UART\n", V850E_UART_CHIP_NAME);
|
||
|
||
rval = uart_register_driver (&v850e_uart_driver);
|
||
if (rval == 0) {
|
||
unsigned chan;
|
||
|
||
for (chan = 0; chan < V850E_UART_NUM_CHANNELS; chan++) {
|
||
struct uart_port *port = &v850e_uart_ports[chan];
|
||
|
||
memset (port, 0, sizeof *port);
|
||
|
||
port->ops = &v850e_uart_ops;
|
||
port->line = chan;
|
||
port->iotype = UPIO_MEM;
|
||
port->flags = UPF_BOOT_AUTOCONF;
|
||
|
||
/* We actually use multiple IRQs, but the serial
|
||
framework seems to mainly use this for
|
||
informational purposes anyway. Here we use the TX
|
||
irq. */
|
||
port->irq = V850E_UART_TX_IRQ (chan);
|
||
|
||
/* The serial framework doesn't really use these
|
||
membase/mapbase fields for anything useful, but
|
||
it requires that they be something non-zero to
|
||
consider the port `valid', and also uses them
|
||
for informational purposes. */
|
||
port->membase = (void *)V850E_UART_BASE_ADDR (chan);
|
||
port->mapbase = V850E_UART_BASE_ADDR (chan);
|
||
|
||
/* The framework insists on knowing the uart's master
|
||
clock freq, though it doesn't seem to do anything
|
||
useful for us with it. We must make it at least
|
||
higher than (the maximum baud rate * 16), otherwise
|
||
the framework will puke during its internal
|
||
calculations, and force the baud rate to be 9600.
|
||
To be accurate though, just repeat the calculation
|
||
we use when actually setting the speed. */
|
||
port->uartclk = v850e_uart_max_clock() * 16;
|
||
|
||
uart_add_one_port (&v850e_uart_driver, port);
|
||
}
|
||
}
|
||
|
||
return rval;
|
||
}
|
||
|
||
static void __exit v850e_uart_exit (void)
|
||
{
|
||
unsigned chan;
|
||
|
||
for (chan = 0; chan < V850E_UART_NUM_CHANNELS; chan++)
|
||
uart_remove_one_port (&v850e_uart_driver,
|
||
&v850e_uart_ports[chan]);
|
||
|
||
uart_unregister_driver (&v850e_uart_driver);
|
||
}
|
||
|
||
module_init (v850e_uart_init);
|
||
module_exit (v850e_uart_exit);
|
||
|
||
MODULE_AUTHOR ("Miles Bader");
|
||
MODULE_DESCRIPTION ("NEC " V850E_UART_CHIP_NAME " on-chip UART");
|
||
MODULE_LICENSE ("GPL");
|