I maintain an extension for the Bash environment called Basta. Basta provides a scroll-protected status line at the bottom of your ANSI/VT100 terminal.
When Basta sets itself up, the effective number of lines, as known by the termios stuff in the kernel and the shell LINES
variable decreases by 1. Resizing the terminal is handled nicely. Almost. There is trap
for the SIGWINCH
signal to call the update routine, which is also called after every command, prior to returning to the prompt.
However, from time to time I see a situation in which the scrolling region is lost. The prompt is being painted over the bottom line, where the user is editing commands and where command output appears, resulting in a mess.
I have a hypothesis on one how it can occur, which has reliable repro steps.
We run a program that waits for input, while Bash is in the background, such as
cat
.While this program is running, we resize the window such that we shrink it by one line.
Terminate the
cat
program.
If the terminal is resized while cat
is running, Bash does not get the SIGWINCH
(because, I think, SIGWINCH
is only sent to the processes in the terminal sessions foreground process group, and Bash is in the Background at that moment). Our trap
does not execute.
Basta's update function also relies on comparing the previous terminal size to the current to detect a size change.
Here is the problem: in this situation where a terminal with a scroll-protected status like has shrunk by one line, the number of lines appears not to have changed.
E.g. 40 line screen with 39 line scrolling region: LINES=39
. Resize terminal to be 39 lines long. Scrolling region is gone. LINES=39
again. To the software, it looks like the same size, so it concludes that there has not been a resize. The status line is now being painted over top of the bottom row, over top of the user's input, because Basta wants to put it on row 40, which doesn't exist any more, so the cursor clamps to 39.
If Basta knows what a window size change occurred, it will not do the size comparison; in that situation it takes the slow path whereby it queries the terminal itself to determine the size. (It would be undesirable to do that for each update, because sending queries to the terminal is dodgy. If the user is typing rapidly, the terminal response can get mixed up with their keystrokes, and there can be a noticeable delay in getting a response from the terminal over laggy remote connections.)
Is there any clever way to know that the terminal has changed (or at least suspect it with a reasonably low false positive rate), in the absence of having received the WINCH
trap, due to having been in the background, without talking to the terminal?
tcgetattr
in<termios.h>
. We are a shell script, so our interface to the line count in the kernel isstty -a
which dumps out info likerows 40 cols 80
. In the problem situation, this doesn't help; the value matches what's in$LINES
and also jibes with the previously known value that the script knows about. I.e. the size appears to be the same, though the terminal is smaller and we lost the scroll-protected line.stty -a
would reportrows 41
, andLINES
was 41. While we are backgrounded due to some program running, the window is resized to 41 lines. When the program terminates, we haverows 41
andLINES=41
. But that's the full size of the window; the scroll-protected line is gone! We didn't get aSIGWINCH
, and the sizes look the same. Oops!LINES=41
and sttyrows 42
. After background you hadLINES=41
and sttyrows 41
. 3) This change you can detect, without needing any signal.