Commit d7261970 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'tty-5.3-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/gregkh/tty

Pull tty / serial driver updates from Greg KH:
 "Here is the "large" TTY and Serial driver update for 5.3-rc1.

  It's in the negative number of lines overall as we removed an obsolete
  serial driver that was causing problems for some people who were
  trying to clean up some apis (the mpsc.c driver, which only worked for
  some pre-production hardware that no one has anymore.)

  Other than that, lots of tiny changes, cleaning up small things along
  with some platform-specific serial driver updates.

  All of these have been in linux-next for a while now with no reported
  issues"

* tag 'tty-5.3-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/gregkh/tty: (68 commits)
  tty: serial: fsl_lpuart: add imx8qxp support
  serial: imx: set_termios(): preserve RTS state
  serial: imx: set_termios(): clarify RTS/CTS bits calculation
  serial: imx: set_termios(): factor-out 'ucr2' initial value
  serial: sh-sci: Terminate TX DMA during buffer flushing
  serial: sh-sci: Fix TX DMA buffer flushing and workqueue races
  serial: mpsc: Remove obsolete MPSC driver
  serial: 8250: 8250_core: Fix missing unlock on error in serial8250_register_8250_port()
  serial: stm32: add RX and TX FIFO flush
  serial: stm32: add support of RX FIFO threshold
  serial: stm32: add support of TX FIFO threshold
  serial: stm32: update PIO transmission
  serial: stm32: add support of timeout interrupt for RX
  Revert "serial: 8250: Don't service RX FIFO if interrupts are disabled"
  tty/serial/8250: use mctrl_gpio helpers
  serial: mctrl_gpio: Check if GPIO property exisits before requesting it
  serial: 8250: pericom_do_set_divisor can be static
  tty: serial_core: Set port active bit in uart_port_activate
  serial: 8250: Add MSR/MCR TIOCM conversion wrapper functions
  serial: 8250: factor out serial8250_{set,clear}_THRI() helpers
  ...
