提交 299245a1 编写于 作者: P Peter Hurley 提交者: Greg Kroah-Hartman

serial: core: Privatize modem status enable flags

The serial core uses the tty port flags, ASYNC_CTS_FLOW and
ASYNC_CD_CHECK, to track whether CTS and DCD changes should be
ignored or handled. However, the tty port flags are not safe for
atomic bit operations and no lock provides serialized updates.

Introduce the struct uart_port status field to track CTS and DCD
enable states, and serialize access with uart port lock. Substitute
uart_cts_enabled() helper for tty_port_cts_enabled().
Signed-off-by: NPeter Hurley <peter@hurleysoftware.com>
Signed-off-by: NGreg Kroah-Hartman <gregkh@linuxfoundation.org>
上级 4d90bb14
...@@ -408,7 +408,7 @@ static void mxs_auart_set_mctrl(struct uart_port *u, unsigned mctrl) ...@@ -408,7 +408,7 @@ static void mxs_auart_set_mctrl(struct uart_port *u, unsigned mctrl)
ctrl &= ~(AUART_CTRL2_RTSEN | AUART_CTRL2_RTS); ctrl &= ~(AUART_CTRL2_RTSEN | AUART_CTRL2_RTS);
if (mctrl & TIOCM_RTS) { if (mctrl & TIOCM_RTS) {
if (tty_port_cts_enabled(&u->state->port)) if (uart_cts_enabled(u))
ctrl |= AUART_CTRL2_RTSEN; ctrl |= AUART_CTRL2_RTSEN;
else else
ctrl |= AUART_CTRL2_RTS; ctrl |= AUART_CTRL2_RTS;
......
...@@ -59,6 +59,11 @@ static void uart_change_pm(struct uart_state *state, ...@@ -59,6 +59,11 @@ static void uart_change_pm(struct uart_state *state,
static void uart_port_shutdown(struct tty_port *port); static void uart_port_shutdown(struct tty_port *port);
static int uart_dcd_enabled(struct uart_port *uport)
{
return uport->status & UPSTAT_DCD_ENABLE;
}
/* /*
* This routine is used by the interrupt handler to schedule processing in * This routine is used by the interrupt handler to schedule processing in
* the software interrupt portion of the driver. * the software interrupt portion of the driver.
...@@ -130,7 +135,6 @@ static int uart_port_startup(struct tty_struct *tty, struct uart_state *state, ...@@ -130,7 +135,6 @@ static int uart_port_startup(struct tty_struct *tty, struct uart_state *state,
int init_hw) int init_hw)
{ {
struct uart_port *uport = state->uart_port; struct uart_port *uport = state->uart_port;
struct tty_port *port = &state->port;
unsigned long page; unsigned long page;
int retval = 0; int retval = 0;
...@@ -176,12 +180,12 @@ static int uart_port_startup(struct tty_struct *tty, struct uart_state *state, ...@@ -176,12 +180,12 @@ static int uart_port_startup(struct tty_struct *tty, struct uart_state *state,
uart_set_mctrl(uport, TIOCM_RTS | TIOCM_DTR); uart_set_mctrl(uport, TIOCM_RTS | TIOCM_DTR);
} }
if (tty_port_cts_enabled(port)) {
spin_lock_irq(&uport->lock); spin_lock_irq(&uport->lock);
if (uart_cts_enabled(uport)) {
if (!(uport->ops->get_mctrl(uport) & TIOCM_CTS)) if (!(uport->ops->get_mctrl(uport) & TIOCM_CTS))
tty->hw_stopped = 1; tty->hw_stopped = 1;
spin_unlock_irq(&uport->lock);
} }
spin_unlock_irq(&uport->lock);
} }
/* /*
...@@ -435,7 +439,6 @@ EXPORT_SYMBOL(uart_get_divisor); ...@@ -435,7 +439,6 @@ EXPORT_SYMBOL(uart_get_divisor);
static void uart_change_speed(struct tty_struct *tty, struct uart_state *state, static void uart_change_speed(struct tty_struct *tty, struct uart_state *state,
struct ktermios *old_termios) struct ktermios *old_termios)
{ {
struct tty_port *port = &state->port;
struct uart_port *uport = state->uart_port; struct uart_port *uport = state->uart_port;
struct ktermios *termios; struct ktermios *termios;
...@@ -450,17 +453,19 @@ static void uart_change_speed(struct tty_struct *tty, struct uart_state *state, ...@@ -450,17 +453,19 @@ static void uart_change_speed(struct tty_struct *tty, struct uart_state *state,
uport->ops->set_termios(uport, termios, old_termios); uport->ops->set_termios(uport, termios, old_termios);
/* /*
* Set flags based on termios cflag * Set modem status enables based on termios cflag
*/ */
spin_lock_irq(&uport->lock);
if (termios->c_cflag & CRTSCTS) if (termios->c_cflag & CRTSCTS)
set_bit(ASYNCB_CTS_FLOW, &port->flags); uport->status |= UPSTAT_CTS_ENABLE;
else else
clear_bit(ASYNCB_CTS_FLOW, &port->flags); uport->status &= ~UPSTAT_CTS_ENABLE;
if (termios->c_cflag & CLOCAL) if (termios->c_cflag & CLOCAL)
clear_bit(ASYNCB_CHECK_CD, &port->flags); uport->status &= ~UPSTAT_DCD_ENABLE;
else else
set_bit(ASYNCB_CHECK_CD, &port->flags); uport->status |= UPSTAT_DCD_ENABLE;
spin_unlock_irq(&uport->lock);
} }
static inline int __uart_put_char(struct uart_port *port, static inline int __uart_put_char(struct uart_port *port,
...@@ -2765,7 +2770,7 @@ void uart_handle_dcd_change(struct uart_port *uport, unsigned int status) ...@@ -2765,7 +2770,7 @@ void uart_handle_dcd_change(struct uart_port *uport, unsigned int status)
uport->icount.dcd++; uport->icount.dcd++;
if (port->flags & ASYNC_CHECK_CD) { if (uart_dcd_enabled(uport)) {
if (status) if (status)
wake_up_interruptible(&port->open_wait); wake_up_interruptible(&port->open_wait);
else if (tty) else if (tty)
...@@ -2790,7 +2795,7 @@ void uart_handle_cts_change(struct uart_port *uport, unsigned int status) ...@@ -2790,7 +2795,7 @@ void uart_handle_cts_change(struct uart_port *uport, unsigned int status)
uport->icount.cts++; uport->icount.cts++;
if (tty_port_cts_enabled(port)) { if (uart_cts_enabled(uport)) {
if (tty->hw_stopped) { if (tty->hw_stopped) {
if (status) { if (status) {
tty->hw_stopped = 0; tty->hw_stopped = 0;
......
...@@ -112,6 +112,7 @@ struct uart_icount { ...@@ -112,6 +112,7 @@ struct uart_icount {
}; };
typedef unsigned int __bitwise__ upf_t; typedef unsigned int __bitwise__ upf_t;
typedef unsigned int __bitwise__ upstat_t;
struct uart_port { struct uart_port {
spinlock_t lock; /* port lock */ spinlock_t lock; /* port lock */
...@@ -190,6 +191,12 @@ struct uart_port { ...@@ -190,6 +191,12 @@ struct uart_port {
#define UPF_CHANGE_MASK ((__force upf_t) (0x17fff)) #define UPF_CHANGE_MASK ((__force upf_t) (0x17fff))
#define UPF_USR_MASK ((__force upf_t) (UPF_SPD_MASK|UPF_LOW_LATENCY)) #define UPF_USR_MASK ((__force upf_t) (UPF_SPD_MASK|UPF_LOW_LATENCY))
/* status must be updated while holding port lock */
upstat_t status;
#define UPSTAT_CTS_ENABLE ((__force upstat_t) (1 << 0))
#define UPSTAT_DCD_ENABLE ((__force upstat_t) (1 << 1))
unsigned int mctrl; /* current modem ctrl settings */ unsigned int mctrl; /* current modem ctrl settings */
unsigned int timeout; /* character-based timeout */ unsigned int timeout; /* character-based timeout */
unsigned int type; /* port type */ unsigned int type; /* port type */
...@@ -355,6 +362,11 @@ static inline int uart_tx_stopped(struct uart_port *port) ...@@ -355,6 +362,11 @@ static inline int uart_tx_stopped(struct uart_port *port)
return 0; return 0;
} }
static inline bool uart_cts_enabled(struct uart_port *uport)
{
return uport->status & UPSTAT_CTS_ENABLE;
}
/* /*
* The following are helper functions for the low level drivers. * The following are helper functions for the low level drivers.
*/ */
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册