Commit a88a69c9 authored by Joe Peterson's avatar Joe Peterson Committed by Linus Torvalds

n_tty: Fix loss of echoed characters and remove bkl from n_tty

Fixes the loss of echoed (and other ldisc-generated characters) when
the tty is stopped or when the driver output buffer is full (happens
frequently for input during continuous program output, such as ^C)
and removes the Big Kernel Lock from the N_TTY line discipline.

Adds an "echo buffer" to the N_TTY line discipline that handles all
ldisc-generated output (including echoed characters).  Along with the
loss of characters, this also fixes the associated loss of sync between
tty output and the ldisc state when characters cannot be immediately
written to the tty driver.

The echo buffer stores (in addition to characters) state operations that need
to be done at the time of character output (like management of the column
position).  This allows echo to cooperate correctly with program output,
since the ldisc state remains consistent with actual characters written.

Since the echo buffer code now isolates the tty column state code
to the process_out* and process_echoes functions, we can remove the
Big Kernel Lock (BKL) and replace it with mutex locks.

Highlights are:

* Handles echo (and other ldisc output) when tty driver buffer is full
  - continuous program output can block echo
* Saves echo when tty is in stopped state (e.g. ^S)
  - (e.g.: ^Q will correctly cause held characters to be released for output)
* Control character pairs (e.g. "^C") are treated atomically and not
  split up by interleaved program output
* Line discipline state is kept consistent with characters sent to
  the tty driver