parents e786741f 35a4ed01
...@@ -2693,8 +2693,8 @@ ...@@ -2693,8 +2693,8 @@
41 = /dev/ttySMX0 Motorola i.MX - port 0 41 = /dev/ttySMX0 Motorola i.MX - port 0
42 = /dev/ttySMX1 Motorola i.MX - port 1 42 = /dev/ttySMX1 Motorola i.MX - port 1
43 = /dev/ttySMX2 Motorola i.MX - port 2 43 = /dev/ttySMX2 Motorola i.MX - port 2
44 = /dev/ttyMM0 Marvell MPSC - port 0 44 = /dev/ttyMM0 Marvell MPSC - port 0 (obsolete unused)
45 = /dev/ttyMM1 Marvell MPSC - port 1 45 = /dev/ttyMM1 Marvell MPSC - port 1 (obsolete unused)
46 = /dev/ttyCPM0 PPC CPM (SCC or SMC) - port 0 46 = /dev/ttyCPM0 PPC CPM (SCC or SMC) - port 0
... ...
47 = /dev/ttyCPM5 PPC CPM (SCC or SMC) - port 5 47 = /dev/ttyCPM5 PPC CPM (SCC or SMC) - port 5
......
...@@ -53,6 +53,9 @@ Optional properties: ...@@ -53,6 +53,9 @@ Optional properties:
programmable TX FIFO thresholds. programmable TX FIFO thresholds.
- resets : phandle + reset specifier pairs - resets : phandle + reset specifier pairs
- overrun-throttle-ms : how long to pause uart rx when input overrun is encountered. - overrun-throttle-ms : how long to pause uart rx when input overrun is encountered.
- {rts,cts,dtr,dsr,rng,dcd}-gpios: specify a GPIO for RTS/CTS/DTR/DSR/RI/DCD
line respectively. It will use specified GPIO instead of the peripheral
function pin for the UART feature. If unsure, don't specify this property.
Note: Note:
* fsl,ns16550: * fsl,ns16550:
...@@ -74,3 +77,19 @@ Example: ...@@ -74,3 +77,19 @@ Example:
interrupts = <10>; interrupts = <10>;
reg-shift = <2>; reg-shift = <2>;
}; };
Example for OMAP UART using GPIO-based modem control signals:
uart4: serial@49042000 {
compatible = "ti,omap3-uart";
reg = <0x49042000 0x400>;
interrupts = <80>;
ti,hwmods = "uart4";
clock-frequency = <48000000>;
cts-gpios = <&gpio3 5 GPIO_ACTIVE_LOW>;
rts-gpios = <&gpio3 6 GPIO_ACTIVE_LOW>;
dtr-gpios = <&gpio1 12 GPIO_ACTIVE_LOW>;
dsr-gpios = <&gpio1 13 GPIO_ACTIVE_LOW>;
dcd-gpios = <&gpio1 14 GPIO_ACTIVE_LOW>;
rng-gpios = <&gpio1 15 GPIO_ACTIVE_LOW>;
};
...@@ -324,8 +324,6 @@ static int rs_ioctl(struct tty_struct *tty, unsigned int cmd, unsigned long arg) ...@@ -324,8 +324,6 @@ static int rs_ioctl(struct tty_struct *tty, unsigned int cmd, unsigned long arg)
return -ENOIOCTLCMD; return -ENOIOCTLCMD;
} }
#define RELEVANT_IFLAG(iflag) (iflag & (IGNBRK|BRKINT|IGNPAR|PARMRK|INPCK))
/* /*
* This routine will shutdown a serial port; interrupts are disabled, and * This routine will shutdown a serial port; interrupts are disabled, and
* DTR is dropped if the hangup on close termio flag is on. * DTR is dropped if the hangup on close termio flag is on.
......
...@@ -15,7 +15,6 @@ ...@@ -15,7 +15,6 @@
#define NR_CHANNELS 8 #define NR_CHANNELS 8
#define IPOCTAL_MAX_BOARDS 16 #define IPOCTAL_MAX_BOARDS 16
#define MAX_DEVICES (NR_CHANNELS * IPOCTAL_MAX_BOARDS) #define MAX_DEVICES (NR_CHANNELS * IPOCTAL_MAX_BOARDS)
#define RELEVANT_IFLAG(iflag) ((iflag) & (IGNBRK|BRKINT|IGNPAR|PARMRK|INPCK))
/** /**
* struct ipoctal_stats -- Stats since last reset * struct ipoctal_stats -- Stats since last reset
......
...@@ -11,6 +11,8 @@ ...@@ -11,6 +11,8 @@
#include <linux/serial_reg.h> #include <linux/serial_reg.h>
#include <linux/dmaengine.h> #include <linux/dmaengine.h>
#include "../serial_mctrl_gpio.h"
struct uart_8250_dma { struct uart_8250_dma {
int (*tx_dma)(struct uart_8250_port *p); int (*tx_dma)(struct uart_8250_port *p);
int (*rx_dma)(struct uart_8250_port *p); int (*rx_dma)(struct uart_8250_port *p);
...@@ -128,6 +130,24 @@ static inline void serial_dl_write(struct uart_8250_port *up, int value) ...@@ -128,6 +130,24 @@ static inline void serial_dl_write(struct uart_8250_port *up, int value)
up->dl_write(up, value); up->dl_write(up, value);
} }
static inline bool serial8250_set_THRI(struct uart_8250_port *up)
{
if (up->ier & UART_IER_THRI)
return false;
up->ier |= UART_IER_THRI;
serial_out(up, UART_IER, up->ier);
return true;
}
static inline bool serial8250_clear_THRI(struct uart_8250_port *up)
{
if (!(up->ier & UART_IER_THRI))
return false;
up->ier &= ~UART_IER_THRI;
serial_out(up, UART_IER, up->ier);
return true;
}
struct uart_8250_port *serial8250_get_port(int line); struct uart_8250_port *serial8250_get_port(int line);
void serial8250_rpm_get(struct uart_8250_port *p); void serial8250_rpm_get(struct uart_8250_port *p);
...@@ -139,14 +159,82 @@ void serial8250_rpm_put_tx(struct uart_8250_port *p); ...@@ -139,14 +159,82 @@ void serial8250_rpm_put_tx(struct uart_8250_port *p);
int serial8250_em485_init(struct uart_8250_port *p); int serial8250_em485_init(struct uart_8250_port *p);
void serial8250_em485_destroy(struct uart_8250_port *p); void serial8250_em485_destroy(struct uart_8250_port *p);
/* MCR <-> TIOCM conversion */
static inline int serial8250_TIOCM_to_MCR(int tiocm)
{
int mcr = 0;
if (tiocm & TIOCM_RTS)
mcr |= UART_MCR_RTS;
if (tiocm & TIOCM_DTR)
mcr |= UART_MCR_DTR;
if (tiocm & TIOCM_OUT1)
mcr |= UART_MCR_OUT1;
if (tiocm & TIOCM_OUT2)
mcr |= UART_MCR_OUT2;
if (tiocm & TIOCM_LOOP)
mcr |= UART_MCR_LOOP;
return mcr;
}
static inline int serial8250_MCR_to_TIOCM(int mcr)
{
int tiocm = 0;
if (mcr & UART_MCR_RTS)
tiocm |= TIOCM_RTS;
if (mcr & UART_MCR_DTR)
tiocm |= TIOCM_DTR;
if (mcr & UART_MCR_OUT1)
tiocm |= TIOCM_OUT1;
if (mcr & UART_MCR_OUT2)
tiocm |= TIOCM_OUT2;
if (mcr & UART_MCR_LOOP)
tiocm |= TIOCM_LOOP;
return tiocm;
}
/* MSR <-> TIOCM conversion */
static inline int serial8250_MSR_to_TIOCM(int msr)
{
int tiocm = 0;
if (msr & UART_MSR_DCD)
tiocm |= TIOCM_CAR;
if (msr & UART_MSR_RI)
tiocm |= TIOCM_RNG;
if (msr & UART_MSR_DSR)
tiocm |= TIOCM_DSR;
if (msr & UART_MSR_CTS)
tiocm |= TIOCM_CTS;
return tiocm;
}
static inline void serial8250_out_MCR(struct uart_8250_port *up, int value) static inline void serial8250_out_MCR(struct uart_8250_port *up, int value)
{ {
serial_out(up, UART_MCR, value); serial_out(up, UART_MCR, value);
if (up->gpios)
mctrl_gpio_set(up->gpios, serial8250_MCR_to_TIOCM(value));
} }
static inline int serial8250_in_MCR(struct uart_8250_port *up) static inline int serial8250_in_MCR(struct uart_8250_port *up)
{ {
return serial_in(up, UART_MCR); int mctrl;
mctrl = serial_in(up, UART_MCR);
if (up->gpios) {
unsigned int mctrl_gpio = 0;
mctrl_gpio = mctrl_gpio_get_outputs(up->gpios, &mctrl_gpio);
mctrl |= serial8250_TIOCM_to_MCR(mctrl_gpio);
}
return mctrl;
} }
#if defined(__alpha__) && !defined(CONFIG_PCI) #if defined(__alpha__) && !defined(CONFIG_PCI)
......
...@@ -14,6 +14,7 @@ ...@@ -14,6 +14,7 @@
* serial8250_register_8250_port() ports * serial8250_register_8250_port() ports
*/ */
#include <linux/acpi.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/moduleparam.h> #include <linux/moduleparam.h>
#include <linux/ioport.h> #include <linux/ioport.h>
...@@ -982,6 +983,8 @@ int serial8250_register_8250_port(struct uart_8250_port *up) ...@@ -982,6 +983,8 @@ int serial8250_register_8250_port(struct uart_8250_port *up)
uart = serial8250_find_match_or_unused(&up->port); uart = serial8250_find_match_or_unused(&up->port);
if (uart && uart->port.type != PORT_8250_CIR) { if (uart && uart->port.type != PORT_8250_CIR) {
struct mctrl_gpios *gpios;
if (uart->port.dev) if (uart->port.dev)
uart_remove_one_port(&serial8250_reg, &uart->port); uart_remove_one_port(&serial8250_reg, &uart->port);
...@@ -1016,6 +1019,22 @@ int serial8250_register_8250_port(struct uart_8250_port *up) ...@@ -1016,6 +1019,22 @@ int serial8250_register_8250_port(struct uart_8250_port *up)
if (up->port.flags & UPF_FIXED_TYPE) if (up->port.flags & UPF_FIXED_TYPE)
uart->port.type = up->port.type; uart->port.type = up->port.type;
/*
* Only call mctrl_gpio_init(), if the device has no ACPI
* companion device
*/
if (!has_acpi_companion(uart->port.dev)) {
gpios = mctrl_gpio_init(&uart->port, 0);
if (IS_ERR(gpios)) {
if (PTR_ERR(gpios) != -ENOSYS) {
ret = PTR_ERR(gpios);
goto out_unlock;
}
} else {
uart->gpios = gpios;
}
}
serial8250_set_defaults(uart); serial8250_set_defaults(uart);
/* Possibly override default I/O functions. */ /* Possibly override default I/O functions. */
...@@ -1082,6 +1101,7 @@ int serial8250_register_8250_port(struct uart_8250_port *up) ...@@ -1082,6 +1101,7 @@ int serial8250_register_8250_port(struct uart_8250_port *up)
} }
} }
out_unlock:
mutex_unlock(&serial_mutex); mutex_unlock(&serial_mutex);
return ret; return ret;
......
...@@ -34,10 +34,8 @@ static void __dma_tx_complete(void *param) ...@@ -34,10 +34,8 @@ static void __dma_tx_complete(void *param)
uart_write_wakeup(&p->port); uart_write_wakeup(&p->port);
ret = serial8250_tx_dma(p); ret = serial8250_tx_dma(p);
if (ret) { if (ret)
p->ier |= UART_IER_THRI; serial8250_set_THRI(p);
serial_port_out(&p->port, UART_IER, p->ier);
}
spin_unlock_irqrestore(&p->port.lock, flags); spin_unlock_irqrestore(&p->port.lock, flags);
} }
...@@ -100,10 +98,7 @@ int serial8250_tx_dma(struct uart_8250_port *p) ...@@ -100,10 +98,7 @@ int serial8250_tx_dma(struct uart_8250_port *p)
dma_async_issue_pending(dma->txchan); dma_async_issue_pending(dma->txchan);
if (dma->tx_err) { if (dma->tx_err) {
dma->tx_err = 0; dma->tx_err = 0;
if (p->ier & UART_IER_THRI) { serial8250_clear_THRI(p);
p->ier &= ~UART_IER_THRI;
serial_out(p, UART_IER, p->ier);
}
} }
return 0; return 0;
err: err:
......
...@@ -10,6 +10,7 @@ ...@@ -10,6 +10,7 @@
#include <linux/module.h> #include <linux/module.h>
#include <linux/of_irq.h> #include <linux/of_irq.h>
#include <linux/of_platform.h> #include <linux/of_platform.h>
#include <linux/pinctrl/consumer.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/pm_runtime.h> #include <linux/pm_runtime.h>
#include <linux/serial_8250.h> #include <linux/serial_8250.h>
...@@ -47,7 +48,6 @@ ...@@ -47,7 +48,6 @@
#define MTK_UART_DMA_EN_RX 0x5 #define MTK_UART_DMA_EN_RX 0x5
#define MTK_UART_ESCAPE_CHAR 0x77 /* Escape char added under sw fc */ #define MTK_UART_ESCAPE_CHAR 0x77 /* Escape char added under sw fc */
#define MTK_UART_TX_SIZE UART_XMIT_SIZE
#define MTK_UART_RX_SIZE 0x8000 #define MTK_UART_RX_SIZE 0x8000
#define MTK_UART_TX_TRIGGER 1 #define MTK_UART_TX_TRIGGER 1
#define MTK_UART_RX_TRIGGER MTK_UART_RX_SIZE #define MTK_UART_RX_TRIGGER MTK_UART_RX_SIZE
...@@ -70,6 +70,7 @@ struct mtk8250_data { ...@@ -70,6 +70,7 @@ struct mtk8250_data {
#ifdef CONFIG_SERIAL_8250_DMA #ifdef CONFIG_SERIAL_8250_DMA
enum dma_rx_status rx_status; enum dma_rx_status rx_status;
#endif #endif
int rx_wakeup_irq;
}; };
/* flow control mode */ /* flow control mode */
...@@ -89,28 +90,30 @@ static void mtk8250_dma_rx_complete(void *param) ...@@ -89,28 +90,30 @@ static void mtk8250_dma_rx_complete(void *param)
struct mtk8250_data *data = up->port.private_data; struct mtk8250_data *data = up->port.private_data;
struct tty_port *tty_port = &up->port.state->port; struct tty_port *tty_port = &up->port.state->port;
struct dma_tx_state state; struct dma_tx_state state;
int copied, total, cnt;
unsigned char *ptr; unsigned char *ptr;
int copied;
dma_sync_single_for_cpu(dma->rxchan->device->dev, dma->rx_addr, if (data->rx_status == DMA_RX_SHUTDOWN)
dma->rx_size, DMA_FROM_DEVICE); return;
dmaengine_tx_status(dma->rxchan, dma->rx_cookie, &state); dmaengine_tx_status(dma->rxchan, dma->rx_cookie, &state);
total = dma->rx_size - state.residue;
cnt = total;
if (data->rx_status == DMA_RX_SHUTDOWN) if ((data->rx_pos + cnt) > dma->rx_size)
return; cnt = dma->rx_size - data->rx_pos;
if ((data->rx_pos + state.residue) <= dma->rx_size) { ptr = (unsigned char *)(data->rx_pos + dma->rx_buf);
ptr = (unsigned char *)(data->rx_pos + dma->rx_buf); copied = tty_insert_flip_string(tty_port, ptr, cnt);
copied = tty_insert_flip_string(tty_port, ptr, state.residue); data->rx_pos += cnt;
} else {
ptr = (unsigned char *)(data->rx_pos + dma->rx_buf); if (total > cnt) {
copied = tty_insert_flip_string(tty_port, ptr,
dma->rx_size - data->rx_pos);
ptr = (unsigned char *)(dma->rx_buf); ptr = (unsigned char *)(dma->rx_buf);
copied += tty_insert_flip_string(tty_port, ptr, cnt = total - cnt;
data->rx_pos + state.residue - dma->rx_size); copied += tty_insert_flip_string(tty_port, ptr, cnt);
data->rx_pos = cnt;
} }
up->port.icount.rx += copied; up->port.icount.rx += copied;
tty_flip_buffer_push(tty_port); tty_flip_buffer_push(tty_port);
...@@ -121,9 +124,7 @@ static void mtk8250_dma_rx_complete(void *param) ...@@ -121,9 +124,7 @@ static void mtk8250_dma_rx_complete(void *param)
static void mtk8250_rx_dma(struct uart_8250_port *up) static void mtk8250_rx_dma(struct uart_8250_port *up)
{ {
struct uart_8250_dma *dma = up->dma; struct uart_8250_dma *dma = up->dma;
struct mtk8250_data *data = up->port.private_data;
struct dma_async_tx_descriptor *desc; struct dma_async_tx_descriptor *desc;
struct dma_tx_state state;
desc = dmaengine_prep_slave_single(dma->rxchan, dma->rx_addr, desc = dmaengine_prep_slave_single(dma->rxchan, dma->rx_addr,
dma->rx_size, DMA_DEV_TO_MEM, dma->rx_size, DMA_DEV_TO_MEM,
...@@ -138,12 +139,6 @@ static void mtk8250_rx_dma(struct uart_8250_port *up) ...@@ -138,12 +139,6 @@ static void mtk8250_rx_dma(struct uart_8250_port *up)
dma->rx_cookie = dmaengine_submit(desc); dma->rx_cookie = dmaengine_submit(desc);
dmaengine_tx_status(dma->rxchan, dma->rx_cookie, &state);
data->rx_pos = state.residue;
dma_sync_single_for_device(dma->rxchan->device->dev, dma->rx_addr,
dma->rx_size, DMA_FROM_DEVICE);
dma_async_issue_pending(dma->rxchan); dma_async_issue_pending(dma->rxchan);
} }
...@@ -156,13 +151,11 @@ static void mtk8250_dma_enable(struct uart_8250_port *up) ...@@ -156,13 +151,11 @@ static void mtk8250_dma_enable(struct uart_8250_port *up)
if (data->rx_status != DMA_RX_START) if (data->rx_status != DMA_RX_START)
return; return;
dma->rxconf.direction = DMA_DEV_TO_MEM; dma->rxconf.src_port_window_size = dma->rx_size;
dma->rxconf.src_addr_width = dma->rx_size / 1024; dma->rxconf.src_addr = dma->rx_addr;
dma->rxconf.src_addr = dma->rx_addr;
dma->txconf.direction = DMA_MEM_TO_DEV; dma->txconf.dst_port_window_size = UART_XMIT_SIZE;
dma->txconf.dst_addr_width = MTK_UART_TX_SIZE / 1024; dma->txconf.dst_addr = dma->tx_addr;
dma->txconf.dst_addr = dma->tx_addr;
serial_out(up, UART_FCR, UART_FCR_ENABLE_FIFO | UART_FCR_CLEAR_RCVR | serial_out(up, UART_FCR, UART_FCR_ENABLE_FIFO | UART_FCR_CLEAR_RCVR |
UART_FCR_CLEAR_XMIT); UART_FCR_CLEAR_XMIT);
...@@ -551,6 +544,8 @@ static int mtk8250_probe(struct platform_device *pdev) ...@@ -551,6 +544,8 @@ static int mtk8250_probe(struct platform_device *pdev)
pm_runtime_set_active(&pdev->dev); pm_runtime_set_active(&pdev->dev);
pm_runtime_enable(&pdev->dev); pm_runtime_enable(&pdev->dev);
data->rx_wakeup_irq = platform_get_irq(pdev, 1);
return 0; return 0;
} }
...@@ -572,15 +567,35 @@ static int mtk8250_remove(struct platform_device *pdev) ...@@ -572,15 +567,35 @@ static int mtk8250_remove(struct platform_device *pdev)
static int __maybe_unused mtk8250_suspend(struct device *dev) static int __maybe_unused mtk8250_suspend(struct device *dev)
{ {
struct mtk8250_data *data = dev_get_drvdata(dev); struct mtk8250_data *data = dev_get_drvdata(dev);
int irq = data->rx_wakeup_irq;
int err;
serial8250_suspend_port(data->line); serial8250_suspend_port(data->line);
pinctrl_pm_select_sleep_state(dev);
if (irq >= 0) {
err = enable_irq_wake(irq);
if (err) {
dev_err(dev,
"failed to enable irq wake on IRQ %d: %d\n",
irq, err);
pinctrl_pm_select_default_state(dev);
serial8250_resume_port(data->line);
return err;
}
}
return 0; return 0;
} }
static int __maybe_unused mtk8250_resume(struct device *dev) static int __maybe_unused mtk8250_resume(struct device *dev)
{ {
struct mtk8250_data *data = dev_get_drvdata(dev); struct mtk8250_data *data = dev_get_drvdata(dev);
int irq = data->rx_wakeup_irq;
if (irq >= 0)
disable_irq_wake(irq);
pinctrl_pm_select_default_state(dev);
serial8250_resume_port(data->line); serial8250_resume_port(data->line);
......
...@@ -70,9 +70,10 @@ static int of_platform_serial_setup(struct platform_device *ofdev, ...@@ -70,9 +70,10 @@ static int of_platform_serial_setup(struct platform_device *ofdev,
/* Get clk rate through clk driver if present */ /* Get clk rate through clk driver if present */
info->clk = devm_clk_get(&ofdev->dev, NULL); info->clk = devm_clk_get(&ofdev->dev, NULL);
if (IS_ERR(info->clk)) { if (IS_ERR(info->clk)) {
dev_warn(&ofdev->dev,
"clk or clock-frequency not defined\n");
ret = PTR_ERR(info->clk); ret = PTR_ERR(info->clk);
if (ret != -EPROBE_DEFER)
dev_warn(&ofdev->dev,
"failed to get clock: %d\n", ret);
goto err_pmruntime; goto err_pmruntime;
} }
...@@ -205,18 +206,16 @@ static int of_platform_serial_setup(struct platform_device *ofdev, ...@@ -205,18 +206,16 @@ static int of_platform_serial_setup(struct platform_device *ofdev,
/* /*
* Try to register a serial port * Try to register a serial port
*/ */
static const struct of_device_id of_platform_serial_table[];
static int of_platform_serial_probe(struct platform_device *ofdev) static int of_platform_serial_probe(struct platform_device *ofdev)
{ {
const struct of_device_id *match;
struct of_serial_info *info; struct of_serial_info *info;
struct uart_8250_port port8250; struct uart_8250_port port8250;
unsigned int port_type;
u32 tx_threshold; u32 tx_threshold;
int port_type;
int ret; int ret;
match = of_match_device(of_platform_serial_table, &ofdev->dev); port_type = (unsigned long)of_device_get_match_data(&ofdev->dev);
if (!match) if (port_type == PORT_UNKNOWN)
return -EINVAL; return -EINVAL;
if (of_property_read_bool(ofdev->dev.of_node, "used-by-rtas")) if (of_property_read_bool(ofdev->dev.of_node, "used-by-rtas"))
...@@ -226,7 +225,6 @@ static int of_platform_serial_probe(struct platform_device *ofdev) ...@@ -226,7 +225,6 @@ static int of_platform_serial_probe(struct platform_device *ofdev)
if (info == NULL) if (info == NULL)
return -ENOMEM; return -ENOMEM;
port_type = (unsigned long)match->data;
memset(&port8250, 0, sizeof(port8250)); memset(&port8250, 0, sizeof(port8250));
ret = of_platform_serial_setup(ofdev, port_type, &port8250.port, info); ret = of_platform_serial_setup(ofdev, port_type, &port8250.port, info);
if (ret) if (ret)
......
...@@ -141,18 +141,20 @@ static void omap8250_set_mctrl(struct uart_port *port, unsigned int mctrl) ...@@ -141,18 +141,20 @@ static void omap8250_set_mctrl(struct uart_port *port, unsigned int mctrl)
serial8250_do_set_mctrl(port, mctrl); serial8250_do_set_mctrl(port, mctrl);
/* if (!up->gpios) {
* Turn off autoRTS if RTS is lowered and restore autoRTS setting /*
* if RTS is raised * Turn off autoRTS if RTS is lowered and restore autoRTS
*/ * setting if RTS is raised
lcr = serial_in(up, UART_LCR); */
serial_out(up, UART_LCR, UART_LCR_CONF_MODE_B); lcr = serial_in(up, UART_LCR);
if ((mctrl & TIOCM_RTS) && (port->status & UPSTAT_AUTORTS)) serial_out(up, UART_LCR, UART_LCR_CONF_MODE_B);
priv->efr |= UART_EFR_RTS; if ((mctrl & TIOCM_RTS) && (port->status & UPSTAT_AUTORTS))
else priv->efr |= UART_EFR_RTS;
priv->efr &= ~UART_EFR_RTS; else
serial_out(up, UART_EFR, priv->efr); priv->efr &= ~UART_EFR_RTS;
serial_out(up, UART_LCR, lcr); serial_out(up, UART_EFR, priv->efr);
serial_out(up, UART_LCR, lcr);
}
} }
/* /*
...@@ -453,7 +455,8 @@ static void omap_8250_set_termios(struct uart_port *port, ...@@ -453,7 +455,8 @@ static void omap_8250_set_termios(struct uart_port *port,
priv->efr = 0; priv->efr = 0;
up->port.status &= ~(UPSTAT_AUTOCTS | UPSTAT_AUTORTS | UPSTAT_AUTOXOFF); up->port.status &= ~(UPSTAT_AUTOCTS | UPSTAT_AUTORTS | UPSTAT_AUTOXOFF);
if (termios->c_cflag & CRTSCTS && up->port.flags & UPF_HARD_FLOW) { if (termios->c_cflag & CRTSCTS && up->port.flags & UPF_HARD_FLOW &&
!up->gpios) {
/* Enable AUTOCTS (autoRTS is enabled when RTS is raised) */ /* Enable AUTOCTS (autoRTS is enabled when RTS is raised) */
up->port.status |= UPSTAT_AUTOCTS | UPSTAT_AUTORTS; up->port.status |= UPSTAT_AUTOCTS | UPSTAT_AUTORTS;
priv->efr |= UART_EFR_CTS; priv->efr |= UART_EFR_CTS;
...@@ -923,15 +926,13 @@ static void omap_8250_dma_tx_complete(void *param) ...@@ -923,15 +926,13 @@ static void omap_8250_dma_tx_complete(void *param)
ret = omap_8250_tx_dma(p); ret = omap_8250_tx_dma(p);
if (ret) if (ret)
en_thri = true; en_thri = true;
} else if (p->capabilities & UART_CAP_RPM) { } else if (p->capabilities & UART_CAP_RPM) {
en_thri = true; en_thri = true;
} }
if (en_thri) { if (en_thri) {
dma->tx_err = 1; dma->tx_err = 1;
p->ier |= UART_IER_THRI; serial8250_set_THRI(p);
serial_port_out(&p->port, UART_IER, p->ier);
} }
spin_unlock_irqrestore(&p->port.lock, flags); spin_unlock_irqrestore(&p->port.lock, flags);
...@@ -959,10 +960,7 @@ static int omap_8250_tx_dma(struct uart_8250_port *p) ...@@ -959,10 +960,7 @@ static int omap_8250_tx_dma(struct uart_8250_port *p)
ret = -EBUSY; ret = -EBUSY;
goto err; goto err;
} }
if (p->ier & UART_IER_THRI) { serial8250_clear_THRI(p);
p->ier &= ~UART_IER_THRI;
serial_out(p, UART_IER, p->ier);
}
return 0; return 0;
} }
...@@ -1020,10 +1018,7 @@ static int omap_8250_tx_dma(struct uart_8250_port *p) ...@@ -1020,10 +1018,7 @@ static int omap_8250_tx_dma(struct uart_8250_port *p)
if (dma->tx_err) if (dma->tx_err)
dma->tx_err = 0; dma->tx_err = 0;
if (p->ier & UART_IER_THRI) { serial8250_clear_THRI(p);
p->ier &= ~UART_IER_THRI;
serial_out(p, UART_IER, p->ier);
}
if (skip_byte) if (skip_byte)
serial_out(p, UART_TX, xmit->buf[xmit->tail]); serial_out(p, UART_TX, xmit->buf[xmit->tail]);
return 0; return 0;
......
...@@ -1326,13 +1326,66 @@ static int pci_default_setup(struct serial_private *priv, ...@@ -1326,13 +1326,66 @@ static int pci_default_setup(struct serial_private *priv,
return setup_port(priv, port, bar, offset, board->reg_shift); return setup_port(priv, port, bar, offset, board->reg_shift);
} }
static void
pericom_do_set_divisor(struct uart_port *port, unsigned int baud,
unsigned int quot, unsigned int quot_frac)
{
int scr;
int lcr;
int actual_baud;
int tolerance;
for (scr = 5 ; scr <= 15 ; scr++) {
actual_baud = 921600 * 16 / scr;
tolerance = actual_baud / 50;
if ((baud < actual_baud + tolerance) &&
(baud > actual_baud - tolerance)) {
lcr = serial_port_in(port, UART_LCR);
serial_port_out(port, UART_LCR, lcr | 0x80);
serial_port_out(port, UART_DLL, 1);
serial_port_out(port, UART_DLM, 0);
serial_port_out(port, 2, 16 - scr);
serial_port_out(port, UART_LCR, lcr);
return;
} else if (baud > actual_baud) {
break;
}
}
serial8250_do_set_divisor(port, baud, quot, quot_frac);
}
static int pci_pericom_setup(struct serial_private *priv, static int pci_pericom_setup(struct serial_private *priv,
const struct pciserial_board *board, const struct pciserial_board *board,
struct uart_8250_port *port, int idx) struct uart_8250_port *port, int idx)
{ {
unsigned int bar, offset = board->first_offset, maxnr; unsigned int bar, offset = board->first_offset, maxnr;
bar = FL_GET_BASE(board->flags);
if (board->flags & FL_BASE_BARS)
bar += idx;
else
offset += idx * board->uart_offset;
maxnr = (pci_resource_len(priv->dev, bar) - board->first_offset) >>
(board->reg_shift + 3);
if (board->flags & FL_REGION_SZ_CAP && idx >= maxnr)
return 1;
port->port.set_divisor = pericom_do_set_divisor;
return setup_port(priv, port, bar, offset, board->reg_shift);
}
static int pci_pericom_setup_four_at_eight(struct serial_private *priv,
const struct pciserial_board *board,
struct uart_8250_port *port, int idx)
{
unsigned int bar, offset = board->first_offset, maxnr;
bar = FL_GET_BASE(board->flags); bar = FL_GET_BASE(board->flags);
if (board->flags & FL_BASE_BARS) if (board->flags & FL_BASE_BARS)
bar += idx; bar += idx;
...@@ -1348,6 +1401,8 @@ static int pci_pericom_setup(struct serial_private *priv, ...@@ -1348,6 +1401,8 @@ static int pci_pericom_setup(struct serial_private *priv,
if (board->flags & FL_REGION_SZ_CAP && idx >= maxnr) if (board->flags & FL_REGION_SZ_CAP && idx >= maxnr)
return 1; return 1;
port->port.set_divisor = pericom_do_set_divisor;
return setup_port(priv, port, bar, offset, board->reg_shift); return setup_port(priv, port, bar, offset, board->reg_shift);
} }
...@@ -1995,7 +2050,7 @@ static struct pci_serial_quirk pci_serial_quirks[] __refdata = { ...@@ -1995,7 +2050,7 @@ static struct pci_serial_quirk pci_serial_quirks[] __refdata = {
.device = PCI_DEVICE_ID_PERICOM_PI7C9X7954, .device = PCI_DEVICE_ID_PERICOM_PI7C9X7954,
.subvendor = PCI_ANY_ID, .subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID, .subdevice = PCI_ANY_ID,
.setup = pci_pericom_setup, .setup = pci_pericom_setup_four_at_eight,
}, },
/* /*
* PLX * PLX
...@@ -2032,107 +2087,113 @@ static struct pci_serial_quirk pci_serial_quirks[] __refdata = { ...@@ -2032,107 +2087,113 @@ static struct pci_serial_quirk pci_serial_quirks[] __refdata = {
.device = PCI_DEVICE_ID_ACCESIO_PCIE_COM_4SDB, .device = PCI_DEVICE_ID_ACCESIO_PCIE_COM_4SDB,
.subvendor = PCI_ANY_ID, .subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID, .subdevice = PCI_ANY_ID,
.setup = pci_pericom_setup, .setup = pci_pericom_setup_four_at_eight,
}, },
{ {
.vendor = PCI_VENDOR_ID_ACCESIO, .vendor = PCI_VENDOR_ID_ACCESIO,
.device = PCI_DEVICE_ID_ACCESIO_MPCIE_COM_4S, .device = PCI_DEVICE_ID_ACCESIO_MPCIE_COM_4S,
.subvendor = PCI_ANY_ID, .subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID, .subdevice = PCI_ANY_ID,
.setup = pci_pericom_setup, .setup = pci_pericom_setup_four_at_eight,
}, },
{ {
.vendor = PCI_VENDOR_ID_ACCESIO, .vendor = PCI_VENDOR_ID_ACCESIO,
.device = PCI_DEVICE_ID_ACCESIO_PCIE_COM232_4DB, .device = PCI_DEVICE_ID_ACCESIO_PCIE_COM232_4DB,
.subvendor = PCI_ANY_ID, .subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID, .subdevice = PCI_ANY_ID,
.setup = pci_pericom_setup, .setup = pci_pericom_setup_four_at_eight,
}, },
{ {
.vendor = PCI_VENDOR_ID_ACCESIO, .vendor = PCI_VENDOR_ID_ACCESIO,
.device = PCI_DEVICE_ID_ACCESIO_MPCIE_COM232_4, .device = PCI_DEVICE_ID_ACCESIO_MPCIE_COM232_4,
.subvendor = PCI_ANY_ID, .subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID, .subdevice = PCI_ANY_ID,
.setup = pci_pericom_setup, .setup = pci_pericom_setup_four_at_eight,
}, },
{ {
.vendor = PCI_VENDOR_ID_ACCESIO, .vendor = PCI_VENDOR_ID_ACCESIO,
.device = PCI_DEVICE_ID_ACCESIO_PCIE_COM_4SMDB, .device = PCI_DEVICE_ID_ACCESIO_PCIE_COM_4SMDB,
.subvendor = PCI_ANY_ID, .subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID, .subdevice = PCI_ANY_ID,
.setup = pci_pericom_setup, .setup = pci_pericom_setup_four_at_eight,
}, },
{ {
.vendor = PCI_VENDOR_ID_ACCESIO, .vendor = PCI_VENDOR_ID_ACCESIO,
.device = PCI_DEVICE_ID_ACCESIO_MPCIE_COM_4SM, .device = PCI_DEVICE_ID_ACCESIO_MPCIE_COM_4SM,
.subvendor = PCI_ANY_ID, .subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID, .subdevice = PCI_ANY_ID,
.setup = pci_pericom_setup, .setup = pci_pericom_setup_four_at_eight,
}, },
{ {
.vendor = PCI_VENDOR_ID_ACCESIO, .vendor = PCI_VENDOR_ID_ACCESIO,
.device = PCI_DEVICE_ID_ACCESIO_MPCIE_ICM422_4, .device = PCI_DEVICE_ID_ACCESIO_MPCIE_ICM422_4,
.subvendor = PCI_ANY_ID, .subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID, .subdevice = PCI_ANY_ID,
.setup = pci_pericom_setup, .setup = pci_pericom_setup_four_at_eight,
}, },
{ {
.vendor = PCI_VENDOR_ID_ACCESIO, .vendor = PCI_VENDOR_ID_ACCESIO,
.device = PCI_DEVICE_ID_ACCESIO_MPCIE_ICM485_4, .device = PCI_DEVICE_ID_ACCESIO_MPCIE_ICM485_4,
.subvendor = PCI_ANY_ID, .subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID, .subdevice = PCI_ANY_ID,
.setup = pci_pericom_setup, .setup = pci_pericom_setup_four_at_eight,
}, },
{ {
.vendor = PCI_DEVICE_ID_ACCESIO_PCIE_ICM_4S, .vendor = PCI_DEVICE_ID_ACCESIO_PCIE_ICM_4S,
.device = PCI_DEVICE_ID_ACCESIO_PCIE_ICM232_4, .device = PCI_DEVICE_ID_ACCESIO_PCIE_ICM232_4,
.subvendor = PCI_ANY_ID, .subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID, .subdevice = PCI_ANY_ID,
.setup = pci_pericom_setup, .setup = pci_pericom_setup_four_at_eight,
}, },
{ {
.vendor = PCI_VENDOR_ID_ACCESIO, .vendor = PCI_VENDOR_ID_ACCESIO,
.device = PCI_DEVICE_ID_ACCESIO_MPCIE_ICM232_4, .device = PCI_DEVICE_ID_ACCESIO_MPCIE_ICM232_4,
.subvendor = PCI_ANY_ID, .subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID, .subdevice = PCI_ANY_ID,
.setup = pci_pericom_setup, .setup = pci_pericom_setup_four_at_eight,
}, },
{ {
.vendor = PCI_VENDOR_ID_ACCESIO, .vendor = PCI_VENDOR_ID_ACCESIO,
.device = PCI_DEVICE_ID_ACCESIO_PCIE_COM422_4, .device = PCI_DEVICE_ID_ACCESIO_PCIE_COM422_4,
.subvendor = PCI_ANY_ID, .subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID, .subdevice = PCI_ANY_ID,
.setup = pci_pericom_setup, .setup = pci_pericom_setup_four_at_eight,
}, },
{ {
.vendor = PCI_VENDOR_ID_ACCESIO, .vendor = PCI_VENDOR_ID_ACCESIO,
.device = PCI_DEVICE_ID_ACCESIO_PCIE_COM485_4, .device = PCI_DEVICE_ID_ACCESIO_PCIE_COM485_4,
.subvendor = PCI_ANY_ID, .subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID, .subdevice = PCI_ANY_ID,
.setup = pci_pericom_setup, .setup = pci_pericom_setup_four_at_eight,
}, },
{ {
.vendor = PCI_VENDOR_ID_ACCESIO, .vendor = PCI_VENDOR_ID_ACCESIO,
.device = PCI_DEVICE_ID_ACCESIO_PCIE_COM232_4, .device = PCI_DEVICE_ID_ACCESIO_PCIE_COM232_4,
.subvendor = PCI_ANY_ID, .subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID, .subdevice = PCI_ANY_ID,
.setup = pci_pericom_setup, .setup = pci_pericom_setup_four_at_eight,
}, },
{ {
.vendor = PCI_VENDOR_ID_ACCESIO, .vendor = PCI_VENDOR_ID_ACCESIO,
.device = PCI_DEVICE_ID_ACCESIO_PCIE_COM_4SM, .device = PCI_DEVICE_ID_ACCESIO_PCIE_COM_4SM,
.subvendor = PCI_ANY_ID, .subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID, .subdevice = PCI_ANY_ID,
.setup = pci_pericom_setup, .setup = pci_pericom_setup_four_at_eight,
}, },
{ {
.vendor = PCI_VENDOR_ID_ACCESIO, .vendor = PCI_VENDOR_ID_ACCESIO,
.device = PCI_DEVICE_ID_ACCESIO_PCIE_ICM_4SM, .device = PCI_DEVICE_ID_ACCESIO_PCIE_ICM_4SM,
.subvendor = PCI_ANY_ID, .subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID, .subdevice = PCI_ANY_ID,
.setup = pci_pericom_setup, .setup = pci_pericom_setup_four_at_eight,
}, },
/* {
.vendor = PCI_VENDOR_ID_ACCESIO,
.device = PCI_ANY_ID,
.subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID,
.setup = pci_pericom_setup,
}, /*
* SBS Technologies, Inc., PMC-OCTALPRO 232 * SBS Technologies, Inc., PMC-OCTALPRO 232
*/ */
{ {
......
...@@ -462,8 +462,8 @@ serial_pnp_probe(struct pnp_dev *dev, const struct pnp_device_id *dev_id) ...@@ -462,8 +462,8 @@ serial_pnp_probe(struct pnp_dev *dev, const struct pnp_device_id *dev_id)
return -ENODEV; return -ENODEV;
dev_dbg(&dev->dev, dev_dbg(&dev->dev,
"Setup PNP port: port %lx, mem %pa, irq %d, type %d\n", "Setup PNP port: port %#lx, mem %#llx, irq %u, type %u\n",
uart.port.iobase, &uart.port.mapbase, uart.port.iobase, (unsigned long long)uart.port.mapbase,
uart.port.irq, uart.port.iotype); uart.port.irq, uart.port.iotype);
if (flags & CIR_PORT) { if (flags & CIR_PORT) {
......
...@@ -1502,11 +1502,8 @@ static void __stop_tx_rs485(struct uart_8250_port *p) ...@@ -1502,11 +1502,8 @@ static void __stop_tx_rs485(struct uart_8250_port *p)
static inline void __do_stop_tx(struct uart_8250_port *p) static inline void __do_stop_tx(struct uart_8250_port *p)
{ {
if (p->ier & UART_IER_THRI) { if (serial8250_clear_THRI(p))
p->ier &= ~UART_IER_THRI;
serial_out(p, UART_IER, p->ier);
serial8250_rpm_put_tx(p); serial8250_rpm_put_tx(p);
}
} }
static inline void __stop_tx(struct uart_8250_port *p) static inline void __stop_tx(struct uart_8250_port *p)
...@@ -1555,10 +1552,7 @@ static inline void __start_tx(struct uart_port *port) ...@@ -1555,10 +1552,7 @@ static inline void __start_tx(struct uart_port *port)
if (up->dma && !up->dma->tx_dma(up)) if (up->dma && !up->dma->tx_dma(up))
return; return;
if (!(up->ier & UART_IER_THRI)) { if (serial8250_set_THRI(up)) {
up->ier |= UART_IER_THRI;
serial_port_out(port, UART_IER, up->ier);
if (up->bugs & UART_BUG_TXEN) { if (up->bugs & UART_BUG_TXEN) {
unsigned char lsr; unsigned char lsr;
...@@ -1662,6 +1656,8 @@ static void serial8250_disable_ms(struct uart_port *port) ...@@ -1662,6 +1656,8 @@ static void serial8250_disable_ms(struct uart_port *port)
if (up->bugs & UART_BUG_NOMSR) if (up->bugs & UART_BUG_NOMSR)
return; return;
mctrl_gpio_disable_ms(up->gpios);
up->ier &= ~UART_IER_MSI; up->ier &= ~UART_IER_MSI;
serial_port_out(port, UART_IER, up->ier); serial_port_out(port, UART_IER, up->ier);
} }
...@@ -1674,6 +1670,8 @@ static void serial8250_enable_ms(struct uart_port *port) ...@@ -1674,6 +1670,8 @@ static void serial8250_enable_ms(struct uart_port *port)
if (up->bugs & UART_BUG_NOMSR) if (up->bugs & UART_BUG_NOMSR)
return; return;
mctrl_gpio_enable_ms(up->gpios);
up->ier |= UART_IER_MSI; up->ier |= UART_IER_MSI;
serial8250_rpm_get(up); serial8250_rpm_get(up);
...@@ -1869,13 +1867,13 @@ int serial8250_handle_irq(struct uart_port *port, unsigned int iir) ...@@ -1869,13 +1867,13 @@ int serial8250_handle_irq(struct uart_port *port, unsigned int iir)
status = serial_port_in(port, UART_LSR); status = serial_port_in(port, UART_LSR);
if (status & (UART_LSR_DR | UART_LSR_BI) && if (status & (UART_LSR_DR | UART_LSR_BI)) {
iir & UART_IIR_RDI) {
if (!up->dma || handle_rx_dma(up, iir)) if (!up->dma || handle_rx_dma(up, iir))
status = serial8250_rx_chars(up, status); status = serial8250_rx_chars(up, status);
} }
serial8250_modem_status(up); serial8250_modem_status(up);
if ((!up->dma || up->dma->tx_err) && (status & UART_LSR_THRE)) if ((!up->dma || up->dma->tx_err) && (status & UART_LSR_THRE) &&
(up->ier & UART_IER_THRI))
serial8250_tx_chars(up); serial8250_tx_chars(up);
uart_unlock_and_check_sysrq(port, flags); uart_unlock_and_check_sysrq(port, flags);
...@@ -1944,22 +1942,17 @@ unsigned int serial8250_do_get_mctrl(struct uart_port *port) ...@@ -1944,22 +1942,17 @@ unsigned int serial8250_do_get_mctrl(struct uart_port *port)
{ {
struct uart_8250_port *up = up_to_u8250p(port); struct uart_8250_port *up = up_to_u8250p(port);
unsigned int status; unsigned int status;
unsigned int ret; unsigned int val;
serial8250_rpm_get(up); serial8250_rpm_get(up);
status = serial8250_modem_status(up); status = serial8250_modem_status(up);
serial8250_rpm_put(up); serial8250_rpm_put(up);
ret = 0; val = serial8250_MSR_to_TIOCM(status);
if (status & UART_MSR_DCD) if (up->gpios)
ret |= TIOCM_CAR; return mctrl_gpio_get(up->gpios, &val);
if (status & UART_MSR_RI)
ret |= TIOCM_RNG; return val;
if (status & UART_MSR_DSR)
ret |= TIOCM_DSR;
if (status & UART_MSR_CTS)
ret |= TIOCM_CTS;
return ret;
} }
EXPORT_SYMBOL_GPL(serial8250_do_get_mctrl); EXPORT_SYMBOL_GPL(serial8250_do_get_mctrl);
...@@ -1973,18 +1966,9 @@ static unsigned int serial8250_get_mctrl(struct uart_port *port) ...@@ -1973,18 +1966,9 @@ static unsigned int serial8250_get_mctrl(struct uart_port *port)
void serial8250_do_set_mctrl(struct uart_port *port, unsigned int mctrl) void serial8250_do_set_mctrl(struct uart_port *port, unsigned int mctrl)
{ {
struct uart_8250_port *up = up_to_u8250p(port); struct uart_8250_port *up = up_to_u8250p(port);
unsigned char mcr = 0; unsigned char mcr;
if (mctrl & TIOCM_RTS) mcr = serial8250_TIOCM_to_MCR(mctrl);
mcr |= UART_MCR_RTS;
if (mctrl & TIOCM_DTR)
mcr |= UART_MCR_DTR;
if (mctrl & TIOCM_OUT1)
mcr |= UART_MCR_OUT1;
if (mctrl & TIOCM_OUT2)
mcr |= UART_MCR_OUT2;
if (mctrl & TIOCM_LOOP)
mcr |= UART_MCR_LOOP;
mcr = (mcr & up->mcr_mask) | up->mcr_force | up->mcr; mcr = (mcr & up->mcr_mask) | up->mcr_force | up->mcr;
......
...@@ -8,6 +8,7 @@ config SERIAL_8250 ...@@ -8,6 +8,7 @@ config SERIAL_8250
tristate "8250/16550 and compatible serial support" tristate "8250/16550 and compatible serial support"
depends on !S390 depends on !S390
select SERIAL_CORE select SERIAL_CORE
select SERIAL_MCTRL_GPIO if GPIOLIB
---help--- ---help---
This selects whether you want to include the driver for the standard This selects whether you want to include the driver for the standard
serial ports. The standard answer is Y. People who might say N serial ports. The standard answer is Y. People who might say N
......
...@@ -457,20 +457,6 @@ config SERIAL_21285_CONSOLE ...@@ -457,20 +457,6 @@ config SERIAL_21285_CONSOLE
your boot loader (lilo or loadlin) about how to pass options to the your boot loader (lilo or loadlin) about how to pass options to the
kernel at boot time.) kernel at boot time.)
config SERIAL_MPSC
bool "Marvell MPSC serial port support"
depends on MV64X60
select SERIAL_CORE
help
Say Y here if you want to use the Marvell MPSC serial controller.
config SERIAL_MPSC_CONSOLE
bool "Support for console on Marvell MPSC serial port"
depends on SERIAL_MPSC
select SERIAL_CORE_CONSOLE
help
Say Y here if you want to support a serial console on a Marvell MPSC.
config SERIAL_PXA config SERIAL_PXA
bool "PXA serial port support (DEPRECATED)" bool "PXA serial port support (DEPRECATED)"
depends on ARCH_PXA || ARCH_MMP depends on ARCH_PXA || ARCH_MMP
......
...@@ -46,7 +46,6 @@ obj-$(CONFIG_SERIAL_CPM) += cpm_uart/ ...@@ -46,7 +46,6 @@ obj-$(CONFIG_SERIAL_CPM) += cpm_uart/
obj-$(CONFIG_SERIAL_IMX) += imx.o obj-$(CONFIG_SERIAL_IMX) += imx.o
obj-$(CONFIG_SERIAL_MPC52xx) += mpc52xx_uart.o obj-$(CONFIG_SERIAL_MPC52xx) += mpc52xx_uart.o
obj-$(CONFIG_SERIAL_ICOM) += icom.o obj-$(CONFIG_SERIAL_ICOM) += icom.o
obj-$(CONFIG_SERIAL_MPSC) += mpsc.o
obj-$(CONFIG_SERIAL_MESON) += meson_uart.o obj-$(CONFIG_SERIAL_MESON) += meson_uart.o
obj-$(CONFIG_SERIAL_SB1250_DUART) += sb1250-duart.o obj-$(CONFIG_SERIAL_SB1250_DUART) += sb1250-duart.o
obj-$(CONFIG_SERIAL_SCCNXP) += sccnxp.o obj-$(CONFIG_SERIAL_SCCNXP) += sccnxp.o
......
...@@ -1717,7 +1717,7 @@ static int pl011_allocate_irq(struct uart_amba_port *uap) ...@@ -1717,7 +1717,7 @@ static int pl011_allocate_irq(struct uart_amba_port *uap)
{ {
pl011_write(uap->im, uap, REG_IMSC); pl011_write(uap->im, uap, REG_IMSC);
return request_irq(uap->port.irq, pl011_int, 0, "uart-pl011", uap); return request_irq(uap->port.irq, pl011_int, IRQF_SHARED, "uart-pl011", uap);
} }
/* /*
......
...@@ -407,7 +407,16 @@ static int cpm_uart_startup(struct uart_port *port) ...@@ -407,7 +407,16 @@ static int cpm_uart_startup(struct uart_port *port)
clrbits16(&pinfo->sccp->scc_sccm, UART_SCCM_RX); clrbits16(&pinfo->sccp->scc_sccm, UART_SCCM_RX);
} }
cpm_uart_initbd(pinfo); cpm_uart_initbd(pinfo);
cpm_line_cr_cmd(pinfo, CPM_CR_INIT_TRX); if (IS_SMC(pinfo)) {
out_be32(&pinfo->smcup->smc_rstate, 0);
out_be32(&pinfo->smcup->smc_tstate, 0);
out_be16(&pinfo->smcup->smc_rbptr,
in_be16(&pinfo->smcup->smc_rbase));
out_be16(&pinfo->smcup->smc_tbptr,
in_be16(&pinfo->smcup->smc_tbase));
} else {
cpm_line_cr_cmd(pinfo, CPM_CR_INIT_TRX);
}
} }
/* Install interrupt handler. */ /* Install interrupt handler. */
retval = request_irq(port->irq, cpm_uart_int, 0, "cpm_uart", port); retval = request_irq(port->irq, cpm_uart_int, 0, "cpm_uart", port);
...@@ -567,8 +576,6 @@ static void cpm_uart_set_termios(struct uart_port *port, ...@@ -567,8 +576,6 @@ static void cpm_uart_set_termios(struct uart_port *port,
/* /*
* Set up parity check flag * Set up parity check flag
*/ */
#define RELEVANT_IFLAG(iflag) (iflag & (IGNBRK|BRKINT|IGNPAR|PARMRK|INPCK))
port->read_status_mask = (BD_SC_EMPTY | BD_SC_OV); port->read_status_mask = (BD_SC_EMPTY | BD_SC_OV);
if (termios->c_iflag & INPCK) if (termios->c_iflag & INPCK)
port->read_status_mask |= BD_SC_FR | BD_SC_PR; port->read_status_mask |= BD_SC_FR | BD_SC_PR;
...@@ -861,16 +868,14 @@ static void cpm_uart_init_smc(struct uart_cpm_port *pinfo) ...@@ -861,16 +868,14 @@ static void cpm_uart_init_smc(struct uart_cpm_port *pinfo)
(u8 __iomem *)pinfo->tx_bd_base - DPRAM_BASE); (u8 __iomem *)pinfo->tx_bd_base - DPRAM_BASE);
/* /*
* In case SMC1 is being relocated... * In case SMC is being relocated...
*/ */
#if defined (CONFIG_I2C_SPI_SMC1_UCODE_PATCH)
out_be16(&up->smc_rbptr, in_be16(&pinfo->smcup->smc_rbase)); out_be16(&up->smc_rbptr, in_be16(&pinfo->smcup->smc_rbase));
out_be16(&up->smc_tbptr, in_be16(&pinfo->smcup->smc_tbase)); out_be16(&up->smc_tbptr, in_be16(&pinfo->smcup->smc_tbase));
out_be32(&up->smc_rstate, 0); out_be32(&up->smc_rstate, 0);
out_be32(&up->smc_tstate, 0); out_be32(&up->smc_tstate, 0);
out_be16(&up->smc_brkcr, 1); /* number of break chars */ out_be16(&up->smc_brkcr, 1); /* number of break chars */
out_be16(&up->smc_brkec, 0); out_be16(&up->smc_brkec, 0);
#endif
/* Set up the uart parameters in the /* Set up the uart parameters in the
* parameter ram. * parameter ram.
...@@ -884,8 +889,6 @@ static void cpm_uart_init_smc(struct uart_cpm_port *pinfo) ...@@ -884,8 +889,6 @@ static void cpm_uart_init_smc(struct uart_cpm_port *pinfo)
out_be16(&up->smc_brkec, 0); out_be16(&up->smc_brkec, 0);
out_be16(&up->smc_brkcr, 1); out_be16(&up->smc_brkcr, 1);
cpm_line_cr_cmd(pinfo, CPM_CR_INIT_TRX);
/* Set UART mode, 8 bit, no parity, one stop. /* Set UART mode, 8 bit, no parity, one stop.
* Enable receive and transmit. * Enable receive and transmit.
*/ */
......
...@@ -541,7 +541,11 @@ static int __init digicolor_uart_init(void) ...@@ -541,7 +541,11 @@ static int __init digicolor_uart_init(void)
if (ret) if (ret)
return ret; return ret;
return platform_driver_register(&digicolor_uart_platform); ret = platform_driver_register(&digicolor_uart_platform);
if (ret)
uart_unregister_driver(&digicolor_uart);
return ret;
} }
module_init(digicolor_uart_init); module_init(digicolor_uart_init);
......
...@@ -234,9 +234,18 @@ ...@@ -234,9 +234,18 @@
static DEFINE_IDA(fsl_lpuart_ida); static DEFINE_IDA(fsl_lpuart_ida);
enum lpuart_type {
VF610_LPUART,
LS1021A_LPUART,
IMX7ULP_LPUART,
IMX8QXP_LPUART,
};
struct lpuart_port { struct lpuart_port {
struct uart_port port; struct uart_port port;
struct clk *clk; enum lpuart_type devtype;
struct clk *ipg_clk;
struct clk *baud_clk;
unsigned int txfifo_size; unsigned int txfifo_size;
unsigned int rxfifo_size; unsigned int rxfifo_size;
...@@ -261,19 +270,29 @@ struct lpuart_port { ...@@ -261,19 +270,29 @@ struct lpuart_port {
}; };
struct lpuart_soc_data { struct lpuart_soc_data {
char iotype; enum lpuart_type devtype;
u8 reg_off; char iotype;
u8 reg_off;
}; };
static const struct lpuart_soc_data vf_data = { static const struct lpuart_soc_data vf_data = {
.devtype = VF610_LPUART,
.iotype = UPIO_MEM, .iotype = UPIO_MEM,
}; };
static const struct lpuart_soc_data ls_data = { static const struct lpuart_soc_data ls_data = {
.devtype = LS1021A_LPUART,
.iotype = UPIO_MEM32BE, .iotype = UPIO_MEM32BE,
}; };
static struct lpuart_soc_data imx_data = { static struct lpuart_soc_data imx7ulp_data = {
.devtype = IMX7ULP_LPUART,
.iotype = UPIO_MEM32,
.reg_off = IMX_REG_OFF,
};
static struct lpuart_soc_data imx8qxp_data = {
.devtype = IMX8QXP_LPUART,
.iotype = UPIO_MEM32, .iotype = UPIO_MEM32,
.reg_off = IMX_REG_OFF, .reg_off = IMX_REG_OFF,
}; };
...@@ -281,7 +300,8 @@ static struct lpuart_soc_data imx_data = { ...@@ -281,7 +300,8 @@ static struct lpuart_soc_data imx_data = {
static const struct of_device_id lpuart_dt_ids[] = { static const struct of_device_id lpuart_dt_ids[] = {
{ .compatible = "fsl,vf610-lpuart", .data = &vf_data, }, { .compatible = "fsl,vf610-lpuart", .data = &vf_data, },
{ .compatible = "fsl,ls1021a-lpuart", .data = &ls_data, }, { .compatible = "fsl,ls1021a-lpuart", .data = &ls_data, },
{ .compatible = "fsl,imx7ulp-lpuart", .data = &imx_data, }, { .compatible = "fsl,imx7ulp-lpuart", .data = &imx7ulp_data, },
{ .compatible = "fsl,imx8qxp-lpuart", .data = &imx8qxp_data, },
{ /* sentinel */ } { /* sentinel */ }
}; };
MODULE_DEVICE_TABLE(of, lpuart_dt_ids); MODULE_DEVICE_TABLE(of, lpuart_dt_ids);
...@@ -289,6 +309,11 @@ MODULE_DEVICE_TABLE(of, lpuart_dt_ids); ...@@ -289,6 +309,11 @@ MODULE_DEVICE_TABLE(of, lpuart_dt_ids);
/* Forward declare this for the dma callbacks*/ /* Forward declare this for the dma callbacks*/
static void lpuart_dma_tx_complete(void *arg); static void lpuart_dma_tx_complete(void *arg);
static inline bool is_imx8qxp_lpuart(struct lpuart_port *sport)
{
return sport->devtype == IMX8QXP_LPUART;
}
static inline u32 lpuart32_read(struct uart_port *port, u32 off) static inline u32 lpuart32_read(struct uart_port *port, u32 off)
{ {
switch (port->iotype) { switch (port->iotype) {
...@@ -314,6 +339,39 @@ static inline void lpuart32_write(struct uart_port *port, u32 val, ...@@ -314,6 +339,39 @@ static inline void lpuart32_write(struct uart_port *port, u32 val,
} }
} }
static int __lpuart_enable_clks(struct lpuart_port *sport, bool is_en)
{
int ret = 0;
if (is_en) {
ret = clk_prepare_enable(sport->ipg_clk);
if (ret)
return ret;
ret = clk_prepare_enable(sport->baud_clk);
if (ret) {
clk_disable_unprepare(sport->ipg_clk);
return ret;
}
} else {
clk_disable_unprepare(sport->baud_clk);
clk_disable_unprepare(sport->ipg_clk);
}
return 0;
}
static unsigned int lpuart_get_baud_clk_rate(struct lpuart_port *sport)
{
if (is_imx8qxp_lpuart(sport))
return clk_get_rate(sport->baud_clk);
return clk_get_rate(sport->ipg_clk);
}
#define lpuart_enable_clks(x) __lpuart_enable_clks(x, true)
#define lpuart_disable_clks(x) __lpuart_enable_clks(x, false)
static void lpuart_stop_tx(struct uart_port *port) static void lpuart_stop_tx(struct uart_port *port)
{ {
unsigned char temp; unsigned char temp;
...@@ -1040,10 +1098,8 @@ static inline int lpuart_start_rx_dma(struct lpuart_port *sport) ...@@ -1040,10 +1098,8 @@ static inline int lpuart_start_rx_dma(struct lpuart_port *sport)
sport->rx_dma_rng_buf_len = 16; sport->rx_dma_rng_buf_len = 16;
ring->buf = kmalloc(sport->rx_dma_rng_buf_len, GFP_ATOMIC); ring->buf = kmalloc(sport->rx_dma_rng_buf_len, GFP_ATOMIC);
if (!ring->buf) { if (!ring->buf)
dev_err(sport->port.dev, "Ring buf alloc failed\n");
return -ENOMEM; return -ENOMEM;
}
sg_init_one(&sport->rx_sgl, ring->buf, sport->rx_dma_rng_buf_len); sg_init_one(&sport->rx_sgl, ring->buf, sport->rx_dma_rng_buf_len);
sg_set_buf(&sport->rx_sgl, ring->buf, sport->rx_dma_rng_buf_len); sg_set_buf(&sport->rx_sgl, ring->buf, sport->rx_dma_rng_buf_len);
...@@ -2071,14 +2127,14 @@ lpuart_console_get_options(struct lpuart_port *sport, int *baud, ...@@ -2071,14 +2127,14 @@ lpuart_console_get_options(struct lpuart_port *sport, int *baud,
brfa = readb(sport->port.membase + UARTCR4); brfa = readb(sport->port.membase + UARTCR4);
brfa &= UARTCR4_BRFA_MASK; brfa &= UARTCR4_BRFA_MASK;
uartclk = clk_get_rate(sport->clk); uartclk = lpuart_get_baud_clk_rate(sport);
/* /*
* baud = mod_clk/(16*(sbr[13]+(brfa)/32) * baud = mod_clk/(16*(sbr[13]+(brfa)/32)
*/ */
baud_raw = uartclk / (16 * (sbr + brfa / 32)); baud_raw = uartclk / (16 * (sbr + brfa / 32));
if (*baud != baud_raw) if (*baud != baud_raw)
printk(KERN_INFO "Serial: Console lpuart rounded baud rate" dev_info(sport->port.dev, "Serial: Console lpuart rounded baud rate"
"from %d to %d\n", baud_raw, *baud); "from %d to %d\n", baud_raw, *baud);
} }
...@@ -2114,14 +2170,14 @@ lpuart32_console_get_options(struct lpuart_port *sport, int *baud, ...@@ -2114,14 +2170,14 @@ lpuart32_console_get_options(struct lpuart_port *sport, int *baud,
bd = lpuart32_read(&sport->port, UARTBAUD); bd = lpuart32_read(&sport->port, UARTBAUD);
bd &= UARTBAUD_SBR_MASK; bd &= UARTBAUD_SBR_MASK;
sbr = bd; sbr = bd;
uartclk = clk_get_rate(sport->clk); uartclk = lpuart_get_baud_clk_rate(sport);
/* /*
* baud = mod_clk/(16*(sbr[13]+(brfa)/32) * baud = mod_clk/(16*(sbr[13]+(brfa)/32)
*/ */
baud_raw = uartclk / (16 * sbr); baud_raw = uartclk / (16 * sbr);
if (*baud != baud_raw) if (*baud != baud_raw)
printk(KERN_INFO "Serial: Console lpuart rounded baud rate" dev_info(sport->port.dev, "Serial: Console lpuart rounded baud rate"
"from %d to %d\n", baud_raw, *baud); "from %d to %d\n", baud_raw, *baud);
} }
...@@ -2288,6 +2344,7 @@ static int lpuart_probe(struct platform_device *pdev) ...@@ -2288,6 +2344,7 @@ static int lpuart_probe(struct platform_device *pdev)
sport->port.mapbase = res->start; sport->port.mapbase = res->start;
sport->port.dev = &pdev->dev; sport->port.dev = &pdev->dev;
sport->port.type = PORT_LPUART; sport->port.type = PORT_LPUART;
sport->devtype = sdata->devtype;
ret = platform_get_irq(pdev, 0); ret = platform_get_irq(pdev, 0);
if (ret < 0) { if (ret < 0) {
dev_err(&pdev->dev, "cannot obtain irq\n"); dev_err(&pdev->dev, "cannot obtain irq\n");
...@@ -2303,20 +2360,27 @@ static int lpuart_probe(struct platform_device *pdev) ...@@ -2303,20 +2360,27 @@ static int lpuart_probe(struct platform_device *pdev)
sport->port.rs485_config = lpuart_config_rs485; sport->port.rs485_config = lpuart_config_rs485;
sport->clk = devm_clk_get(&pdev->dev, "ipg"); sport->ipg_clk = devm_clk_get(&pdev->dev, "ipg");
if (IS_ERR(sport->clk)) { if (IS_ERR(sport->ipg_clk)) {
ret = PTR_ERR(sport->clk); ret = PTR_ERR(sport->ipg_clk);
dev_err(&pdev->dev, "failed to get uart clk: %d\n", ret); dev_err(&pdev->dev, "failed to get uart ipg clk: %d\n", ret);
return ret; return ret;
} }
ret = clk_prepare_enable(sport->clk); sport->baud_clk = NULL;
if (ret) { if (is_imx8qxp_lpuart(sport)) {
dev_err(&pdev->dev, "failed to enable uart clk: %d\n", ret); sport->baud_clk = devm_clk_get(&pdev->dev, "baud");
return ret; if (IS_ERR(sport->baud_clk)) {
ret = PTR_ERR(sport->baud_clk);
dev_err(&pdev->dev, "failed to get uart baud clk: %d\n", ret);
return ret;
}
} }
sport->port.uartclk = clk_get_rate(sport->clk); ret = lpuart_enable_clks(sport);
if (ret)
return ret;
sport->port.uartclk = lpuart_get_baud_clk_rate(sport);
lpuart_ports[sport->port.line] = sport; lpuart_ports[sport->port.line] = sport;
...@@ -2364,7 +2428,7 @@ static int lpuart_probe(struct platform_device *pdev) ...@@ -2364,7 +2428,7 @@ static int lpuart_probe(struct platform_device *pdev)
failed_attach_port: failed_attach_port:
failed_irq_request: failed_irq_request:
clk_disable_unprepare(sport->clk); lpuart_disable_clks(sport);
return ret; return ret;
} }
...@@ -2376,7 +2440,7 @@ static int lpuart_remove(struct platform_device *pdev) ...@@ -2376,7 +2440,7 @@ static int lpuart_remove(struct platform_device *pdev)
ida_simple_remove(&fsl_lpuart_ida, sport->port.line); ida_simple_remove(&fsl_lpuart_ida, sport->port.line);
clk_disable_unprepare(sport->clk); lpuart_disable_clks(sport);
if (sport->dma_tx_chan) if (sport->dma_tx_chan)
dma_release_channel(sport->dma_tx_chan); dma_release_channel(sport->dma_tx_chan);
...@@ -2441,7 +2505,7 @@ static int lpuart_suspend(struct device *dev) ...@@ -2441,7 +2505,7 @@ static int lpuart_suspend(struct device *dev)
} }
if (sport->port.suspended && !irq_wake) if (sport->port.suspended && !irq_wake)
clk_disable_unprepare(sport->clk); lpuart_disable_clks(sport);
return 0; return 0;
} }
...@@ -2453,7 +2517,7 @@ static int lpuart_resume(struct device *dev) ...@@ -2453,7 +2517,7 @@ static int lpuart_resume(struct device *dev)
unsigned long temp; unsigned long temp;
if (sport->port.suspended && !irq_wake) if (sport->port.suspended && !irq_wake)
clk_prepare_enable(sport->clk); lpuart_enable_clks(sport);
if (lpuart_is_32(sport)) { if (lpuart_is_32(sport)) {
lpuart32_setup_watermark(sport); lpuart32_setup_watermark(sport);
......
...@@ -383,6 +383,7 @@ static void imx_uart_ucrs_restore(struct imx_port *sport, ...@@ -383,6 +383,7 @@ static void imx_uart_ucrs_restore(struct imx_port *sport,
} }
#endif #endif
/* called with port.lock taken and irqs caller dependent */
static void imx_uart_rts_active(struct imx_port *sport, u32 *ucr2) static void imx_uart_rts_active(struct imx_port *sport, u32 *ucr2)
{ {
*ucr2 &= ~(UCR2_CTSC | UCR2_CTS); *ucr2 &= ~(UCR2_CTSC | UCR2_CTS);
...@@ -391,6 +392,7 @@ static void imx_uart_rts_active(struct imx_port *sport, u32 *ucr2) ...@@ -391,6 +392,7 @@ static void imx_uart_rts_active(struct imx_port *sport, u32 *ucr2)
mctrl_gpio_set(sport->gpios, sport->port.mctrl); mctrl_gpio_set(sport->gpios, sport->port.mctrl);
} }
/* called with port.lock taken and irqs caller dependent */
static void imx_uart_rts_inactive(struct imx_port *sport, u32 *ucr2) static void imx_uart_rts_inactive(struct imx_port *sport, u32 *ucr2)
{ {
*ucr2 &= ~UCR2_CTSC; *ucr2 &= ~UCR2_CTSC;
...@@ -400,6 +402,7 @@ static void imx_uart_rts_inactive(struct imx_port *sport, u32 *ucr2) ...@@ -400,6 +402,7 @@ static void imx_uart_rts_inactive(struct imx_port *sport, u32 *ucr2)
mctrl_gpio_set(sport->gpios, sport->port.mctrl); mctrl_gpio_set(sport->gpios, sport->port.mctrl);
} }
/* called with port.lock taken and irqs caller dependent */
static void imx_uart_rts_auto(struct imx_port *sport, u32 *ucr2) static void imx_uart_rts_auto(struct imx_port *sport, u32 *ucr2)
{ {
*ucr2 |= UCR2_CTSC; *ucr2 |= UCR2_CTSC;
...@@ -1549,40 +1552,46 @@ imx_uart_set_termios(struct uart_port *port, struct ktermios *termios, ...@@ -1549,40 +1552,46 @@ imx_uart_set_termios(struct uart_port *port, struct ktermios *termios,
old_csize = CS8; old_csize = CS8;
} }
del_timer_sync(&sport->timer);
/*
* Ask the core to calculate the divisor for us.
*/
baud = uart_get_baud_rate(port, termios, old, 50, port->uartclk / 16);
quot = uart_get_divisor(port, baud);
spin_lock_irqsave(&sport->port.lock, flags);
/*
* Read current UCR2 and save it for future use, then clear all the bits
* except those we will or may need to preserve.
*/
old_ucr2 = imx_uart_readl(sport, UCR2);
ucr2 = old_ucr2 & (UCR2_TXEN | UCR2_RXEN | UCR2_ATEN | UCR2_CTS);
ucr2 |= UCR2_SRST | UCR2_IRTS;
if ((termios->c_cflag & CSIZE) == CS8) if ((termios->c_cflag & CSIZE) == CS8)
ucr2 = UCR2_WS | UCR2_SRST | UCR2_IRTS; ucr2 |= UCR2_WS;
else
ucr2 = UCR2_SRST | UCR2_IRTS; if (!sport->have_rtscts)
termios->c_cflag &= ~CRTSCTS;
if (termios->c_cflag & CRTSCTS) {
if (sport->have_rtscts) { if (port->rs485.flags & SER_RS485_ENABLED) {
ucr2 &= ~UCR2_IRTS; /*
* RTS is mandatory for rs485 operation, so keep
if (port->rs485.flags & SER_RS485_ENABLED) { * it under manual control and keep transmitter
/* * disabled.
* RTS is mandatory for rs485 operation, so keep */
* it under manual control and keep transmitter
* disabled.
*/
if (port->rs485.flags &
SER_RS485_RTS_AFTER_SEND)
imx_uart_rts_active(sport, &ucr2);
else
imx_uart_rts_inactive(sport, &ucr2);
} else {
imx_uart_rts_auto(sport, &ucr2);
}
} else {
termios->c_cflag &= ~CRTSCTS;
}
} else if (port->rs485.flags & SER_RS485_ENABLED) {
/* disable transmitter */
if (port->rs485.flags & SER_RS485_RTS_AFTER_SEND) if (port->rs485.flags & SER_RS485_RTS_AFTER_SEND)
imx_uart_rts_active(sport, &ucr2); imx_uart_rts_active(sport, &ucr2);
else else
imx_uart_rts_inactive(sport, &ucr2); imx_uart_rts_inactive(sport, &ucr2);
}
} else if (termios->c_cflag & CRTSCTS)
imx_uart_rts_auto(sport, &ucr2);
if (termios->c_cflag & CRTSCTS)
ucr2 &= ~UCR2_IRTS;
if (termios->c_cflag & CSTOPB) if (termios->c_cflag & CSTOPB)
ucr2 |= UCR2_STPB; ucr2 |= UCR2_STPB;
...@@ -1592,16 +1601,6 @@ imx_uart_set_termios(struct uart_port *port, struct ktermios *termios, ...@@ -1592,16 +1601,6 @@ imx_uart_set_termios(struct uart_port *port, struct ktermios *termios,
ucr2 |= UCR2_PROE; ucr2 |= UCR2_PROE;
} }
del_timer_sync(&sport->timer);
/*
* Ask the core to calculate the divisor for us.
*/
baud = uart_get_baud_rate(port, termios, old, 50, port->uartclk / 16);
quot = uart_get_divisor(port, baud);
spin_lock_irqsave(&sport->port.lock, flags);
sport->port.read_status_mask = 0; sport->port.read_status_mask = 0;
if (termios->c_iflag & INPCK) if (termios->c_iflag & INPCK)
sport->port.read_status_mask |= (URXD_FRMERR | URXD_PRERR); sport->port.read_status_mask |= (URXD_FRMERR | URXD_PRERR);
...@@ -1639,7 +1638,6 @@ imx_uart_set_termios(struct uart_port *port, struct ktermios *termios, ...@@ -1639,7 +1638,6 @@ imx_uart_set_termios(struct uart_port *port, struct ktermios *termios,
imx_uart_writel(sport, imx_uart_writel(sport,
old_ucr1 & ~(UCR1_TXMPTYEN | UCR1_RRDYEN | UCR1_RTSDEN), old_ucr1 & ~(UCR1_TXMPTYEN | UCR1_RRDYEN | UCR1_RTSDEN),
UCR1); UCR1);
old_ucr2 = imx_uart_readl(sport, UCR2);
imx_uart_writel(sport, old_ucr2 & ~UCR2_ATEN, UCR2); imx_uart_writel(sport, old_ucr2 & ~UCR2_ATEN, UCR2);
while (!(imx_uart_readl(sport, USR2) & USR2_TXDC)) while (!(imx_uart_readl(sport, USR2) & USR2_TXDC))
...@@ -1647,7 +1645,6 @@ imx_uart_set_termios(struct uart_port *port, struct ktermios *termios, ...@@ -1647,7 +1645,6 @@ imx_uart_set_termios(struct uart_port *port, struct ktermios *termios,
/* then, disable everything */ /* then, disable everything */
imx_uart_writel(sport, old_ucr2 & ~(UCR2_TXEN | UCR2_RXEN | UCR2_ATEN), UCR2); imx_uart_writel(sport, old_ucr2 & ~(UCR2_TXEN | UCR2_RXEN | UCR2_ATEN), UCR2);
old_ucr2 &= (UCR2_TXEN | UCR2_RXEN | UCR2_ATEN);
/* custom-baudrate handling */ /* custom-baudrate handling */
div = sport->port.uartclk / (baud * 16); div = sport->port.uartclk / (baud * 16);
...@@ -1685,8 +1682,7 @@ imx_uart_set_termios(struct uart_port *port, struct ktermios *termios, ...@@ -1685,8 +1682,7 @@ imx_uart_set_termios(struct uart_port *port, struct ktermios *termios,
imx_uart_writel(sport, old_ucr1, UCR1); imx_uart_writel(sport, old_ucr1, UCR1);
/* set the parity, stop bits and data size */ imx_uart_writel(sport, ucr2, UCR2);
imx_uart_writel(sport, ucr2 | old_ucr2, UCR2);
if (UART_ENABLE_MS(&sport->port, termios->c_cflag)) if (UART_ENABLE_MS(&sport->port, termios->c_cflag))
imx_uart_enable_ms(&sport->port); imx_uart_enable_ms(&sport->port);
...@@ -2015,7 +2011,7 @@ imx_uart_console_get_options(struct imx_port *sport, int *baud, ...@@ -2015,7 +2011,7 @@ imx_uart_console_get_options(struct imx_port *sport, int *baud,
} }
if (*baud != baud_raw) if (*baud != baud_raw)
pr_info("Console IMX rounded baud rate from %d to %d\n", dev_info(sport->port.dev, "Console IMX rounded baud rate from %d to %d\n",
baud_raw, *baud); baud_raw, *baud);
} }
} }
......
This diff is collapsed.
This diff is collapsed.
...@@ -383,10 +383,14 @@ static void msm_request_rx_dma(struct msm_port *msm_port, resource_size_t base) ...@@ -383,10 +383,14 @@ static void msm_request_rx_dma(struct msm_port *msm_port, resource_size_t base)
static inline void msm_wait_for_xmitr(struct uart_port *port) static inline void msm_wait_for_xmitr(struct uart_port *port)
{ {
unsigned int timeout = 500000;
while (!(msm_read(port, UART_SR) & UART_SR_TX_EMPTY)) { while (!(msm_read(port, UART_SR) & UART_SR_TX_EMPTY)) {
if (msm_read(port, UART_ISR) & UART_ISR_TX_READY) if (msm_read(port, UART_ISR) & UART_ISR_TX_READY)
break; break;
udelay(1); udelay(1);
if (!timeout--)
break;
} }
msm_write(port, UART_CR_CMD_RESET_TX_READY, UART_CR); msm_write(port, UART_CR_CMD_RESET_TX_READY, UART_CR);
} }
......
...@@ -1777,6 +1777,7 @@ static int uart_port_activate(struct tty_port *port, struct tty_struct *tty) ...@@ -1777,6 +1777,7 @@ static int uart_port_activate(struct tty_port *port, struct tty_struct *tty)
{ {
struct uart_state *state = container_of(port, struct uart_state, port); struct uart_state *state = container_of(port, struct uart_state, port);
struct uart_port *uport; struct uart_port *uport;
int ret;
uport = uart_port_check(state); uport = uart_port_check(state);
if (!uport || uport->flags & UPF_DEAD) if (!uport || uport->flags & UPF_DEAD)
...@@ -1787,7 +1788,11 @@ static int uart_port_activate(struct tty_port *port, struct tty_struct *tty) ...@@ -1787,7 +1788,11 @@ static int uart_port_activate(struct tty_port *port, struct tty_struct *tty)
/* /*
* Start up the serial port. * Start up the serial port.
*/ */
return uart_startup(tty, state, 0); ret = uart_startup(tty, state, 0);
if (ret > 0)
tty_port_set_active(port, 1);
return ret;
} }
static const char *uart_type(struct uart_port *port) static const char *uart_type(struct uart_port *port)
......
...@@ -12,6 +12,7 @@ ...@@ -12,6 +12,7 @@
#include <linux/termios.h> #include <linux/termios.h>
#include <linux/serial_core.h> #include <linux/serial_core.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/property.h>
#include "serial_mctrl_gpio.h" #include "serial_mctrl_gpio.h"
...@@ -116,6 +117,19 @@ struct mctrl_gpios *mctrl_gpio_init_noauto(struct device *dev, unsigned int idx) ...@@ -116,6 +117,19 @@ struct mctrl_gpios *mctrl_gpio_init_noauto(struct device *dev, unsigned int idx)
for (i = 0; i < UART_GPIO_MAX; i++) { for (i = 0; i < UART_GPIO_MAX; i++) {
enum gpiod_flags flags; enum gpiod_flags flags;
char *gpio_str;
bool present;
/* Check if GPIO property exists and continue if not */
gpio_str = kasprintf(GFP_KERNEL, "%s-gpios",
mctrl_gpios_desc[i].name);
if (!gpio_str)
continue;
present = device_property_present(dev, gpio_str);
kfree(gpio_str);
if (!present)
continue;
if (mctrl_gpios_desc[i].dir_out) if (mctrl_gpios_desc[i].dir_out)
flags = GPIOD_OUT_LOW; flags = GPIOD_OUT_LOW;
......
...@@ -1398,6 +1398,7 @@ static void sci_dma_tx_work_fn(struct work_struct *work) ...@@ -1398,6 +1398,7 @@ static void sci_dma_tx_work_fn(struct work_struct *work)
struct circ_buf *xmit = &port->state->xmit; struct circ_buf *xmit = &port->state->xmit;
unsigned long flags; unsigned long flags;
dma_addr_t buf; dma_addr_t buf;
int head, tail;
/* /*
* DMA is idle now. * DMA is idle now.
...@@ -1407,16 +1408,23 @@ static void sci_dma_tx_work_fn(struct work_struct *work) ...@@ -1407,16 +1408,23 @@ static void sci_dma_tx_work_fn(struct work_struct *work)
* consistent xmit buffer state. * consistent xmit buffer state.
*/ */
spin_lock_irq(&port->lock); spin_lock_irq(&port->lock);
buf = s->tx_dma_addr + (xmit->tail & (UART_XMIT_SIZE - 1)); head = xmit->head;
tail = xmit->tail;
buf = s->tx_dma_addr + (tail & (UART_XMIT_SIZE - 1));
s->tx_dma_len = min_t(unsigned int, s->tx_dma_len = min_t(unsigned int,
CIRC_CNT(xmit->head, xmit->tail, UART_XMIT_SIZE), CIRC_CNT(head, tail, UART_XMIT_SIZE),
CIRC_CNT_TO_END(xmit->head, xmit->tail, UART_XMIT_SIZE)); CIRC_CNT_TO_END(head, tail, UART_XMIT_SIZE));
spin_unlock_irq(&port->lock); if (!s->tx_dma_len) {
/* Transmit buffer has been flushed */
spin_unlock_irq(&port->lock);
return;
}
desc = dmaengine_prep_slave_single(chan, buf, s->tx_dma_len, desc = dmaengine_prep_slave_single(chan, buf, s->tx_dma_len,
DMA_MEM_TO_DEV, DMA_MEM_TO_DEV,
DMA_PREP_INTERRUPT | DMA_CTRL_ACK); DMA_PREP_INTERRUPT | DMA_CTRL_ACK);
if (!desc) { if (!desc) {
spin_unlock_irq(&port->lock);
dev_warn(port->dev, "Failed preparing Tx DMA descriptor\n"); dev_warn(port->dev, "Failed preparing Tx DMA descriptor\n");
goto switch_to_pio; goto switch_to_pio;
} }
...@@ -1424,18 +1432,18 @@ static void sci_dma_tx_work_fn(struct work_struct *work) ...@@ -1424,18 +1432,18 @@ static void sci_dma_tx_work_fn(struct work_struct *work)
dma_sync_single_for_device(chan->device->dev, buf, s->tx_dma_len, dma_sync_single_for_device(chan->device->dev, buf, s->tx_dma_len,
DMA_TO_DEVICE); DMA_TO_DEVICE);
spin_lock_irq(&port->lock);
desc->callback = sci_dma_tx_complete; desc->callback = sci_dma_tx_complete;
desc->callback_param = s; desc->callback_param = s;
spin_unlock_irq(&port->lock);
s->cookie_tx = dmaengine_submit(desc); s->cookie_tx = dmaengine_submit(desc);
if (dma_submit_error(s->cookie_tx)) { if (dma_submit_error(s->cookie_tx)) {
spin_unlock_irq(&port->lock);
dev_warn(port->dev, "Failed submitting Tx DMA descriptor\n"); dev_warn(port->dev, "Failed submitting Tx DMA descriptor\n");
goto switch_to_pio; goto switch_to_pio;
} }
spin_unlock_irq(&port->lock);
dev_dbg(port->dev, "%s: %p: %d...%d, cookie %d\n", dev_dbg(port->dev, "%s: %p: %d...%d, cookie %d\n",
__func__, xmit->buf, xmit->tail, xmit->head, s->cookie_tx); __func__, xmit->buf, tail, head, s->cookie_tx);
dma_async_issue_pending(chan); dma_async_issue_pending(chan);
return; return;
...@@ -1648,11 +1656,18 @@ static void sci_free_dma(struct uart_port *port) ...@@ -1648,11 +1656,18 @@ static void sci_free_dma(struct uart_port *port)
static void sci_flush_buffer(struct uart_port *port) static void sci_flush_buffer(struct uart_port *port)
{ {
struct sci_port *s = to_sci_port(port);
/* /*
* In uart_flush_buffer(), the xmit circular buffer has just been * In uart_flush_buffer(), the xmit circular buffer has just been
* cleared, so we have to reset tx_dma_len accordingly. * cleared, so we have to reset tx_dma_len accordingly, and stop any
* pending transfers
*/ */
to_sci_port(port)->tx_dma_len = 0; s->tx_dma_len = 0;
if (s->chan_tx) {
dmaengine_terminate_async(s->chan_tx);
s->cookie_tx = -EINVAL;
}
} }
#else /* !CONFIG_SERIAL_SH_SCI_DMA */ #else /* !CONFIG_SERIAL_SH_SCI_DMA */
static inline void sci_request_dma(struct uart_port *port) static inline void sci_request_dma(struct uart_port *port)
......
This diff is collapsed.
...@@ -27,6 +27,7 @@ struct stm32_usart_config { ...@@ -27,6 +27,7 @@ struct stm32_usart_config {
bool has_7bits_data; bool has_7bits_data;
bool has_wakeup; bool has_wakeup;
bool has_fifo; bool has_fifo;
int fifosize;
}; };
struct stm32_usart_info { struct stm32_usart_info {
...@@ -54,6 +55,7 @@ struct stm32_usart_info stm32f4_info = { ...@@ -54,6 +55,7 @@ struct stm32_usart_info stm32f4_info = {
.cfg = { .cfg = {
.uart_enable_bit = 13, .uart_enable_bit = 13,
.has_7bits_data = false, .has_7bits_data = false,
.fifosize = 1,
} }
}; };
...@@ -74,6 +76,7 @@ struct stm32_usart_info stm32f7_info = { ...@@ -74,6 +76,7 @@ struct stm32_usart_info stm32f7_info = {
.cfg = { .cfg = {
.uart_enable_bit = 0, .uart_enable_bit = 0,
.has_7bits_data = true, .has_7bits_data = true,
.fifosize = 1,
} }
}; };
...@@ -96,6 +99,7 @@ struct stm32_usart_info stm32h7_info = { ...@@ -96,6 +99,7 @@ struct stm32_usart_info stm32h7_info = {
.has_7bits_data = true, .has_7bits_data = true,
.has_wakeup = true, .has_wakeup = true,
.has_fifo = true, .has_fifo = true,
.fifosize = 16,
} }
}; };
...@@ -108,7 +112,6 @@ struct stm32_usart_info stm32h7_info = { ...@@ -108,7 +112,6 @@ struct stm32_usart_info stm32h7_info = {
#define USART_SR_RXNE BIT(5) #define USART_SR_RXNE BIT(5)
#define USART_SR_TC BIT(6) #define USART_SR_TC BIT(6)
#define USART_SR_TXE BIT(7) #define USART_SR_TXE BIT(7)
#define USART_SR_LBD BIT(8)
#define USART_SR_CTSIF BIT(9) #define USART_SR_CTSIF BIT(9)
#define USART_SR_CTS BIT(10) /* F7 */ #define USART_SR_CTS BIT(10) /* F7 */
#define USART_SR_RTOF BIT(11) /* F7 */ #define USART_SR_RTOF BIT(11) /* F7 */
...@@ -120,8 +123,7 @@ struct stm32_usart_info stm32h7_info = { ...@@ -120,8 +123,7 @@ struct stm32_usart_info stm32h7_info = {
#define USART_SR_SBKF BIT(18) /* F7 */ #define USART_SR_SBKF BIT(18) /* F7 */
#define USART_SR_WUF BIT(20) /* H7 */ #define USART_SR_WUF BIT(20) /* H7 */
#define USART_SR_TEACK BIT(21) /* F7 */ #define USART_SR_TEACK BIT(21) /* F7 */
#define USART_SR_ERR_MASK (USART_SR_LBD | USART_SR_ORE | \ #define USART_SR_ERR_MASK (USART_SR_ORE | USART_SR_FE | USART_SR_PE)
USART_SR_FE | USART_SR_PE)
/* Dummy bits */ /* Dummy bits */
#define USART_SR_DUMMY_RX BIT(16) #define USART_SR_DUMMY_RX BIT(16)
...@@ -151,8 +153,7 @@ struct stm32_usart_info stm32h7_info = { ...@@ -151,8 +153,7 @@ struct stm32_usart_info stm32h7_info = {
#define USART_CR1_PS BIT(9) #define USART_CR1_PS BIT(9)
#define USART_CR1_PCE BIT(10) #define USART_CR1_PCE BIT(10)
#define USART_CR1_WAKE BIT(11) #define USART_CR1_WAKE BIT(11)
#define USART_CR1_M BIT(12) #define USART_CR1_M0 BIT(12) /* F7 (CR1_M for F4) */
#define USART_CR1_M0 BIT(12) /* F7 */
#define USART_CR1_MME BIT(13) /* F7 */ #define USART_CR1_MME BIT(13) /* F7 */
#define USART_CR1_CMIE BIT(14) /* F7 */ #define USART_CR1_CMIE BIT(14) /* F7 */
#define USART_CR1_OVER8 BIT(15) #define USART_CR1_OVER8 BIT(15)
...@@ -169,8 +170,6 @@ struct stm32_usart_info stm32h7_info = { ...@@ -169,8 +170,6 @@ struct stm32_usart_info stm32h7_info = {
/* USART_CR2 */ /* USART_CR2 */
#define USART_CR2_ADD_MASK GENMASK(3, 0) /* F4 */ #define USART_CR2_ADD_MASK GENMASK(3, 0) /* F4 */
#define USART_CR2_ADDM7 BIT(4) /* F7 */ #define USART_CR2_ADDM7 BIT(4) /* F7 */
#define USART_CR2_LBDL BIT(5)
#define USART_CR2_LBDIE BIT(6)
#define USART_CR2_LBCL BIT(8) #define USART_CR2_LBCL BIT(8)
#define USART_CR2_CPHA BIT(9) #define USART_CR2_CPHA BIT(9)
#define USART_CR2_CPOL BIT(10) #define USART_CR2_CPOL BIT(10)
...@@ -209,6 +208,19 @@ struct stm32_usart_info stm32h7_info = { ...@@ -209,6 +208,19 @@ struct stm32_usart_info stm32h7_info = {
#define USART_CR3_WUS_MASK GENMASK(21, 20) /* H7 */ #define USART_CR3_WUS_MASK GENMASK(21, 20) /* H7 */
#define USART_CR3_WUS_START_BIT BIT(21) /* H7 */ #define USART_CR3_WUS_START_BIT BIT(21) /* H7 */
#define USART_CR3_WUFIE BIT(22) /* H7 */ #define USART_CR3_WUFIE BIT(22) /* H7 */
#define USART_CR3_TXFTIE BIT(23) /* H7 */
#define USART_CR3_TCBGTIE BIT(24) /* H7 */
#define USART_CR3_RXFTCFG_MASK GENMASK(27, 25) /* H7 */
#define USART_CR3_RXFTCFG_SHIFT 25 /* H7 */
#define USART_CR3_RXFTIE BIT(28) /* H7 */
#define USART_CR3_TXFTCFG_MASK GENMASK(31, 29) /* H7 */
#define USART_CR3_TXFTCFG_SHIFT 29 /* H7 */
/* TX FIFO threashold set to half of its depth */
#define USART_CR3_TXFTCFG_HALF 0x2
/* RX FIFO threashold set to half of its depth */
#define USART_CR3_RXFTCFG_HALF 0x2
/* USART_GTPR */ /* USART_GTPR */
#define USART_GTPR_PSC_MASK GENMASK(7, 0) #define USART_GTPR_PSC_MASK GENMASK(7, 0)
...@@ -227,12 +239,10 @@ struct stm32_usart_info stm32h7_info = { ...@@ -227,12 +239,10 @@ struct stm32_usart_info stm32h7_info = {
/* USART_ICR */ /* USART_ICR */
#define USART_ICR_PECF BIT(0) /* F7 */ #define USART_ICR_PECF BIT(0) /* F7 */
#define USART_ICR_FFECF BIT(1) /* F7 */ #define USART_ICR_FECF BIT(1) /* F7 */
#define USART_ICR_NCF BIT(2) /* F7 */
#define USART_ICR_ORECF BIT(3) /* F7 */ #define USART_ICR_ORECF BIT(3) /* F7 */
#define USART_ICR_IDLECF BIT(4) /* F7 */ #define USART_ICR_IDLECF BIT(4) /* F7 */
#define USART_ICR_TCCF BIT(6) /* F7 */ #define USART_ICR_TCCF BIT(6) /* F7 */
#define USART_ICR_LBDCF BIT(8) /* F7 */
#define USART_ICR_CTSCF BIT(9) /* F7 */ #define USART_ICR_CTSCF BIT(9) /* F7 */
#define USART_ICR_RTOCF BIT(11) /* F7 */ #define USART_ICR_RTOCF BIT(11) /* F7 */
#define USART_ICR_EOBCF BIT(12) /* F7 */ #define USART_ICR_EOBCF BIT(12) /* F7 */
...@@ -256,11 +266,14 @@ struct stm32_port { ...@@ -256,11 +266,14 @@ struct stm32_port {
struct dma_chan *tx_ch; /* dma tx channel */ struct dma_chan *tx_ch; /* dma tx channel */
dma_addr_t tx_dma_buf; /* dma tx buffer bus address */ dma_addr_t tx_dma_buf; /* dma tx buffer bus address */
unsigned char *tx_buf; /* dma tx buffer cpu address */ unsigned char *tx_buf; /* dma tx buffer cpu address */
u32 cr1_irq; /* USART_CR1_RXNEIE or RTOIE */
u32 cr3_irq; /* USART_CR3_RXFTIE */
int last_res; int last_res;
bool tx_dma_busy; /* dma tx busy */ bool tx_dma_busy; /* dma tx busy */
bool hw_flow_control; bool hw_flow_control;
bool fifoen; bool fifoen;
int wakeirq; int wakeirq;
int rdr_mask; /* receive data register mask */
}; };
static struct stm32_port stm32_ports[STM32_MAX_PORTS]; static struct stm32_port stm32_ports[STM32_MAX_PORTS];
......
...@@ -26,21 +26,23 @@ ...@@ -26,21 +26,23 @@
#include <linux/of.h> #include <linux/of.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/pm_runtime.h> #include <linux/pm_runtime.h>
#include <linux/iopoll.h>
#define CDNS_UART_TTY_NAME "ttyPS" #define CDNS_UART_TTY_NAME "ttyPS"
#define CDNS_UART_NAME "xuartps" #define CDNS_UART_NAME "xuartps"
#define CDNS_UART_MAJOR 0 /* use dynamic node allocation */
#define CDNS_UART_FIFO_SIZE 64 /* FIFO size */ #define CDNS_UART_FIFO_SIZE 64 /* FIFO size */
#define CDNS_UART_REGISTER_SPACE 0x1000 #define CDNS_UART_REGISTER_SPACE 0x1000
#define TX_TIMEOUT 500000
/* Rx Trigger level */ /* Rx Trigger level */
static int rx_trigger_level = 56; static int rx_trigger_level = 56;
module_param(rx_trigger_level, uint, S_IRUGO); static int uartps_major;
module_param(rx_trigger_level, uint, 0444);
MODULE_PARM_DESC(rx_trigger_level, "Rx trigger level, 1-63 bytes"); MODULE_PARM_DESC(rx_trigger_level, "Rx trigger level, 1-63 bytes");
/* Rx Timeout */ /* Rx Timeout */
static int rx_timeout = 10; static int rx_timeout = 10;
module_param(rx_timeout, uint, S_IRUGO); module_param(rx_timeout, uint, 0444);
MODULE_PARM_DESC(rx_timeout, "Rx timeout, 1-255"); MODULE_PARM_DESC(rx_timeout, "Rx timeout, 1-255");
/* Register offsets for the UART. */ /* Register offsets for the UART. */
...@@ -199,7 +201,7 @@ struct cdns_platform_data { ...@@ -199,7 +201,7 @@ struct cdns_platform_data {
u32 quirks; u32 quirks;
}; };
#define to_cdns_uart(_nb) container_of(_nb, struct cdns_uart, \ #define to_cdns_uart(_nb) container_of(_nb, struct cdns_uart, \
clk_rate_change_nb); clk_rate_change_nb)
/** /**
* cdns_uart_handle_rx - Handle the received bytes along with Rx errors. * cdns_uart_handle_rx - Handle the received bytes along with Rx errors.
...@@ -312,15 +314,16 @@ static void cdns_uart_handle_tx(void *dev_id) ...@@ -312,15 +314,16 @@ static void cdns_uart_handle_tx(void *dev_id)
} else { } else {
numbytes = port->fifosize; numbytes = port->fifosize;
while (numbytes && !uart_circ_empty(&port->state->xmit) && while (numbytes && !uart_circ_empty(&port->state->xmit) &&
!(readl(port->membase + CDNS_UART_SR) & CDNS_UART_SR_TXFULL)) { !(readl(port->membase + CDNS_UART_SR) &
CDNS_UART_SR_TXFULL)) {
/* /*
* Get the data from the UART circular buffer * Get the data from the UART circular buffer
* and write it to the cdns_uart's TX_FIFO * and write it to the cdns_uart's TX_FIFO
* register. * register.
*/ */
writel( writel(
port->state->xmit.buf[port->state->xmit. port->state->xmit.buf[port->state->xmit.tail],
tail], port->membase + CDNS_UART_FIFO); port->membase + CDNS_UART_FIFO);
port->icount.tx++; port->icount.tx++;
...@@ -684,18 +687,21 @@ static void cdns_uart_set_termios(struct uart_port *port, ...@@ -684,18 +687,21 @@ static void cdns_uart_set_termios(struct uart_port *port,
unsigned int cval = 0; unsigned int cval = 0;
unsigned int baud, minbaud, maxbaud; unsigned int baud, minbaud, maxbaud;
unsigned long flags; unsigned long flags;
unsigned int ctrl_reg, mode_reg; unsigned int ctrl_reg, mode_reg, val;
int err;
spin_lock_irqsave(&port->lock, flags);
/* Wait for the transmit FIFO to empty before making changes */ /* Wait for the transmit FIFO to empty before making changes */
if (!(readl(port->membase + CDNS_UART_CR) & if (!(readl(port->membase + CDNS_UART_CR) &
CDNS_UART_CR_TX_DIS)) { CDNS_UART_CR_TX_DIS)) {
while (!(readl(port->membase + CDNS_UART_SR) & err = readl_poll_timeout(port->membase + CDNS_UART_SR,
CDNS_UART_SR_TXEMPTY)) { val, (val & CDNS_UART_SR_TXEMPTY),
cpu_relax(); 1000, TX_TIMEOUT);
if (err) {
dev_err(port->dev, "timed out waiting for tx empty");
return;
} }
} }
spin_lock_irqsave(&port->lock, flags);
/* Disable the TX and RX to set baud rate */ /* Disable the TX and RX to set baud rate */
ctrl_reg = readl(port->membase + CDNS_UART_CR); ctrl_reg = readl(port->membase + CDNS_UART_CR);
...@@ -1073,8 +1079,6 @@ static void cdns_uart_poll_put_char(struct uart_port *port, unsigned char c) ...@@ -1073,8 +1079,6 @@ static void cdns_uart_poll_put_char(struct uart_port *port, unsigned char c)
cpu_relax(); cpu_relax();
spin_unlock_irqrestore(&port->lock, flags); spin_unlock_irqrestore(&port->lock, flags);
return;
} }
#endif #endif
...@@ -1517,7 +1521,7 @@ static int cdns_uart_probe(struct platform_device *pdev) ...@@ -1517,7 +1521,7 @@ static int cdns_uart_probe(struct platform_device *pdev)
cdns_uart_uart_driver->owner = THIS_MODULE; cdns_uart_uart_driver->owner = THIS_MODULE;
cdns_uart_uart_driver->driver_name = driver_name; cdns_uart_uart_driver->driver_name = driver_name;
cdns_uart_uart_driver->dev_name = CDNS_UART_TTY_NAME; cdns_uart_uart_driver->dev_name = CDNS_UART_TTY_NAME;
cdns_uart_uart_driver->major = CDNS_UART_MAJOR; cdns_uart_uart_driver->major = uartps_major;
cdns_uart_uart_driver->minor = cdns_uart_data->id; cdns_uart_uart_driver->minor = cdns_uart_data->id;
cdns_uart_uart_driver->nr = 1; cdns_uart_uart_driver->nr = 1;
...@@ -1546,6 +1550,7 @@ static int cdns_uart_probe(struct platform_device *pdev) ...@@ -1546,6 +1550,7 @@ static int cdns_uart_probe(struct platform_device *pdev)
goto err_out_id; goto err_out_id;
} }
uartps_major = cdns_uart_uart_driver->tty_driver->major;
cdns_uart_data->cdns_uart_driver = cdns_uart_uart_driver; cdns_uart_data->cdns_uart_driver = cdns_uart_uart_driver;
/* /*
......
...@@ -1837,7 +1837,7 @@ static struct tty_struct *tty_open_current_tty(dev_t device, struct file *filp) ...@@ -1837,7 +1837,7 @@ static struct tty_struct *tty_open_current_tty(dev_t device, struct file *filp)
static struct tty_driver *tty_lookup_driver(dev_t device, struct file *filp, static struct tty_driver *tty_lookup_driver(dev_t device, struct file *filp,
int *index) int *index)
{ {
struct tty_driver *driver; struct tty_driver *driver = NULL;
switch (device) { switch (device) {
#ifdef CONFIG_VT #ifdef CONFIG_VT
...@@ -1858,6 +1858,8 @@ static struct tty_driver *tty_lookup_driver(dev_t device, struct file *filp, ...@@ -1858,6 +1858,8 @@ static struct tty_driver *tty_lookup_driver(dev_t device, struct file *filp,
break; break;
} }
} }
if (driver)
tty_driver_kref_put(driver);
return ERR_PTR(-ENODEV); return ERR_PTR(-ENODEV);
} }
default: default:
......
...@@ -918,52 +918,6 @@ ...@@ -918,52 +918,6 @@
extern void mv64340_irq_init(unsigned int base); extern void mv64340_irq_init(unsigned int base);
/* MPSC Platform Device, Driver Data (Shared register regions) */
#define MPSC_SHARED_NAME "mpsc_shared"
#define MPSC_ROUTING_BASE_ORDER 0
#define MPSC_SDMA_INTR_BASE_ORDER 1
#define MPSC_ROUTING_REG_BLOCK_SIZE 0x000c
#define MPSC_SDMA_INTR_REG_BLOCK_SIZE 0x0084
struct mpsc_shared_pdata {
u32 mrr_val;
u32 rcrr_val;
u32 tcrr_val;
u32 intr_cause_val;
u32 intr_mask_val;
};
/* MPSC Platform Device, Driver Data */
#define MPSC_CTLR_NAME "mpsc"
#define MPSC_BASE_ORDER 0
#define MPSC_SDMA_BASE_ORDER 1
#define MPSC_BRG_BASE_ORDER 2
#define MPSC_REG_BLOCK_SIZE 0x0038
#define MPSC_SDMA_REG_BLOCK_SIZE 0x0c18
#define MPSC_BRG_REG_BLOCK_SIZE 0x0008
struct mpsc_pdata {
u8 mirror_regs;
u8 cache_mgmt;
u8 max_idle;
int default_baud;
int default_bits;
int default_parity;
int default_flow;
u32 chr_1_val;
u32 chr_2_val;
u32 chr_10_val;
u32 mpcr_val;
u32 bcr_val;
u8 brg_can_tune;
u8 brg_clk_src;
u32 brg_clk_freq;
};
/* Watchdog Platform Device, Driver Data */ /* Watchdog Platform Device, Driver Data */
#define MV64x60_WDT_NAME "mv64x60_wdt" #define MV64x60_WDT_NAME "mv64x60_wdt"
......
...@@ -110,6 +110,7 @@ struct uart_8250_port { ...@@ -110,6 +110,7 @@ struct uart_8250_port {
* if no_console_suspend * if no_console_suspend
*/ */
unsigned char probe; unsigned char probe;
struct mctrl_gpios *gpios;
#define UART_PROBE_RSA (1 << 0) #define UART_PROBE_RSA (1 << 0)
/* /*
......
...@@ -129,7 +129,7 @@ ...@@ -129,7 +129,7 @@
/* Motorola i.MX SoC */ /* Motorola i.MX SoC */
#define PORT_IMX 62 #define PORT_IMX 62
/* Marvell MPSC */ /* Marvell MPSC (obsolete unused) */
#define PORT_MPSC 63 #define PORT_MPSC 63
/* TXX9 type number */ /* TXX9 type number */
......
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