// SPDX-License-Identifier: GPL-2.0 /* * Derived from many drivers using generic_serial interface. * * Copyright (C) 2008 Maxime Bizon <mbizon@freebox.fr> * * Serial driver for BCM63xx integrated UART. * * Hardware flow control was _not_ tested since I only have RX/TX on * my board.
*/
/* * serial core request to set RTS and DTR pin state and loopback mode
*/ staticvoid bcm_uart_set_mctrl(struct uart_port *port, unsignedint mctrl)
{ unsignedint val;
val = bcm_uart_readl(port, UART_MCTL_REG);
val &= ~(UART_MCTL_DTR_MASK | UART_MCTL_RTS_MASK); /* invert of written value is reflected on the pin */ if (!(mctrl & TIOCM_DTR))
val |= UART_MCTL_DTR_MASK; if (!(mctrl & TIOCM_RTS))
val |= UART_MCTL_RTS_MASK;
bcm_uart_writel(port, val, UART_MCTL_REG);
val = bcm_uart_readl(port, UART_CTL_REG); if (mctrl & TIOCM_LOOP)
val |= UART_CTL_LOOPBACK_MASK; else
val &= ~UART_CTL_LOOPBACK_MASK;
bcm_uart_writel(port, val, UART_CTL_REG);
}
/* * serial core request to return RI, CTS, DCD and DSR pin state
*/ staticunsignedint bcm_uart_get_mctrl(struct uart_port *port)
{ unsignedint val, mctrl;
mctrl = 0;
val = bcm_uart_readl(port, UART_EXTINP_REG); if (val & UART_EXTINP_RI_MASK)
mctrl |= TIOCM_RI; if (val & UART_EXTINP_CTS_MASK)
mctrl |= TIOCM_CTS; if (val & UART_EXTINP_DCD_MASK)
mctrl |= TIOCM_CD; if (val & UART_EXTINP_DSR_MASK)
mctrl |= TIOCM_DSR; return mctrl;
}
/* * serial core request to disable tx ASAP (used for flow control)
*/ staticvoid bcm_uart_stop_tx(struct uart_port *port)
{ unsignedint val;
val = bcm_uart_readl(port, UART_CTL_REG);
val &= ~(UART_CTL_TXEN_MASK);
bcm_uart_writel(port, val, UART_CTL_REG);
val = bcm_uart_readl(port, UART_IR_REG);
val &= ~UART_TX_INT_MASK;
bcm_uart_writel(port, val, UART_IR_REG);
}
/* * serial core request to (re)enable tx
*/ staticvoid bcm_uart_start_tx(struct uart_port *port)
{ unsignedint val;
val = bcm_uart_readl(port, UART_IR_REG);
val |= UART_TX_INT_MASK;
bcm_uart_writel(port, val, UART_IR_REG);
val = bcm_uart_readl(port, UART_CTL_REG);
val |= UART_CTL_TXEN_MASK;
bcm_uart_writel(port, val, UART_CTL_REG);
}
/* * serial core request to stop rx, called before port shutdown
*/ staticvoid bcm_uart_stop_rx(struct uart_port *port)
{ unsignedint val;
val = bcm_uart_readl(port, UART_IR_REG);
val &= ~UART_RX_INT_MASK;
bcm_uart_writel(port, val, UART_IR_REG);
}
/* * serial core request to enable modem status interrupt reporting
*/ staticvoid bcm_uart_enable_ms(struct uart_port *port)
{ unsignedint val;
val = bcm_uart_readl(port, UART_IR_REG);
val |= UART_IR_MASK(UART_IR_EXTIP);
bcm_uart_writel(port, val, UART_IR_REG);
}
/* * serial core request to start/stop emitting break char
*/ staticvoid bcm_uart_break_ctl(struct uart_port *port, int ctl)
{ unsignedlong flags; unsignedint val;
uart_port_lock_irqsave(port, &flags);
val = bcm_uart_readl(port, UART_CTL_REG); if (ctl)
val |= UART_CTL_XMITBRK_MASK; else
val &= ~UART_CTL_XMITBRK_MASK;
bcm_uart_writel(port, val, UART_CTL_REG);
uart_port_unlock_irqrestore(port, flags);
}
/* * return port type in string format
*/ staticconstchar *bcm_uart_type(struct uart_port *port)
{ return (port->type == PORT_BCM63XX) ? "bcm63xx_uart" : NULL;
}
/* * read all chars in rx fifo and send them to core
*/ staticvoid bcm_uart_do_rx(struct uart_port *port)
{ struct tty_port *tty_port = &port->state->port; unsignedint max_count;
/* limit number of char read in interrupt, should not be * higher than fifo size anyway since we're much faster than
* serial port */
max_count = 32; do { unsignedint iestat, c, cstat; char flag;
/* get overrun/fifo empty information from ier
* register */
iestat = bcm_uart_readl(port, UART_IR_REG);
if (unlikely(iestat & UART_IR_STAT(UART_IR_RXOVER))) { unsignedint val;
/* fifo reset is required to clear
* interrupt */
val = bcm_uart_readl(port, UART_CTL_REG);
val |= UART_CTL_RSTRXFIFO_MASK;
bcm_uart_writel(port, val, UART_CTL_REG);
if (!(iestat & UART_IR_STAT(UART_IR_RXNOTEMPTY))) break;
cstat = c = bcm_uart_readl(port, UART_FIFO_REG);
port->icount.rx++;
flag = TTY_NORMAL;
c &= 0xff;
if (unlikely((cstat & UART_FIFO_ANYERR_MASK))) { /* do stats first */ if (cstat & UART_FIFO_BRKDET_MASK) {
port->icount.brk++; if (uart_handle_break(port)) continue;
}
if (cstat & UART_FIFO_PARERR_MASK)
port->icount.parity++; if (cstat & UART_FIFO_FRAMEERR_MASK)
port->icount.frame++;
/* update flag wrt read_status_mask */
cstat &= port->read_status_mask; if (cstat & UART_FIFO_BRKDET_MASK)
flag = TTY_BREAK; if (cstat & UART_FIFO_FRAMEERR_MASK)
flag = TTY_FRAME; if (cstat & UART_FIFO_PARERR_MASK)
flag = TTY_PARITY;
}
if (uart_prepare_sysrq_char(port, c)) continue;
if ((cstat & port->ignore_status_mask) == 0)
tty_insert_flip_char(tty_port, c, flag);
} while (--max_count);
tty_flip_buffer_push(tty_port);
}
/* * fill tx fifo with chars to send, stop when fifo is about to be full * or when all chars have been sent.
*/ staticvoid bcm_uart_do_tx(struct uart_port *port)
{ unsignedint val; bool pending;
u8 ch;
val = bcm_uart_readl(port, UART_MCTL_REG);
val = (val & UART_MCTL_TXFIFOFILL_MASK) >> UART_MCTL_TXFIFOFILL_SHIFT;
pending = uart_port_tx_limited_flags(port, ch, UART_TX_NOSTOP,
port->fifosize - val, true,
bcm_uart_writel(port, ch, UART_FIFO_REG),
({})); if (pending) return;
/* nothing to send, disable transmit interrupt */
val = bcm_uart_readl(port, UART_IR_REG);
val &= ~UART_TX_INT_MASK;
bcm_uart_writel(port, val, UART_IR_REG);
if (uart_tx_stopped(port))
bcm_uart_stop_tx(port);
}
irqstat = bcm_uart_readl(port, UART_IR_REG); if (irqstat & UART_RX_INT_STAT)
bcm_uart_do_rx(port);
if (irqstat & UART_TX_INT_STAT)
bcm_uart_do_tx(port);
if (irqstat & UART_IR_MASK(UART_IR_EXTIP)) { unsignedint estat;
estat = bcm_uart_readl(port, UART_EXTINP_REG); if (estat & UART_EXTINP_IRSTAT(UART_EXTINP_IR_CTS))
uart_handle_cts_change(port,
estat & UART_EXTINP_CTS_MASK); if (estat & UART_EXTINP_IRSTAT(UART_EXTINP_IR_DCD))
uart_handle_dcd_change(port,
estat & UART_EXTINP_DCD_MASK);
}
val = bcm_uart_readl(port, UART_CTL_REG);
val &= ~(UART_CTL_BRGEN_MASK | UART_CTL_TXEN_MASK |
UART_CTL_RXEN_MASK);
bcm_uart_writel(port, val, UART_CTL_REG);
}
/* * clear all unread data in rx fifo and unsent data in tx fifo
*/ staticvoid bcm_uart_flush(struct uart_port *port)
{ unsignedint val;
/* empty rx and tx fifo */
val = bcm_uart_readl(port, UART_CTL_REG);
val |= UART_CTL_RSTRXFIFO_MASK | UART_CTL_RSTTXFIFO_MASK;
bcm_uart_writel(port, val, UART_CTL_REG);
/* read any pending char to make sure all irq status are
* cleared */
(void)bcm_uart_readl(port, UART_FIFO_REG);
}
/* * serial core request to initialize uart and start rx operation
*/ staticint bcm_uart_startup(struct uart_port *port)
{ unsignedint val; int ret;
/* mask all irq and flush port */
bcm_uart_disable(port);
bcm_uart_writel(port, 0, UART_IR_REG);
bcm_uart_flush(port);
/* clear any pending external input interrupt */
(void)bcm_uart_readl(port, UART_EXTINP_REG);
/* set rx/tx fifo thresh to fifo half size */
val = bcm_uart_readl(port, UART_MCTL_REG);
val &= ~(UART_MCTL_RXFIFOTHRESH_MASK | UART_MCTL_TXFIFOTHRESH_MASK);
val |= (port->fifosize / 2) << UART_MCTL_RXFIFOTHRESH_SHIFT;
val |= (port->fifosize / 2) << UART_MCTL_TXFIFOTHRESH_SHIFT;
bcm_uart_writel(port, val, UART_MCTL_REG);
/* set rx fifo timeout to 1 char time */
val = bcm_uart_readl(port, UART_CTL_REG);
val &= ~UART_CTL_RXTMOUTCNT_MASK;
val |= 1 << UART_CTL_RXTMOUTCNT_SHIFT;
bcm_uart_writel(port, val, UART_CTL_REG);
/* report any edge on dcd and cts */
val = UART_EXTINP_INT_MASK;
val |= UART_EXTINP_DCD_NOSENSE_MASK;
val |= UART_EXTINP_CTS_NOSENSE_MASK;
bcm_uart_writel(port, val, UART_EXTINP_REG);
/* register irq and enable rx interrupts */
ret = request_irq(port->irq, bcm_uart_interrupt, 0,
dev_name(port->dev), port); if (ret) return ret;
bcm_uart_writel(port, UART_RX_INT_MASK, UART_IR_REG);
bcm_uart_enable(port); return 0;
}
/* * serial core request to flush & disable uart
*/ staticvoid bcm_uart_shutdown(struct uart_port *port)
{ unsignedlong flags;
/* * serial core request to release uart iomem
*/ staticvoid bcm_uart_release_port(struct uart_port *port)
{ /* Nothing to release ... */
}
/* * serial core request to do any port required autoconfiguration
*/ staticvoid bcm_uart_config_port(struct uart_port *port, int flags)
{ if (flags & UART_CONFIG_TYPE) { if (bcm_uart_request_port(port)) return;
port->type = PORT_BCM63XX;
}
}
/* * serial core request to check that port information in serinfo are * suitable
*/ staticint bcm_uart_verify_port(struct uart_port *port, struct serial_struct *serinfo)
{ if (port->type != PORT_BCM63XX) return -EINVAL; if (port->irq != serinfo->irq) return -EINVAL; if (port->iotype != serinfo->io_type) return -EINVAL; if (port->mapbase != (unsignedlong)serinfo->iomem_base) return -EINVAL; return 0;
}
/* * console core request to output given string
*/ staticvoid bcm_console_write(struct console *co, constchar *s, unsignedint count)
{ struct uart_port *port; unsignedlong flags; int locked = 1;
port = &ports[co->index];
if (oops_in_progress)
locked = uart_port_trylock_irqsave(port, &flags); else
uart_port_lock_irqsave(port, &flags);
/* call helper to deal with \r\n */
uart_console_write(port, s, count, bcm_console_putchar);
/* and wait for char to be transmitted */
wait_for_xmitr(port);
if (locked)
uart_port_unlock_irqrestore(port, flags);
}
/* * console core request to setup given console, find matching uart * port and setup it.
*/ staticint bcm_console_setup(struct console *co, char *options)
{ struct uart_port *port; int baud = 9600; int bits = 8; int parity = 'n'; int flow = 'n';
if (co->index < 0 || co->index >= BCM63XX_NR_UARTS) return -EINVAL;
port = &ports[co->index]; if (!port->membase) return -ENODEV; if (options)
uart_parse_options(options, &baud, &parity, &bits, &flow);
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.