* Remove the big kernel lock (BKL) from N_TTY line discipline
Signed-off-by: default avatarJoe Peterson <joe@skyrush.com>
Signed-off-by: default avatarAlan Cox <alan@redhat.com>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent e482a237
...@@ -62,6 +62,17 @@ ...@@ -62,6 +62,17 @@
#define TTY_THRESHOLD_THROTTLE 128 /* now based on remaining room */ #define TTY_THRESHOLD_THROTTLE 128 /* now based on remaining room */
#define TTY_THRESHOLD_UNTHROTTLE 128 #define TTY_THRESHOLD_UNTHROTTLE 128
/*
* Special byte codes used in the echo buffer to represent operations
* or special handling of characters. Bytes in the echo buffer that
* are not part of such special blocks are treated as normal character
* codes.
*/
#define ECHO_OP_START 0xff
#define ECHO_OP_MOVE_BACK_COL 0x80
#define ECHO_OP_SET_CANON_COL 0x81
#define ECHO_OP_ERASE_TAB 0x82
static inline unsigned char *alloc_buf(void) static inline unsigned char *alloc_buf(void)
{ {
gfp_t prio = in_interrupt() ? GFP_ATOMIC : GFP_KERNEL; gfp_t prio = in_interrupt() ? GFP_ATOMIC : GFP_KERNEL;
...@@ -169,6 +180,7 @@ static void check_unthrottle(struct tty_struct *tty) ...@@ -169,6 +180,7 @@ static void check_unthrottle(struct tty_struct *tty)
* *
* Locking: tty_read_lock for read fields. * Locking: tty_read_lock for read fields.
*/ */
static void reset_buffer_flags(struct tty_struct *tty) static void reset_buffer_flags(struct tty_struct *tty)
{ {
unsigned long flags; unsigned long flags;
...@@ -176,6 +188,11 @@ static void reset_buffer_flags(struct tty_struct *tty) ...@@ -176,6 +188,11 @@ static void reset_buffer_flags(struct tty_struct *tty)
spin_lock_irqsave(&tty->read_lock, flags); spin_lock_irqsave(&tty->read_lock, flags);
tty->read_head = tty->read_tail = tty->read_cnt = 0; tty->read_head = tty->read_tail = tty->read_cnt = 0;
spin_unlock_irqrestore(&tty->read_lock, flags); spin_unlock_irqrestore(&tty->read_lock, flags);
mutex_lock(&tty->echo_lock);
tty->echo_pos = tty->echo_cnt = tty->echo_overrun = 0;
mutex_unlock(&tty->echo_lock);
tty->canon_head = tty->canon_data = tty->erasing = 0; tty->canon_head = tty->canon_data = tty->erasing = 0;
memset(&tty->read_flags, 0, sizeof tty->read_flags); memset(&tty->read_flags, 0, sizeof tty->read_flags);
n_tty_set_room(tty); n_tty_set_room(tty);
...@@ -266,89 +283,116 @@ static inline int is_continuation(unsigned char c, struct tty_struct *tty) ...@@ -266,89 +283,116 @@ static inline int is_continuation(unsigned char c, struct tty_struct *tty)
} }
/** /**
* opost - output post processor * do_output_char - output one character
* @c: character (or partial unicode symbol) * @c: character (or partial unicode symbol)
* @tty: terminal device * @tty: terminal device
* @space: space available in tty driver write buffer
* *
* Perform OPOST processing. Returns -1 when the output device is * This is a helper function that handles one output character
* full and the character must be retried. Note that Linux currently * (including special characters like TAB, CR, LF, etc.),
* ignores TABDLY, CRDLY, VTDLY, FFDLY and NLDLY. They simply aren't * putting the results in the tty driver's write buffer.
* relevant in the world today. If you ever need them, add them here. *
* Note that Linux currently ignores TABDLY, CRDLY, VTDLY, FFDLY
* and NLDLY. They simply aren't relevant in the world today.
* If you ever need them, add them here.
* *
* Called from both the receive and transmit sides and can be called * Returns the number of bytes of buffer space used or -1 if
* re-entrantly. Relies on lock_kernel() for tty->column state. * no space left.
*
* Locking: should be called under the output_lock to protect
* the column state and space left in the buffer
*/ */
static int opost(unsigned char c, struct tty_struct *tty) static int do_output_char(unsigned char c, struct tty_struct *tty, int space)
{ {
int space, spaces; int spaces;
space = tty_write_room(tty);
if (!space) if (!space)
return -1; return -1;
lock_kernel(); switch (c) {
if (O_OPOST(tty)) { case '\n':
switch (c) { if (O_ONLRET(tty))
case '\n': tty->column = 0;
if (O_ONLRET(tty)) if (O_ONLCR(tty)) {
tty->column = 0; if (space < 2)
if (O_ONLCR(tty)) { return -1;
if (space < 2) {
unlock_kernel();
return -1;
}
tty_put_char(tty, '\r');
tty->column = 0;
}
tty->canon_column = tty->column;
break;
case '\r':
if (O_ONOCR(tty) && tty->column == 0) {
unlock_kernel();
return 0;
}
if (O_OCRNL(tty)) {
c = '\n';
if (O_ONLRET(tty))
tty->canon_column = tty->column = 0;
break;
}
tty->canon_column = tty->column = 0; tty->canon_column = tty->column = 0;
tty_put_char(tty, '\r');
tty_put_char(tty, c);
return 2;
}
tty->canon_column = tty->column;
break;
case '\r':
if (O_ONOCR(tty) && tty->column == 0)
return 0;
if (O_OCRNL(tty)) {
c = '\n';
if (O_ONLRET(tty))
tty->canon_column = tty->column = 0;
break; break;
case '\t': }
spaces = 8 - (tty->column & 7); tty->canon_column = tty->column = 0;
if (O_TABDLY(tty) == XTABS) { break;
if (space < spaces) { case '\t':
unlock_kernel(); spaces = 8 - (tty->column & 7);
return -1; if (O_TABDLY(tty) == XTABS) {
} if (space < spaces)
tty->column += spaces; return -1;
tty->ops->write(tty, " ", spaces);
unlock_kernel();
return 0;
}
tty->column += spaces; tty->column += spaces;
break; tty->ops->write(tty, " ", spaces);
case '\b': return spaces;
if (tty->column > 0)
tty->column--;
break;
default:
if (O_OLCUC(tty))
c = toupper(c);
if (!iscntrl(c) && !is_continuation(c, tty))
tty->column++;
break;
} }
tty->column += spaces;
break;
case '\b':
if (tty->column > 0)
tty->column--;
break;
default:
if (O_OLCUC(tty))
c = toupper(c);
if (!iscntrl(c) && !is_continuation(c, tty))
tty->column++;
break;
} }
tty_put_char(tty, c); tty_put_char(tty, c);
unlock_kernel(); return 1;
return 0; }
/**
* process_output - output post processor
* @c: character (or partial unicode symbol)
* @tty: terminal device
*
* Perform OPOST processing. Returns -1 when the output device is
* full and the character must be retried.
*
* Locking: output_lock to protect column state and space left
* (also, this is called from n_tty_write under the
* tty layer write lock)
*/
static int process_output(unsigned char c, struct tty_struct *tty)
{
int space, retval;
mutex_lock(&tty->output_lock);
space = tty_write_room(tty);
retval = do_output_char(c, tty, space);
mutex_unlock(&tty->output_lock);
if (retval < 0)
return -1;
else
return 0;
} }
/** /**
* opost_block - block postprocess * process_output_block - block post processor
* @tty: terminal device * @tty: terminal device
* @inbuf: user buffer * @inbuf: user buffer
* @nr: number of bytes * @nr: number of bytes
...@@ -358,24 +402,29 @@ static int opost(unsigned char c, struct tty_struct *tty) ...@@ -358,24 +402,29 @@ static int opost(unsigned char c, struct tty_struct *tty)
* the simple cases normally found and helps to generate blocks of * the simple cases normally found and helps to generate blocks of
* symbols for the console driver and thus improve performance. * symbols for the console driver and thus improve performance.
* *
* Called from n_tty_write under the tty layer write lock. Relies * Locking: output_lock to protect column state and space left
* on lock_kernel for the tty->column state. * (also, this is called from n_tty_write under the
* tty layer write lock)
*/ */
static ssize_t opost_block(struct tty_struct *tty, static ssize_t process_output_block(struct tty_struct *tty,
const unsigned char *buf, unsigned int nr) const unsigned char *buf, unsigned int nr)
{ {
int space; int space;
int i; int i;
const unsigned char *cp; const unsigned char *cp;
mutex_lock(&tty->output_lock);
space = tty_write_room(tty); space = tty_write_room(tty);
if (!space) if (!space)
{
mutex_unlock(&tty->output_lock);
return 0; return 0;
}
if (nr > space) if (nr > space)
nr = space; nr = space;
lock_kernel();
for (i = 0, cp = buf; i < nr; i++, cp++) { for (i = 0, cp = buf; i < nr; i++, cp++) {
switch (*cp) { switch (*cp) {
case '\n': case '\n':
...@@ -407,46 +456,393 @@ static ssize_t opost_block(struct tty_struct *tty, ...@@ -407,46 +456,393 @@ static ssize_t opost_block(struct tty_struct *tty,
} }
} }
break_out: break_out:
if (tty->ops->flush_chars)
tty->ops->flush_chars(tty);
i = tty->ops->write(tty, buf, i); i = tty->ops->write(tty, buf, i);
unlock_kernel();
mutex_unlock(&tty->output_lock);
return i; return i;
} }
/**
* process_echoes - write pending echo characters
* @tty: terminal device
*
* Write previously buffered echo (and other ldisc-generated)
* characters to the tty.
*
* Characters generated by the ldisc (including echoes) need to
* be buffered because the driver's write buffer can fill during
* heavy program output. Echoing straight to the driver will
* often fail under these conditions, causing lost characters and
* resulting mismatches of ldisc state information.
*
* Since the ldisc state must represent the characters actually sent
* to the driver at the time of the write, operations like certain
* changes in column state are also saved in the buffer and executed
* here.
*
* A circular fifo buffer is used so that the most recent characters
* are prioritized. Also, when control characters are echoed with a
* prefixed "^", the pair is treated atomically and thus not separated.
*
* Locking: output_lock to protect column state and space left,
* echo_lock to protect the echo buffer
*/
static void process_echoes(struct tty_struct *tty)
{
int space, nr;
unsigned char c;
unsigned char *cp, *buf_end;
if (!tty->echo_cnt)
return;
mutex_lock(&tty->output_lock);
mutex_lock(&tty->echo_lock);
space = tty_write_room(tty);
buf_end = tty->echo_buf + N_TTY_BUF_SIZE;
cp = tty->echo_buf + tty->echo_pos;
nr = tty->echo_cnt;
while (nr > 0) {
c = *cp;
if (c == ECHO_OP_START) {
unsigned char op;
unsigned char *opp;
int no_space_left = 0;
/*
* If the buffer byte is the start of a multi-byte
* operation, get the next byte, which is either the
* op code or a control character value.
*/
opp = cp + 1;
if (opp == buf_end)
opp -= N_TTY_BUF_SIZE;
op = *opp;
switch (op) {
unsigned int num_chars, num_bs;
case ECHO_OP_ERASE_TAB:
if (++opp == buf_end)
opp -= N_TTY_BUF_SIZE;
num_chars = *opp;
/*
* Determine how many columns to go back
* in order to erase the tab.
* This depends on the number of columns
* used by other characters within the tab
* area. If this (modulo 8) count is from
* the start of input rather than from a
* previous tab, we offset by canon column.
* Otherwise, tab spacing is normal.
*/
if (!(num_chars & 0x80))
num_chars += tty->canon_column;
num_bs = 8 - (num_chars & 7);
if (num_bs > space) {
no_space_left = 1;
break;
}
space -= num_bs;
while (num_bs--) {
tty_put_char(tty, '\b');
if (tty->column > 0)
tty->column--;
}
cp += 3;
nr -= 3;
break;
case ECHO_OP_SET_CANON_COL:
tty->canon_column = tty->column;
cp += 2;
nr -= 2;
break;
case ECHO_OP_MOVE_BACK_COL:
if (tty->column > 0)
tty->column--;
cp += 2;
nr -= 2;
break;
case ECHO_OP_START:
/* This is an escaped echo op start code */
if (!space) {
no_space_left = 1;
break;
}
tty_put_char(tty, ECHO_OP_START);
tty->column++;
space--;
cp += 2;
nr -= 2;
break;
default:
if (iscntrl(op)) {
if (L_ECHOCTL(tty)) {
/*
* Ensure there is enough space
* for the whole ctrl pair.
*/
if (space < 2) {
no_space_left = 1;
break;
}
tty_put_char(tty, '^');
tty_put_char(tty, op ^ 0100);
tty->column += 2;
space -= 2;
} else {
if (!space) {
no_space_left = 1;
break;
}
tty_put_char(tty, op);
space--;
}
}
/*
* If above falls through, this was an
* undefined op.
*/
cp += 2;
nr -= 2;
}
if (no_space_left)
break;
} else {
int retval;
if ((retval = do_output_char(c, tty, space)) < 0)
break;
space -= retval;
cp += 1;
nr -= 1;
}
/* When end of circular buffer reached, wrap around */
if (cp >= buf_end)
cp -= N_TTY_BUF_SIZE;
}
if (nr == 0) {
tty->echo_pos = 0;
tty->echo_cnt = 0;
tty->echo_overrun = 0;
} else {
int num_processed = tty->echo_cnt - nr;
tty->echo_pos += num_processed;
tty->echo_pos &= N_TTY_BUF_SIZE - 1;
tty->echo_cnt = nr;
if (num_processed > 0)
tty->echo_overrun = 0;
}
mutex_unlock(&tty->echo_lock);
mutex_unlock(&tty->output_lock);
if (tty->ops->flush_chars)
tty->ops->flush_chars(tty);
}
/**
* add_echo_byte - add a byte to the echo buffer
* @c: unicode byte to echo
* @tty: terminal device
*
* Add a character or operation byte to the echo buffer.
*
* Should be called under the echo lock to protect the echo buffer.
*/
static void add_echo_byte(unsigned char c, struct tty_struct *tty)
{
int new_byte_pos;
if (tty->echo_cnt == N_TTY_BUF_SIZE) {
/* Circular buffer is already at capacity */
new_byte_pos = tty->echo_pos;
/*
* Since the buffer start position needs to be advanced,
* be sure to step by a whole operation byte group.
*/
if (tty->echo_buf[tty->echo_pos] == ECHO_OP_START)
{
if (tty->echo_buf[(tty->echo_pos + 1) &
(N_TTY_BUF_SIZE - 1)] ==
ECHO_OP_ERASE_TAB) {
tty->echo_pos += 3;
tty->echo_cnt -= 2;
} else {
tty->echo_pos += 2;
tty->echo_cnt -= 1;
}
} else {
tty->echo_pos++;
}
tty->echo_pos &= N_TTY_BUF_SIZE - 1;
tty->echo_overrun = 1;
} else {
new_byte_pos = tty->echo_pos + tty->echo_cnt;
new_byte_pos &= N_TTY_BUF_SIZE - 1;
tty->echo_cnt++;
}
tty->echo_buf[new_byte_pos] = c;
}
/**
* echo_move_back_col - add operation to move back a column
* @tty: terminal device
*
* Add an operation to the echo buffer to move back one column.
*
* Locking: echo_lock to protect the echo buffer
*/
static void echo_move_back_col(struct tty_struct *tty)
{
mutex_lock(&tty->echo_lock);
add_echo_byte(ECHO_OP_START, tty);
add_echo_byte(ECHO_OP_MOVE_BACK_COL, tty);
mutex_unlock(&tty->echo_lock);
}
/**
* echo_set_canon_col - add operation to set the canon column
* @tty: terminal device
*
* Add an operation to the echo buffer to set the canon column
* to the current column.
*
* Locking: echo_lock to protect the echo buffer
*/
static void echo_set_canon_col(struct tty_struct *tty)
{
mutex_lock(&tty->echo_lock);
add_echo_byte(ECHO_OP_START, tty);
add_echo_byte(ECHO_OP_SET_CANON_COL, tty);
mutex_unlock(&tty->echo_lock);
}
/**
* echo_erase_tab - add operation to erase a tab
* @num_chars: number of character columns already used
* @after_tab: true if num_chars starts after a previous tab
* @tty: terminal device
*
* Add an operation to the echo buffer to erase a tab.
*
* Called by the eraser function, which knows how many character
* columns have been used since either a previous tab or the start
* of input. This information will be used later, along with
* canon column (if applicable), to go back the correct number
* of columns.
*
* Locking: echo_lock to protect the echo buffer
*/
static void echo_erase_tab(unsigned int num_chars, int after_tab,
struct tty_struct *tty)
{
mutex_lock(&tty->echo_lock);
add_echo_byte(ECHO_OP_START, tty);
add_echo_byte(ECHO_OP_ERASE_TAB, tty);
/* We only need to know this modulo 8 (tab spacing) */
num_chars &= 7;
/* Set the high bit as a flag if num_chars is after a previous tab */
if (after_tab)
num_chars |= 0x80;
add_echo_byte(num_chars, tty);
mutex_unlock(&tty->echo_lock);
}
/**
* echo_char_raw - echo a character raw
* @c: unicode byte to echo
* @tty: terminal device
*
* Echo user input back onto the screen. This must be called only when
* L_ECHO(tty) is true. Called from the driver receive_buf path.
*
* This variant does not treat control characters specially.
*
* Locking: echo_lock to protect the echo buffer
*/
static void echo_char_raw(unsigned char c, struct tty_struct *tty)
{
mutex_lock(&tty->echo_lock);
if (c == ECHO_OP_START) {
add_echo_byte(ECHO_OP_START, tty);
add_echo_byte(ECHO_OP_START, tty);
} else {
add_echo_byte(c, tty);
}
mutex_unlock(&tty->echo_lock);
}
/** /**
* echo_char - echo characters * echo_char - echo a character
* @c: unicode byte to echo * @c: unicode byte to echo
* @tty: terminal device * @tty: terminal device
* *
* Echo user input back onto the screen. This must be called only when * Echo user input back onto the screen. This must be called only when
* L_ECHO(tty) is true. Called from the driver receive_buf path. * L_ECHO(tty) is true. Called from the driver receive_buf path.
* *
* Relies on BKL for tty column locking * This variant tags control characters to be possibly echoed as
* as "^X" (where X is the letter representing the control char).
*
* Locking: echo_lock to protect the echo buffer
*/ */
static void echo_char(unsigned char c, struct tty_struct *tty) static void echo_char(unsigned char c, struct tty_struct *tty)
{ {
if (L_ECHOCTL(tty) && iscntrl(c) && c != '\t') { mutex_lock(&tty->echo_lock);
tty_put_char(tty, '^');
tty_put_char(tty, c ^ 0100); if (c == ECHO_OP_START) {
tty->column += 2; add_echo_byte(ECHO_OP_START, tty);
} else add_echo_byte(ECHO_OP_START, tty);
opost(c, tty); } else {
if (iscntrl(c) && c != '\t')
add_echo_byte(ECHO_OP_START, tty);
add_echo_byte(c, tty);
}
mutex_unlock(&tty->echo_lock);
} }
/** /**
* finsh_erasing - complete erase * finish_erasing - complete erase
* @tty: tty doing the erase * @tty: tty doing the erase
*
* Relies on BKL for tty column locking
*/ */
static inline void finish_erasing(struct tty_struct *tty) static inline void finish_erasing(struct tty_struct *tty)
{ {
if (tty->erasing) { if (tty->erasing) {
tty_put_char(tty, '/'); echo_char_raw('/', tty);
tty->column++;
tty->erasing = 0; tty->erasing = 0;
} }
} }
...@@ -460,7 +856,7 @@ static inline void finish_erasing(struct tty_struct *tty) ...@@ -460,7 +856,7 @@ static inline void finish_erasing(struct tty_struct *tty)
* present in the stream from the driver layer. Handles the complexities * present in the stream from the driver layer. Handles the complexities
* of UTF-8 multibyte symbols. * of UTF-8 multibyte symbols.
* *
* Locking: read_lock for tty buffers, BKL for column/erasing state * Locking: read_lock for tty buffers
*/ */
static void eraser(unsigned char c, struct tty_struct *tty) static void eraser(unsigned char c, struct tty_struct *tty)
...@@ -471,7 +867,7 @@ static void eraser(unsigned char c, struct tty_struct *tty) ...@@ -471,7 +867,7 @@ static void eraser(unsigned char c, struct tty_struct *tty)
/* FIXME: locking needed ? */ /* FIXME: locking needed ? */
if (tty->read_head == tty->canon_head) { if (tty->read_head == tty->canon_head) {
/* opost('\a', tty); */ /* what do you think? */ /* echo_char_raw('\a', tty); */ /* what do you think? */
return; return;
} }
if (c == ERASE_CHAR(tty)) if (c == ERASE_CHAR(tty))
...@@ -497,7 +893,7 @@ static void eraser(unsigned char c, struct tty_struct *tty) ...@@ -497,7 +893,7 @@ static void eraser(unsigned char c, struct tty_struct *tty)
echo_char(KILL_CHAR(tty), tty); echo_char(KILL_CHAR(tty), tty);
/* Add a newline if ECHOK is on and ECHOKE is off. */ /* Add a newline if ECHOK is on and ECHOKE is off. */
if (L_ECHOK(tty)) if (L_ECHOK(tty))
opost('\n', tty); echo_char_raw('\n', tty);
return; return;
} }
kill_type = KILL; kill_type = KILL;
...@@ -533,67 +929,62 @@ static void eraser(unsigned char c, struct tty_struct *tty) ...@@ -533,67 +929,62 @@ static void eraser(unsigned char c, struct tty_struct *tty)
if (L_ECHO(tty)) { if (L_ECHO(tty)) {
if (L_ECHOPRT(tty)) { if (L_ECHOPRT(tty)) {
if (!tty->erasing) { if (!tty->erasing) {
tty_put_char(tty, '\\'); echo_char_raw('\\', tty);
tty->column++;
tty->erasing = 1; tty->erasing = 1;
} }
/* if cnt > 1, output a multi-byte character */ /* if cnt > 1, output a multi-byte character */
echo_char(c, tty); echo_char(c, tty);
while (--cnt > 0) { while (--cnt > 0) {
head = (head+1) & (N_TTY_BUF_SIZE-1); head = (head+1) & (N_TTY_BUF_SIZE-1);
tty_put_char(tty, tty->read_buf[head]); echo_char_raw(tty->read_buf[head], tty);
echo_move_back_col(tty);
} }
} else if (kill_type == ERASE && !L_ECHOE(tty)) { } else if (kill_type == ERASE && !L_ECHOE(tty)) {
echo_char(ERASE_CHAR(tty), tty); echo_char(ERASE_CHAR(tty), tty);
} else if (c == '\t') { } else if (c == '\t') {
unsigned int col = tty->canon_column; unsigned int num_chars = 0;
unsigned long tail = tty->canon_head; int after_tab = 0;
unsigned long tail = tty->read_head;
/* Find the column of the last char. */
while (tail != tty->read_head) { /*
* Count the columns used for characters
* since the start of input or after a
* previous tab.
* This info is used to go back the correct
* number of columns.
*/
while (tail != tty->canon_head) {
tail = (tail-1) & (N_TTY_BUF_SIZE-1);
c = tty->read_buf[tail]; c = tty->read_buf[tail];
if (c == '\t') if (c == '\t') {
col = (col | 7) + 1; after_tab = 1;
break;
}
else if (iscntrl(c)) { else if (iscntrl(c)) {
if (L_ECHOCTL(tty)) if (L_ECHOCTL(tty))
col += 2; num_chars += 2;
} else if (!is_continuation(c, tty)) } else if (!is_continuation(c, tty)) {
col++; num_chars++;
tail = (tail+1) & (N_TTY_BUF_SIZE-1); }
}
/* should never happen */
if (tty->column > 0x80000000)
tty->column = 0;
/* Now backup to that column. */
while (tty->column > col) {
/* Can't use opost here. */
tty_put_char(tty, '\b');
if (tty->column > 0)
tty->column--;
} }
echo_erase_tab(num_chars, after_tab, tty);
} else { } else {
if (iscntrl(c) && L_ECHOCTL(tty)) { if (iscntrl(c) && L_ECHOCTL(tty)) {
tty_put_char(tty, '\b'); echo_char_raw('\b', tty);
tty_put_char(tty, ' '); echo_char_raw(' ', tty);
tty_put_char(tty, '\b'); echo_char_raw('\b', tty);
if (tty->column > 0)
tty->column--;
} }
if (!iscntrl(c) || L_ECHOCTL(tty)) { if (!iscntrl(c) || L_ECHOCTL(tty)) {
tty_put_char(tty, '\b'); echo_char_raw('\b', tty);
tty_put_char(tty, ' '); echo_char_raw(' ', tty);
tty_put_char(tty, '\b'); echo_char_raw('\b', tty);
if (tty->column > 0)
tty->column--;
} }
} }
} }
if (kill_type == ERASE) if (kill_type == ERASE)
break; break;
} }
if (tty->read_head == tty->canon_head) if (tty->read_head == tty->canon_head && L_ECHO(tty))
finish_erasing(tty); finish_erasing(tty);
} }
...@@ -724,14 +1115,18 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c) ...@@ -724,14 +1115,18 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c)
c=tolower(c); c=tolower(c);
if (tty->stopped && !tty->flow_stopped && I_IXON(tty) && if (tty->stopped && !tty->flow_stopped && I_IXON(tty) &&
((I_IXANY(tty) && c != START_CHAR(tty) && c != STOP_CHAR(tty)) || I_IXANY(tty) && c != START_CHAR(tty) && c != STOP_CHAR(tty) &&
c == INTR_CHAR(tty) || c == QUIT_CHAR(tty) || c == SUSP_CHAR(tty))) c != INTR_CHAR(tty) && c != QUIT_CHAR(tty) && c != SUSP_CHAR(tty)) {
start_tty(tty); start_tty(tty);
process_echoes(tty);
}
if (tty->closing) { if (tty->closing) {
if (I_IXON(tty)) { if (I_IXON(tty)) {
if (c == START_CHAR(tty)) if (c == START_CHAR(tty)) {
start_tty(tty); start_tty(tty);
process_echoes(tty);
}
else if (c == STOP_CHAR(tty)) else if (c == STOP_CHAR(tty))
stop_tty(tty); stop_tty(tty);
} }
...@@ -745,17 +1140,20 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c) ...@@ -745,17 +1140,20 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c)
* up. * up.
*/ */
if (!test_bit(c, tty->process_char_map) || tty->lnext) { if (!test_bit(c, tty->process_char_map) || tty->lnext) {
finish_erasing(tty);
tty->lnext = 0; tty->lnext = 0;
if (L_ECHO(tty)) { if (L_ECHO(tty)) {
finish_erasing(tty);
if (tty->read_cnt >= N_TTY_BUF_SIZE-1) { if (tty->read_cnt >= N_TTY_BUF_SIZE-1) {
tty_put_char(tty, '\a'); /* beep if no space */ /* beep if no space */
echo_char_raw('\a', tty);
process_echoes(tty);
return; return;
} }
/* Record the column of first canon char. */ /* Record the column of first canon char. */
if (tty->canon_head == tty->read_head) if (tty->canon_head == tty->read_head)
tty->canon_column = tty->column; echo_set_canon_col(tty);
echo_char(c, tty); echo_char(c, tty);
process_echoes(tty);
} }
if (I_PARMRK(tty) && c == (unsigned char) '\377') if (I_PARMRK(tty) && c == (unsigned char) '\377')
put_tty_queue(c, tty); put_tty_queue(c, tty);
...@@ -766,6 +1164,7 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c) ...@@ -766,6 +1164,7 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c)
if (I_IXON(tty)) { if (I_IXON(tty)) {
if (c == START_CHAR(tty)) { if (c == START_CHAR(tty)) {
start_tty(tty); start_tty(tty);
process_echoes(tty);
return; return;
} }
if (c == STOP_CHAR(tty)) { if (c == STOP_CHAR(tty)) {
...@@ -786,7 +1185,6 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c) ...@@ -786,7 +1185,6 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c)
if (c == SUSP_CHAR(tty)) { if (c == SUSP_CHAR(tty)) {
send_signal: send_signal:
/* /*
* Echo character, and then send the signal.
* Note that we do not use isig() here because we want * Note that we do not use isig() here because we want
* the order to be: * the order to be:
* 1) flush, 2) echo, 3) signal * 1) flush, 2) echo, 3) signal
...@@ -795,8 +1193,12 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c) ...@@ -795,8 +1193,12 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c)
n_tty_flush_buffer(tty); n_tty_flush_buffer(tty);
tty_driver_flush_buffer(tty); tty_driver_flush_buffer(tty);
} }
if (L_ECHO(tty)) if (I_IXON(tty))
start_tty(tty);
if (L_ECHO(tty)) {
echo_char(c, tty); echo_char(c, tty);
process_echoes(tty);
}
if (tty->pgrp) if (tty->pgrp)
kill_pgrp(tty->pgrp, signal, 1); kill_pgrp(tty->pgrp, signal, 1);
return; return;
...@@ -815,6 +1217,7 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c) ...@@ -815,6 +1217,7 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c)
if (c == ERASE_CHAR(tty) || c == KILL_CHAR(tty) || if (c == ERASE_CHAR(tty) || c == KILL_CHAR(tty) ||
(c == WERASE_CHAR(tty) && L_IEXTEN(tty))) { (c == WERASE_CHAR(tty) && L_IEXTEN(tty))) {
eraser(c, tty); eraser(c, tty);
process_echoes(tty);
return; return;
} }
if (c == LNEXT_CHAR(tty) && L_IEXTEN(tty)) { if (c == LNEXT_CHAR(tty) && L_IEXTEN(tty)) {
...@@ -822,8 +1225,9 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c) ...@@ -822,8 +1225,9 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c)
if (L_ECHO(tty)) { if (L_ECHO(tty)) {
finish_erasing(tty); finish_erasing(tty);
if (L_ECHOCTL(tty)) { if (L_ECHOCTL(tty)) {
tty_put_char(tty, '^'); echo_char_raw('^', tty);
tty_put_char(tty, '\b'); echo_char_raw('\b', tty);
process_echoes(tty);
} }
} }
return; return;
...@@ -834,18 +1238,20 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c) ...@@ -834,18 +1238,20 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c)
finish_erasing(tty); finish_erasing(tty);
echo_char(c, tty); echo_char(c, tty);
opost('\n', tty); echo_char_raw('\n', tty);
while (tail != tty->read_head) { while (tail != tty->read_head) {
echo_char(tty->read_buf[tail], tty); echo_char(tty->read_buf[tail], tty);
tail = (tail+1) & (N_TTY_BUF_SIZE-1); tail = (tail+1) & (N_TTY_BUF_SIZE-1);
} }
process_echoes(tty);
return; return;
} }
if (c == '\n') { if (c == '\n') {
if (L_ECHO(tty) || L_ECHONL(tty)) { if (L_ECHO(tty) || L_ECHONL(tty)) {
if (tty->read_cnt >= N_TTY_BUF_SIZE-1) if (tty->read_cnt >= N_TTY_BUF_SIZE-1)
tty_put_char(tty, '\a'); echo_char_raw('\a', tty);
opost('\n', tty); echo_char_raw('\n', tty);
process_echoes(tty);
} }
goto handle_newline; goto handle_newline;
} }
...@@ -862,11 +1268,12 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c) ...@@ -862,11 +1268,12 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c)
*/ */
if (L_ECHO(tty)) { if (L_ECHO(tty)) {
if (tty->read_cnt >= N_TTY_BUF_SIZE-1) if (tty->read_cnt >= N_TTY_BUF_SIZE-1)
tty_put_char(tty, '\a'); echo_char_raw('\a', tty);
/* Record the column of first canon char. */ /* Record the column of first canon char. */
if (tty->canon_head == tty->read_head) if (tty->canon_head == tty->read_head)
tty->canon_column = tty->column; echo_set_canon_col(tty);
echo_char(c, tty); echo_char(c, tty);
process_echoes(tty);
} }
/* /*
* XXX does PARMRK doubling happen for * XXX does PARMRK doubling happen for
...@@ -889,20 +1296,23 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c) ...@@ -889,20 +1296,23 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c)
} }
} }
finish_erasing(tty);
if (L_ECHO(tty)) { if (L_ECHO(tty)) {
finish_erasing(tty);
if (tty->read_cnt >= N_TTY_BUF_SIZE-1) { if (tty->read_cnt >= N_TTY_BUF_SIZE-1) {
tty_put_char(tty, '\a'); /* beep if no space */ /* beep if no space */
echo_char_raw('\a', tty);
process_echoes(tty);
return; return;
} }
if (c == '\n') if (c == '\n')
opost('\n', tty); echo_char_raw('\n', tty);
else { else {
/* Record the column of first canon char. */ /* Record the column of first canon char. */
if (tty->canon_head == tty->read_head) if (tty->canon_head == tty->read_head)
tty->canon_column = tty->column; echo_set_canon_col(tty);
echo_char(c, tty); echo_char(c, tty);
} }
process_echoes(tty);
} }
if (I_PARMRK(tty) && c == (unsigned char) '\377') if (I_PARMRK(tty) && c == (unsigned char) '\377')
...@@ -923,6 +1333,9 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c) ...@@ -923,6 +1333,9 @@ static inline void n_tty_receive_char(struct tty_struct *tty, unsigned char c)
static void n_tty_write_wakeup(struct tty_struct *tty) static void n_tty_write_wakeup(struct tty_struct *tty)
{ {
/* Write out any echoed characters that are still pending */
process_echoes(tty);
if (tty->fasync) { if (tty->fasync) {
set_bit(TTY_DO_WRITE_WAKEUP, &tty->flags); set_bit(TTY_DO_WRITE_WAKEUP, &tty->flags);
kill_fasync(&tty->fasync, SIGIO, POLL_OUT); kill_fasync(&tty->fasync, SIGIO, POLL_OUT);
...@@ -1134,6 +1547,10 @@ static void n_tty_close(struct tty_struct *tty) ...@@ -1134,6 +1547,10 @@ static void n_tty_close(struct tty_struct *tty)
free_buf(tty->read_buf); free_buf(tty->read_buf);
tty->read_buf = NULL; tty->read_buf = NULL;
} }
if (tty->echo_buf) {
free_buf(tty->echo_buf);
tty->echo_buf = NULL;
}
} }
/** /**
...@@ -1151,13 +1568,19 @@ static int n_tty_open(struct tty_struct *tty) ...@@ -1151,13 +1568,19 @@ static int n_tty_open(struct tty_struct *tty)
if (!tty) if (!tty)
return -EINVAL; return -EINVAL;
/* This one is ugly. Currently a malloc failure here can panic */ /* These are ugly. Currently a malloc failure here can panic */
if (!tty->read_buf) { if (!tty->read_buf) {
tty->read_buf = alloc_buf(); tty->read_buf = alloc_buf();
if (!tty->read_buf) if (!tty->read_buf)
return -ENOMEM; return -ENOMEM;
} }
if (!tty->echo_buf) {
tty->echo_buf = alloc_buf();
if (!tty->echo_buf)
return -ENOMEM;
}
memset(tty->read_buf, 0, N_TTY_BUF_SIZE); memset(tty->read_buf, 0, N_TTY_BUF_SIZE);
memset(tty->echo_buf, 0, N_TTY_BUF_SIZE);
reset_buffer_flags(tty); reset_buffer_flags(tty);
tty->column = 0; tty->column = 0;
n_tty_set_termios(tty, NULL); n_tty_set_termios(tty, NULL);
...@@ -1487,16 +1910,23 @@ static ssize_t n_tty_read(struct tty_struct *tty, struct file *file, ...@@ -1487,16 +1910,23 @@ static ssize_t n_tty_read(struct tty_struct *tty, struct file *file,
* @buf: userspace buffer pointer * @buf: userspace buffer pointer
* @nr: size of I/O * @nr: size of I/O
* *
* Write function of the terminal device. This is serialized with * Write function of the terminal device. This is serialized with
* respect to other write callers but not to termios changes, reads * respect to other write callers but not to termios changes, reads
* and other such events. We must be careful with N_TTY as the receive * and other such events. Since the receive code will echo characters,
* code will echo characters, thus calling driver write methods. * thus calling driver write methods, the output_lock is used in
* the output processing functions called here as well as in the
* echo processing function to protect the column state and space
* left in the buffer.
* *
* This code must be sure never to sleep through a hangup. * This code must be sure never to sleep through a hangup.
*
* Locking: output_lock to protect column state and space left
* (note that the process_output*() functions take this
* lock themselves)
*/ */
static ssize_t n_tty_write(struct tty_struct *tty, struct file *file, static ssize_t n_tty_write(struct tty_struct *tty, struct file *file,
const unsigned char *buf, size_t nr) const unsigned char *buf, size_t nr)
{ {
const unsigned char *b = buf; const unsigned char *b = buf;
DECLARE_WAITQUEUE(wait, current); DECLARE_WAITQUEUE(wait, current);
...@@ -1510,6 +1940,9 @@ static ssize_t n_tty_write(struct tty_struct *tty, struct file *file, ...@@ -1510,6 +1940,9 @@ static ssize_t n_tty_write(struct tty_struct *tty, struct file *file,
return retval; return retval;
} }
/* Write out any echoed characters that are still pending */
process_echoes(tty);
add_wait_queue(&tty->write_wait, &wait); add_wait_queue(&tty->write_wait, &wait);
while (1) { while (1) {
set_current_state(TASK_INTERRUPTIBLE); set_current_state(TASK_INTERRUPTIBLE);
...@@ -1523,7 +1956,7 @@ static ssize_t n_tty_write(struct tty_struct *tty, struct file *file, ...@@ -1523,7 +1956,7 @@ static ssize_t n_tty_write(struct tty_struct *tty, struct file *file,
} }
if (O_OPOST(tty) && !(test_bit(TTY_HW_COOK_OUT, &tty->flags))) { if (O_OPOST(tty) && !(test_bit(TTY_HW_COOK_OUT, &tty->flags))) {
while (nr > 0) { while (nr > 0) {
ssize_t num = opost_block(tty, b, nr); ssize_t num = process_output_block(tty, b, nr);
if (num < 0) { if (num < 0) {
if (num == -EAGAIN) if (num == -EAGAIN)
break; break;
...@@ -1535,7 +1968,7 @@ static ssize_t n_tty_write(struct tty_struct *tty, struct file *file, ...@@ -1535,7 +1968,7 @@ static ssize_t n_tty_write(struct tty_struct *tty, struct file *file,
if (nr == 0) if (nr == 0)
break; break;
c = *b; c = *b;
if (opost(c, tty) < 0) if (process_output(c, tty) < 0)
break; break;
b++; nr--; b++; nr--;
} }
...@@ -1663,4 +2096,3 @@ struct tty_ldisc_ops tty_ldisc_N_TTY = { ...@@ -1663,4 +2096,3 @@ struct tty_ldisc_ops tty_ldisc_N_TTY = {
.receive_buf = n_tty_receive_buf, .receive_buf = n_tty_receive_buf,
.write_wakeup = n_tty_write_wakeup .write_wakeup = n_tty_write_wakeup
}; };
...@@ -1111,9 +1111,7 @@ void tty_write_message(struct tty_struct *tty, char *msg) ...@@ -1111,9 +1111,7 @@ void tty_write_message(struct tty_struct *tty, char *msg)
* Locks the line discipline as required * Locks the line discipline as required
* Writes to the tty driver are serialized by the atomic_write_lock * Writes to the tty driver are serialized by the atomic_write_lock
* and are then processed in chunks to the device. The line discipline * and are then processed in chunks to the device. The line discipline
* write method will not be involked in parallel for each device * write method will not be invoked in parallel for each device.
* The line discipline write method is called under the big
* kernel lock for historical reasons. New code should not rely on this.
*/ */
static ssize_t tty_write(struct file *file, const char __user *buf, static ssize_t tty_write(struct file *file, const char __user *buf,
...@@ -2785,6 +2783,8 @@ void initialize_tty_struct(struct tty_struct *tty, ...@@ -2785,6 +2783,8 @@ void initialize_tty_struct(struct tty_struct *tty,
INIT_WORK(&tty->hangup_work, do_tty_hangup); INIT_WORK(&tty->hangup_work, do_tty_hangup);
mutex_init(&tty->atomic_read_lock); mutex_init(&tty->atomic_read_lock);
mutex_init(&tty->atomic_write_lock); mutex_init(&tty->atomic_write_lock);
mutex_init(&tty->output_lock);
mutex_init(&tty->echo_lock);
spin_lock_init(&tty->read_lock); spin_lock_init(&tty->read_lock);
spin_lock_init(&tty->ctrl_lock); spin_lock_init(&tty->ctrl_lock);
INIT_LIST_HEAD(&tty->tty_files); INIT_LIST_HEAD(&tty->tty_files);
......
...@@ -2679,7 +2679,7 @@ static int con_write_room(struct tty_struct *tty) ...@@ -2679,7 +2679,7 @@ static int con_write_room(struct tty_struct *tty)
{ {
if (tty->stopped) if (tty->stopped)
return 0; return 0;
return 4096; /* No limit, really; we're not buffering */ return 32768; /* No limit, really; we're not buffering */
} }
static int con_chars_in_buffer(struct tty_struct *tty) static int con_chars_in_buffer(struct tty_struct *tty)
......
...@@ -253,6 +253,7 @@ struct tty_struct { ...@@ -253,6 +253,7 @@ struct tty_struct {
unsigned int column; unsigned int column;
unsigned char lnext:1, erasing:1, raw:1, real_raw:1, icanon:1; unsigned char lnext:1, erasing:1, raw:1, real_raw:1, icanon:1;
unsigned char closing:1; unsigned char closing:1;
unsigned char echo_overrun:1;
unsigned short minimum_to_wake; unsigned short minimum_to_wake;
unsigned long overrun_time; unsigned long overrun_time;
int num_overrun; int num_overrun;
...@@ -262,11 +263,16 @@ struct tty_struct { ...@@ -262,11 +263,16 @@ struct tty_struct {
int read_tail; int read_tail;
int read_cnt; int read_cnt;
unsigned long read_flags[N_TTY_BUF_SIZE/(8*sizeof(unsigned long))]; unsigned long read_flags[N_TTY_BUF_SIZE/(8*sizeof(unsigned long))];
unsigned char *echo_buf;
unsigned int echo_pos;
unsigned int echo_cnt;
int canon_data; int canon_data;
unsigned long canon_head; unsigned long canon_head;
unsigned int canon_column; unsigned int canon_column;
struct mutex atomic_read_lock; struct mutex atomic_read_lock;
struct mutex atomic_write_lock; struct mutex atomic_write_lock;
struct mutex output_lock;
struct mutex echo_lock;
unsigned char *write_buf; unsigned char *write_buf;
int write_cnt; int write_cnt;
spinlock_t read_lock; spinlock_t read_lock;
......
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