From 73b41feba5af9980ee7318d09e744a3e749f8fd4 Mon Sep 17 00:00:00 2001 From: Simon Tatham Date: Sun, 3 Sep 2023 09:29:56 +0100 Subject: [PATCH] Rationalise the code that resets terminal scrollback. Recently I encountered a CLI tool that took tens of seconds to run, and produced no _visible_ output, but wrote ESC[0m to the terminal a few times during its operation. (Probably by mistake. In other modes it does print colourful messages, so I expect a 'reset colour' call was accidentally outside the 'if' statement containing the rest of the diagnostic it followed. Or something along those lines.) I noticed this because every ESC[0m reset my pterm scrollback to the bottom, which wasn't very helpful, and was unintentional on pterm's part (as _well_ as on the part of the tool). But I can fix pterm! At first glance the code _looked_ sensible: terminal.c contains calls to seen_disp_event(term) whenever terminal output does something that requires a redraw of the terminal window. Those are also the updates that should count as 'reset scrollback on display activity'. And ESC[0m, along with the rest of the SGR handler, correctly contained no such call. So how did a display update happen at all? The code was confusingly tangled up with the code that responds to terminal activity by resetting the phase of the blinking cursor (if any). term_reset_cblink() was calling seen_disp_event() (when surely it should be the other way round!), and also, term_reset_cblink() was called whenever _any_ terminal output data arrived. That combination meant that any byte output to the terminal at all turned out to count as display activity, whether or not it changed the screen contents. Additionally, the other scrollback-reset flag, 'reset scrollback on keypress', was handled by calling seen_disp_event() from the keyboard handler. But display events and keyboard events are supposed to be _independent_ potential causes of scrollback resets - it doesn't make any sense to handle one by treating it as the other! So I've reorganised the code completely: - the seen_disp_event *flag* is now gone. Instead, the seen_disp_event function tests the scroll_on_disp flag, and if set, resets the scroll position immediately and sets the general 'scrollbar needs updating' flag. - keyboard input is handled by doing exactly the same thing except testing the scroll_on_key flag, so the two systems are properly independent. That code calls term_schedule_update so that the terminal will be redrawn as a result of the scroll, but doesn't also call seen_disp_event() for the rest of the full treatment. - the term_update code that does the scrollbar update is much simpler, since now it only needs to test that one flag. - I also had to set that flag explicitly in scroll() so that the scrollbar would still be updated as a result of the scrollback size changing. I think that must have been happening entirely by accident before. - term_reset_cblink is subsumed into seen_disp_event, so that only _substantive_ display updates cause the cursor blink phase to reset to the start of the solid period. Result: if programs output no-op sequences like ESC[0m, or if you press keys that don't echo, then the cursor will carry on blinking normally, and (if you don't also have scroll_on_key set) the scrollback won't be reset. And the code is slightly shorter than it was before, and hopefully more sensible too. (However, other classes of no-op activity _will_ still cause a cursor blink phase change and a scrollback reset, such as sending a cursor-positioning sequence that puts the cursor in the same place it was already - even something as simple as ^M when already at the start of the line. It might be nice to fix that, but it's much more difficult: you'd have to either put a complicated and error-prone test at every seen_disp_event call site, or else expensively diff the entire visible terminal state against how it was before. And to avoid a nondeterministic dependency on the terminal update cooldown, that diff would have to be done at the granularity of individual control sequences rather than a bounded number of times a second. I'd rather not!) (cherry picked from commit bdbd5f429c79a67cc1d5fee7f121b9fe642466ad) (cherry-picker's note: I also had to remove the initialisation of the removed field term->seen_disp_event from term_init(), which didn't need doing on main because commit 74aa3cb7fb088d8 had already replaced all the boring parts of term_init with a big memset) --- terminal/terminal.c | 54 +++++++++++++++++++++------------------------ terminal/terminal.h | 1 - 2 files changed, 25 insertions(+), 30 deletions(-) diff --git a/terminal/terminal.c b/terminal/terminal.c index c5af4dc6..5c375562 100644 --- a/terminal/terminal.c +++ b/terminal/terminal.c @@ -1314,12 +1314,21 @@ static void term_schedule_update(Terminal *term) } /* - * Call this whenever the terminal window state changes, to queue - * an update. + * Call this whenever the terminal window state changes, to queue an + * update. This also resets the phase of cursor blinking, so that the + * cursor remains visible as it moves with the output, and sets a flag + * to indicate that if we have the 'reset scrollback on display + * activity' setting enabled, then we should activate it. */ static void seen_disp_event(Terminal *term) { - term->seen_disp_event = true; /* for scrollback-reset-on-activity */ + if (term->scroll_on_disp) { + term->disptop = 0; + term->win_scrollbar_update_pending = true; + } + term->cblinker = true; + term->cblink_pending = false; + term_schedule_cblink(term); term_schedule_update(term); } @@ -1354,17 +1363,6 @@ static void term_schedule_cblink(Terminal *term) } } -/* - * Call to reset cursor blinking on new output. - */ -static void term_reset_cblink(Terminal *term) -{ - seen_disp_event(term); - term->cblinker = true; - term->cblink_pending = false; - term_schedule_cblink(term); -} - /* * Call to begin a visual bell. */ @@ -1531,17 +1529,10 @@ void term_update(Terminal *term) } if (win_setup_draw_ctx(term->win)) { - bool need_sbar_update = term->seen_disp_event || - term->win_scrollbar_update_pending; - term->win_scrollbar_update_pending = false; - if (term->seen_disp_event && term->scroll_on_disp) { - term->disptop = 0; /* return to main screen */ - term->seen_disp_event = false; - need_sbar_update = true; - } - - if (need_sbar_update) + if (term->win_scrollbar_update_pending) { + term->win_scrollbar_update_pending = false; update_sbar(term); + } do_paint(term); win_set_cursor_pos( term->win, term->curs.x, term->curs.y - term->disptop); @@ -1574,9 +1565,10 @@ void term_seen_key_event(Terminal *term) /* * Reset the scrollback on keypress, if we're doing that. */ - if (term->scroll_on_key) { - term->disptop = 0; /* return to main screen */ - seen_disp_event(term); + if (term->scroll_on_key && term->disptop != 0) { + term->disptop = 0; + term->win_scrollbar_update_pending = true; + term_schedule_update(term); } } @@ -2042,7 +2034,6 @@ Terminal *term_init(Conf *myconf, struct unicode_data *ucsdata, TermWin *win) term->print_job = NULL; term->vt52_mode = false; term->cr_lf_return = false; - term->seen_disp_event = false; term->mouse_is_down = 0; term->reset_132 = false; term->cblinker = false; @@ -2694,6 +2685,12 @@ static void scroll(Terminal *term, int topline, int botline, */ if (term->disptop > -term->savelines && term->disptop < 0) term->disptop--; + + /* + * We've just modified the data that the terminal's + * scrollbar is based on, so remember to update it. + */ + term->win_scrollbar_update_pending = true; } resizeline(term, line, term->cols); clear_line(term, line); @@ -7815,7 +7812,6 @@ static void term_added_data(Terminal *term, bool called_from_term_data) { if (!term->in_term_out) { term->in_term_out = true; - term_reset_cblink(term); term_out(term, called_from_term_data); term->in_term_out = false; } diff --git a/terminal/terminal.h b/terminal/terminal.h index 87441855..d00133b9 100644 --- a/terminal/terminal.h +++ b/terminal/terminal.h @@ -147,7 +147,6 @@ struct terminal_tag { long vbell_end; bool app_cursor_keys, app_keypad_keys, vt52_mode; bool repeat_off, srm_echo, cr_lf_return; - bool seen_disp_event; bool big_cursor; bool xterm_mouse_forbidden;