generic/tty: Combine settings structures.

It doesn't make much sense to have separate window sizes, termios, etc.
settings for the two halves of a TTY. Moreover, having separate settings
which can get out of sync breaks applications, e.g. double-printing
in shells.

The original TTY driver has unified settings. It seems likely that the
settings were separated as part of the locking simplification, however
the lock separation was reverted a while back, anyway.
This commit is contained in:
Augustin Cavalier 2023-05-09 15:24:20 -04:00
parent 94457adea2
commit bb57ea897d
3 changed files with 65 additions and 58 deletions

View File

@ -64,7 +64,7 @@ dump_tty_struct(struct tty& tty)
kprintf(" writer queue:\n"); kprintf(" writer queue:\n");
tty.writer_queue.Dump(" "); tty.writer_queue.Dump(" ");
dump_tty_settings(tty.settings); dump_tty_settings(*tty.settings);
kprintf(" cookies: "); kprintf(" cookies: ");
TTYCookieList::Iterator it = tty.cookies.GetIterator(); TTYCookieList::Iterator it = tty.cookies.GetIterator();

View File

@ -506,7 +506,7 @@ WriterLocker::WriterLocker(tty_cookie* sourceCookie)
// get the echo mode // get the echo mode
fEcho = (fSource->is_master fEcho = (fSource->is_master
&& fSource->settings.termios.c_lflag & ECHO) != 0; && fSource->settings->termios.c_lflag & ECHO) != 0;
// enqueue ourselves in the respective request queues // enqueue ourselves in the respective request queues
RecursiveLocker locker(gTTYRequestLock); RecursiveLocker locker(gTTYRequestLock);
@ -687,10 +687,10 @@ ReaderLocker::_CheckAvailableBytes() const
{ {
// Reading from the slave with canonical input processing enabled means // Reading from the slave with canonical input processing enabled means
// that we read at max until hitting a line end or EOF. // that we read at max until hitting a line end or EOF.
if (!fTTY->is_master && (fTTY->settings.termios.c_lflag & ICANON) != 0) { if (!fTTY->is_master && (fTTY->settings->termios.c_lflag & ICANON) != 0) {
return line_buffer_readable_line(fTTY->input_buffer, return line_buffer_readable_line(fTTY->input_buffer,
fTTY->settings.termios.c_cc[VEOL], fTTY->settings->termios.c_cc[VEOL],
fTTY->settings.termios.c_cc[VEOF]); fTTY->settings->termios.c_cc[VEOF]);
} }
return line_buffer_readable(fTTY->input_buffer); return line_buffer_readable(fTTY->input_buffer);
@ -751,36 +751,36 @@ reset_tty_settings(tty_settings& settings)
static void static void
tty_input_putc_locked(struct tty* tty, int c) tty_input_putc_locked(struct tty* tty, int c)
{ {
// process signals if needed const termios& termios = tty->settings->termios;
if ((tty->settings.termios.c_lflag & ISIG) != 0) { // process signals if needed
if ((termios.c_lflag & ISIG) != 0) {
// enable signals, process INTR, QUIT, and SUSP // enable signals, process INTR, QUIT, and SUSP
int signal = -1; int signal = -1;
if (c == tty->settings.termios.c_cc[VINTR]) if (c == termios.c_cc[VINTR])
signal = SIGINT; signal = SIGINT;
else if (c == tty->settings.termios.c_cc[VQUIT]) else if (c == termios.c_cc[VQUIT])
signal = SIGQUIT; signal = SIGQUIT;
else if (c == tty->settings.termios.c_cc[VSUSP]) else if (c == termios.c_cc[VSUSP])
signal = SIGTSTP; signal = SIGTSTP;
// do we need to deliver a signal? // do we need to deliver a signal?
if (signal != -1) { if (signal != -1) {
// we may have to flush the input buffer // we may have to flush the input buffer
if ((tty->settings.termios.c_lflag & NOFLSH) == 0) if ((termios.c_lflag & NOFLSH) == 0)
clear_line_buffer(tty->input_buffer); clear_line_buffer(tty->input_buffer);
if (tty->settings.pgrp_id != 0) if (tty->settings->pgrp_id != 0)
send_signal(-tty->settings.pgrp_id, signal); send_signal(-tty->settings->pgrp_id, signal);
return; return;
} }
} }
// process special canonical input characters // process special canonical input characters
if ((termios.c_lflag & ICANON) != 0) {
if ((tty->settings.termios.c_lflag & ICANON) != 0) {
// canonical mode, process ERASE and KILL // canonical mode, process ERASE and KILL
cc_t* controlChars = tty->settings.termios.c_cc; const cc_t* controlChars = termios.c_cc;
if (c == controlChars[VERASE]) { if (c == controlChars[VERASE]) {
// erase one character // erase one character
@ -824,10 +824,10 @@ tty_input_putc_locked(struct tty* tty, int c)
static int32 static int32
tty_readable(struct tty* tty) tty_readable(struct tty* tty)
{ {
if (!tty->is_master && (tty->settings.termios.c_lflag & ICANON) != 0) { if (!tty->is_master && (tty->settings->termios.c_lflag & ICANON) != 0) {
return line_buffer_readable_line(tty->input_buffer, return line_buffer_readable_line(tty->input_buffer,
tty->settings.termios.c_cc[VEOL], tty->settings->termios.c_cc[VEOL],
tty->settings.termios.c_cc[VEOF]); tty->settings->termios.c_cc[VEOF]);
} }
return line_buffer_readable(tty->input_buffer); return line_buffer_readable(tty->input_buffer);
@ -906,13 +906,14 @@ static bool
process_input_char(struct tty* tty, char c, char* buffer, process_input_char(struct tty* tty, char c, char* buffer,
size_t* _bytesNeeded) size_t* _bytesNeeded)
{ {
tcflag_t flags = tty->settings.termios.c_iflag; const termios& termios = tty->settings->termios;
tcflag_t flags = termios.c_iflag;
// signals // signals
if (tty->settings.termios.c_lflag & ISIG) { if (termios.c_lflag & ISIG) {
if (c == tty->settings.termios.c_cc[VINTR] if (c == termios.c_cc[VINTR]
|| c == tty->settings.termios.c_cc[VQUIT] || c == termios.c_cc[VQUIT]
|| c == tty->settings.termios.c_cc[VSUSP]) { || c == termios.c_cc[VSUSP]) {
*buffer = c; *buffer = c;
*_bytesNeeded = 0; *_bytesNeeded = 0;
return true; return true;
@ -920,9 +921,9 @@ process_input_char(struct tty* tty, char c, char* buffer,
} }
// canonical input characters // canonical input characters
if (tty->settings.termios.c_lflag & ICANON) { if (termios.c_lflag & ICANON) {
if (c == tty->settings.termios.c_cc[VERASE] if (c == termios.c_cc[VERASE]
|| c == tty->settings.termios.c_cc[VKILL]) { || c == termios.c_cc[VKILL]) {
*buffer = c; *buffer = c;
*_bytesNeeded = 0; *_bytesNeeded = 0;
return true; return true;
@ -961,11 +962,12 @@ static void
process_output_char(struct tty* tty, char c, char* buffer, process_output_char(struct tty* tty, char c, char* buffer,
size_t* _bytesWritten, bool echoed) size_t* _bytesWritten, bool echoed)
{ {
tcflag_t flags = tty->settings.termios.c_oflag; const termios& termios = tty->settings->termios;
tcflag_t flags = termios.c_oflag;
if (flags & OPOST) { if (flags & OPOST) {
if (echoed && c == tty->settings.termios.c_cc[VERASE]) { if (echoed && c == termios.c_cc[VERASE]) {
if (tty->settings.termios.c_lflag & ECHOE) { if (termios.c_lflag & ECHOE) {
// ERASE -> BS SPACE BS // ERASE -> BS SPACE BS
buffer[0] = CTRL('H'); buffer[0] = CTRL('H');
buffer[1] = ' '; buffer[1] = ' ';
@ -973,18 +975,18 @@ process_output_char(struct tty* tty, char c, char* buffer,
*_bytesWritten = 3; *_bytesWritten = 3;
return; return;
} }
} else if (echoed && c == tty->settings.termios.c_cc[VKILL]) { } else if (echoed && c == termios.c_cc[VKILL]) {
if (!(tty->settings.termios.c_lflag & ECHOK)) { if (!(termios.c_lflag & ECHOK)) {
// don't echo KILL // don't echo KILL
*_bytesWritten = 0; *_bytesWritten = 0;
return; return;
} }
} else if (echoed && c == tty->settings.termios.c_cc[VEOF]) { } else if (echoed && c == termios.c_cc[VEOF]) {
// don't echo EOF // don't echo EOF
*_bytesWritten = 0; *_bytesWritten = 0;
return; return;
} else if (c == '\n') { } else if (c == '\n') {
if (echoed && !(tty->settings.termios.c_lflag & ECHONL)) { if (echoed && !(termios.c_lflag & ECHONL)) {
// don't echo NL // don't echo NL
*_bytesWritten = 0; *_bytesWritten = 0;
return; return;
@ -1040,7 +1042,7 @@ tty_write_to_tty_master_unsafe(tty_cookie* sourceCookie, const char* data,
if (target->open_count <= 0) if (target->open_count <= 0)
return B_FILE_ERROR; return B_FILE_ERROR;
bool echo = (source->settings.termios.c_lflag & ECHO) != 0; bool echo = (source->settings->termios.c_lflag & ECHO) != 0;
TRACE(("tty_write_to_tty_master(source = %p, target = %p, " TRACE(("tty_write_to_tty_master(source = %p, target = %p, "
"length = %lu%s)\n", source, target, length, "length = %lu%s)\n", source, target, length,
@ -1282,14 +1284,20 @@ tty_create(tty_service_func func, struct tty* master)
if (master == NULL) { if (master == NULL) {
tty->is_master = true; tty->is_master = true;
tty->lock = new(std::nothrow) recursive_lock; tty->lock = new(std::nothrow) recursive_lock;
if (tty->lock == NULL) { tty->settings = new(std::nothrow) tty_settings;
if (tty->lock == NULL || tty->settings == NULL) {
delete tty->lock;
delete tty->settings;
delete tty; delete tty;
return NULL; return NULL;
} }
recursive_lock_init(tty->lock, "tty lock"); recursive_lock_init(tty->lock, "tty lock");
reset_tty_settings(*tty->settings);
} else { } else {
tty->is_master = false; tty->is_master = false;
tty->lock = master->lock; tty->lock = master->lock;
tty->settings = master->settings;
} }
tty->ref_count = 0; tty->ref_count = 0;
@ -1299,8 +1307,6 @@ tty_create(tty_service_func func, struct tty* master)
tty->pending_eof = 0; tty->pending_eof = 0;
tty->hardware_bits = 0; tty->hardware_bits = 0;
reset_tty_settings(tty->settings);
if (init_line_buffer(tty->input_buffer, TTY_BUFFER_SIZE) < B_OK) { if (init_line_buffer(tty->input_buffer, TTY_BUFFER_SIZE) < B_OK) {
if (tty->is_master) { if (tty->is_master) {
recursive_lock_destroy(tty->lock); recursive_lock_destroy(tty->lock);
@ -1325,6 +1331,7 @@ tty_destroy(struct tty* tty)
if (tty->is_master) { if (tty->is_master) {
recursive_lock_destroy(tty->lock); recursive_lock_destroy(tty->lock);
delete tty->lock; delete tty->lock;
delete tty->settings;
} }
delete tty; delete tty;
@ -1501,12 +1508,12 @@ tty_read(tty_cookie* cookie, void* _buffer, size_t* _length)
ReaderLocker locker(cookie); ReaderLocker locker(cookie);
// handle raw mode // handle raw mode
if ((!tty->is_master) && ((tty->settings.termios.c_lflag & ICANON) == 0)) { if ((!tty->is_master) && ((tty->settings->termios.c_lflag & ICANON) == 0)) {
canon = false; canon = false;
if (!dontBlock) { if (!dontBlock) {
// Non-blocking mode. Handle VMIN and VTIME. // Non-blocking mode. Handle VMIN and VTIME.
bytesNeeded = tty->settings.termios.c_cc[VMIN]; bytesNeeded = tty->settings->termios.c_cc[VMIN];
bigtime_t vtime = tty->settings.termios.c_cc[VTIME] * 100000; bigtime_t vtime = tty->settings->termios.c_cc[VTIME] * 100000;
TRACE(("tty_input_read: icanon vmin %lu, vtime %" B_PRIdBIGTIME TRACE(("tty_input_read: icanon vmin %lu, vtime %" B_PRIdBIGTIME
"us\n", bytesNeeded, vtime)); "us\n", bytesNeeded, vtime));
@ -1547,7 +1554,7 @@ tty_read(tty_cookie* cookie, void* _buffer, size_t* _length)
bool* hitEOF = canon && tty->pending_eof > 0 ? &_hitEOF : NULL; bool* hitEOF = canon && tty->pending_eof > 0 ? &_hitEOF : NULL;
ssize_t bytesRead = line_buffer_user_read(tty->input_buffer, buffer, ssize_t bytesRead = line_buffer_user_read(tty->input_buffer, buffer,
toRead, tty->settings.termios.c_cc[VEOF], hitEOF); toRead, tty->settings->termios.c_cc[VEOF], hitEOF);
if (bytesRead < 0) { if (bytesRead < 0) {
status = bytesRead; status = bytesRead;
break; break;
@ -1622,7 +1629,7 @@ tty_control(tty_cookie* cookie, uint32 op, void* buffer, size_t length)
case TCGETA: case TCGETA:
TRACE(("tty: get attributes\n")); TRACE(("tty: get attributes\n"));
return user_memcpy(buffer, &tty->settings.termios, return user_memcpy(buffer, &tty->settings->termios,
sizeof(struct termios)); sizeof(struct termios));
case TCSETA: case TCSETA:
@ -1631,16 +1638,16 @@ tty_control(tty_cookie* cookie, uint32 op, void* buffer, size_t length)
{ {
TRACE(("tty: set attributes (iflag = %" B_PRIx32 ", oflag = %" TRACE(("tty: set attributes (iflag = %" B_PRIx32 ", oflag = %"
B_PRIx32 ", cflag = %" B_PRIx32 ", lflag = %" B_PRIx32 ")\n", B_PRIx32 ", cflag = %" B_PRIx32 ", lflag = %" B_PRIx32 ")\n",
tty->settings.termios.c_iflag, tty->settings.termios.c_oflag, tty->settings->termios.c_iflag, tty->settings->termios.c_oflag,
tty->settings.termios.c_cflag, tty->settings->termios.c_cflag,
tty->settings.termios.c_lflag)); tty->settings->termios.c_lflag));
status_t status = user_memcpy(&tty->settings.termios, buffer, status_t status = user_memcpy(&tty->settings->termios, buffer,
sizeof(struct termios)); sizeof(struct termios));
if (status != B_OK) if (status != B_OK)
return status; return status;
tty->service_func(tty, TTYSETMODES, &tty->settings.termios, tty->service_func(tty, TTYSETMODES, &tty->settings->termios,
sizeof(struct termios)); sizeof(struct termios));
return status; return status;
} }
@ -1649,25 +1656,25 @@ tty_control(tty_cookie* cookie, uint32 op, void* buffer, size_t length)
case TIOCGWINSZ: case TIOCGWINSZ:
TRACE(("tty: get window size\n")); TRACE(("tty: get window size\n"));
return user_memcpy(buffer, &tty->settings.window_size, return user_memcpy(buffer, &tty->settings->window_size,
sizeof(struct winsize)); sizeof(struct winsize));
case TIOCSWINSZ: case TIOCSWINSZ:
{ {
uint16 oldColumns = tty->settings.window_size.ws_col; uint16 oldColumns = tty->settings->window_size.ws_col;
uint16 oldRows = tty->settings.window_size.ws_row; uint16 oldRows = tty->settings->window_size.ws_row;
TRACE(("tty: set window size\n")); TRACE(("tty: set window size\n"));
if (user_memcpy(&tty->settings.window_size, buffer, if (user_memcpy(&tty->settings->window_size, buffer,
sizeof(struct winsize)) < B_OK) { sizeof(struct winsize)) < B_OK) {
return B_BAD_ADDRESS; return B_BAD_ADDRESS;
} }
// send a signal only if the window size has changed // send a signal only if the window size has changed
if ((oldColumns != tty->settings.window_size.ws_col if ((oldColumns != tty->settings->window_size.ws_col
|| oldRows != tty->settings.window_size.ws_row) || oldRows != tty->settings->window_size.ws_row)
&& tty->settings.pgrp_id != 0) { && tty->settings->pgrp_id != 0) {
send_signal(-tty->settings.pgrp_id, SIGWINCH); send_signal(-tty->settings->pgrp_id, SIGWINCH);
} }
return B_OK; return B_OK;
@ -1855,7 +1862,7 @@ tty_select(tty_cookie* cookie, uint8 event, uint32 ref, selectsync* sync)
// In case input is echoed, we have to check, whether we can // In case input is echoed, we have to check, whether we can
// currently can write to our TTY as well. // currently can write to our TTY as well.
bool echo = (tty->is_master bool echo = (tty->is_master
&& tty->settings.termios.c_lflag & ECHO); && tty->settings->termios.c_lflag & ECHO);
if (otherTTY->writer_queue.IsEmpty() if (otherTTY->writer_queue.IsEmpty()
&& line_buffer_writable(otherTTY->input_buffer) > 0) { && line_buffer_writable(otherTTY->input_buffer) > 0) {

View File

@ -138,7 +138,7 @@ struct tty {
uint32 pending_eof; uint32 pending_eof;
bool is_master; bool is_master;
recursive_lock* lock; recursive_lock* lock;
tty_settings settings; tty_settings* settings;
uint8 hardware_bits; uint8 hardware_bits;
}; };