-
Notifications
You must be signed in to change notification settings - Fork 299
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
TTY always created with size 24 x 80 and then resized #1777
Comments
There's a brand new relevant VTE bugreport at https://gitlab.gnome.org/GNOME/vte/issues/188. I'm wondering if 80x24 is perhaps Tilix's workaround against 0x0...? |
Thanks for the link! If I were getting 0x0, that would be fantastic! I would know that I have to wait for the real data to come through. Either with busy waiting or by listening to
As I mentioned above, Is it possibly that some kernel versions give 0x0 while others give the fake 24x80? I've only tested it on 4.15.0-65, which is pretty old. |
I honestly don't know if it can somehow override the temporary default.
It's also possible, but I wouldn't bet on this, I'd guess the kernel never had a hardwired 24x80. It could also be that we're facing a slightly different issue. It's all about to be investigated. |
I should've mentioned that this affects not just my personal code. It breaks zsh prompt if it's printed fast enough. To reproduce:
typeset COLUMNS LINES
PROMPT='left> '
RPROMPT='<right'
return
Here's what I've got: The left window is the original. Notice that it reports correct dimensions (10 x 90 is what I set as the default size in Tilix preferences). The right window reports incorrect size of 24 x 80. It also has broken prompt. That inverted |
I've seen the zsh |
Here's a bit of extra information about that When options
If this sequence is printed when the cursor is at the very first column, it will simply clear the current line. If the cursor is not on the first column, it'll append Now let's see what happens when we create a new terminal in Tilix that has real width of 45 but incorrectly reports it as 80. Instead of printing |
I was wrong when I stated that whenever the initial size is fake there will always be setopt trapsasync
trap 'echo WINCH ${LINES}x${COLUMNS}' WINCH
echo BEGIN ${LINES}x${COLUMNS}
sleep 1
echo END ${LINES}x${COLUMNS} With default terminal size set to 80x24 in Preferences:
With default terminal size set to 90x10 in Preferences:
The very last output looks like an independent bug. Based on these test cases I've written the following workaround for zsh. When executed upon zsh startup, it will block until the correct terminal size is determined. () {
emulate -L zsh
if [[ $LINES == (24|0) && $COLUMNS == (80|0) ]]; then
setopt monitor trapsasync
zmodload zsh/datetime
zmodload zsh/system
local -F deadline=$((EPOCHREALTIME+0.03))
local -i fd pid
trap 'kill -- -$pid' WINCH
exec {fd}< <(
echo $sysparams[pid]
while [[ $LINES == (24|0) && $COLUMNS == (80|0) && $EPOCHREALTIME -lt $deadline &&
"$(/bin/stty size 2>/dev/null)" == (24|0)\ (80|0) ]]; do
done)
IFS= read -u $fd pid
( read -u $fd )
exec {fd}>&-
trap - WINCH
fi
} The biggest problem is that it blocks for 30 milliseconds (configurable timeout) when the terminal is in fact 24x80. In all other cases it's faster. |
Could you please test the patch from the VTE bug? |
I've never built VTE (I only learned about its existence today), so I went to https://gitlab.gnome.org/GNOME/vte and started following the instructions.
This is where I stopped. How deep is the rabbit hole? If you are able to build VTE with your patch, perhaps you can check whether it fixes Tilix? It might be easier that way.
echo ${LINES}x${COLUMNS}
return
|
The rabbit hole is not deep at all. Just get new meson (e.g. from a newer version of your distro, or manually), or VTE version 0.56 which uses the autotools build system. I tried with tilix but I couldn't reproduce the problem (the one that I posted in the xfce bugtracker – not the zsh one). I could only reproduce that problem with xfce4-terminal and gnome-terminal, and the patch fixes them for me. |
Meson is python only. You can update to the latest release using |
Went through the whack-a-mole of errors for some time and gave up on this:
|
Sorry, closed prematurely. I managed to built vte after the usual whack-a-mole with errors. Here's what worked: sudo apt install python3-pip meson gtk-doc-tools intltool libgtk-3-dev libfribidi-dev libpcre2-dev libpango1.0-dev libgnutls28-dev
pip3 install meson --upgrade
git clone https://gitlab.gnome.org/GNOME/vte.git/
cd vte
git checkout vte-0-56
curl -fsSLO https://gitlab.gnome.org/GNOME/vte/uploads/52f7c8febfc94b3e8e2c3e7a7a7d2abf/vte-188-initial-size.patch
git apply vte-188-initial-size.patch
./autogen.sh --disable-introspection --disable-vala
make -j 20
sudo cp src/.libs/libvte-2.91.so.0.5600.5 /usr/lib/x86_64-linux-gnu/
sudo mv /usr/lib/x86_64-linux-gnu/libvte-2.91.so.0 /usr/lib/x86_64-linux-gnu/libvte-2.91.so.0.bak
sudo ln -s libvte-2.91.so.0.5600.5 /usr/lib/x86_64-linux-gnu/libvte-2.91.so.0 My original I then restarted my machine for good measure and verified that Then I reverted the patch and rebuilt vte to check that the patch actually makes a difference. It's a good thing that I did, because there is no effect on |
That sounds bad.
Do you mean that the exact same binary (same md5sum) is produced, no matter if you apply the patch or not? I seriously doubt that (and just verified with the vte-0-56 branch that the generated file becomes different for me). |
This is explained by the fact that the binary I built is the same with the patch as without.
Correct.
Thanks for confirming that your patch is expected to have an effect on these files and that it's indeed these files that I need to install to fix Tilix. I was uncertain about both of these points. I'll get back to this later today or tomorrow and will once again build vte with and without the patch. I must have messed something up in my first attempt. |
Indeed I have. Here's what I did now: Built two versions of vte: regular and patched. function build-vte() (
set -e
git clone https://gitlab.gnome.org/GNOME/vte.git/ vte"$1"
cd vte"$1"
git checkout vte-0-56
if [[ -n "$1" ]]; then
curl -fsSLO 'https://gitlab.gnome.org/GNOME/vte/uploads/52f7c8febfc94b3e8e2c3e7a7a7d2abf/vte-188-initial-size.patch'
git apply vte-188-initial-size.patch
fi
./autogen.sh --disable-introspection --disable-vala
make -j 20
)
cd "$(mktemp -d)"
build-vte
build-vte -patched Installed the patched sudo cp vte-patched/src/.libs/libvte-2.91.so.0.5600.5 /usr/lib/x86_64-linux-gnu/
sudo mv /usr/lib/x86_64-linux-gnu/libvte-2.91.so.0{,.bak}
sudo ln -s libvte-2.91.so.0.5600.5 /usr/lib/x86_64-linux-gnu/libvte-2.91.so.0 Verified that the patched version is installed:
Verified that the patched and the regular versions are different.
Rebooted. Populated my setopt trapsasync
trap 'echo WINCH ${LINES}x${COLUMNS}' WINCH
echo BEGIN ${LINES}x${COLUMNS}
sleep 1
echo END ${LINES}x${COLUMNS} Launched
This exhibits the same bug as before: the initial terminal size is reported as 24x80 even though it's 24x39. Soon after the terminal is created, zsh receives WINCH. After that the size is reported correctly. This is the same behavior as I observed without the patch. Did I do anything wrong? Should I try something else? Do you need any information from me? P.S. Just in case this matters.
Tilix doesn't have a dependency on |
So, Tilix indeed doesn't work correctly for you. However, it's unclear to me if it's VTE's fault or Tilix's, we could be seeing a different bug.
You did everything great. Let's do the following, both to double check that you're indeed using a patched VTE, and to gather further information: Place this line right after (or before) the one added by the patch:
These will obviously be printed in the terminal where you start Tilix from (presumably a different terminal emulator). Oh, maybe you don't even have to do it. I've tried it and I get 80x24 all the time, even when splitting. This is bad. These numbers come from VTE itself, near the top of You might want to repeat these steps to confirm that you get the same results as me, but it's not necessary, as you wish. PS. Don't waste your time rebooting. Just make sure to quit all Tilix instances at once, in order to pick up the freshly installed VTE. Note for myself: The next for me to try is to place the D equivalent of |
Confirmed.
Also confirmed.
Thanks for looking into this! |
Seems like there is another bug in Tilix or underlying infrastructure. From the Tilix user's point of view, there are 3 bugs related to the terminal size reporting. They may or may not have a single culprit. Bug 1 When creating a new terminal of size 40x100 (or any other size as long as it's not 24x80), (But we can just wait for But 2 When creating a new terminal of size 40x100 (or any other size as long as it's not 24x80), (But if we do receive Bug 3 When creating a new terminal of size 40x100 (or any other size as long as it's not 24x80), (To reproduce this bug it may be necessary to create very high number of terminals quickly.) The practical outcome of these bugs is that The only way to get the correct terminal size appears to be this:
The presence of step (3) implies that we cannot always get the correct terminal size, no matter what we do and how long we wait. Without this step the loop will never terminate when the real terminal size is If |
Thanks for your input! I suspect that there are two race condition we're hunting, one (between the parent and child after forking, nothing to do with GTK) is already fixed in the VTE bug, and the other one (around the GTK event loop) is yet to be fixed. I'd really like to get to the bottom of these. I suspect that your bugs 2 & 3 are bash+zsh bugs (limitations) and not VTE/Tilix ones. Try this:
You'll notice that during those 5 seconds, subsequently arriving WINCHes (while the handler of a previous one is running) are swallowed by bash and zsh, that is, you can't reliably handle quickly arriving WINCHes in shells. The
Just my 2 cents: It looks overengineering for me, at least for such a workaround that shouldn't be necessary once we track down and fix the VTE problem. I'd personally just go with a |
To amend my previous post:
|
I'm not sure I understand how this explains bug (2). In that bug I'm receiving zero WINCH signals when I would expect to receive one. I'm also not sure how this explains bug (3). In that bug I'm calling
I wasn't expecting it to. I set this option so that my signal handler gets called while the top-level zsh is waiting for the foreground child to terminate. This point is moot because I no longer handle WINCH in my workaround code.
That's a good idea. I wrote this: #include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/ioctl.h>
#include <unistd.h>
#define CHECK(...) \
if (!(__VA_ARGS__)) { \
perror(#__VA_ARGS__); \
exit(1); \
} else (void)0
static void print_tty_size(const char* when) {
struct winsize w;
CHECK(ioctl(STDOUT_FILENO, TIOCGWINSZ, &w) != -1);
CHECK(printf("%s: %dx%d\n", when, w.ws_row, w.ws_col) >= 0);
}
int main(void) {
sigset_t sigset;
siginfo_t siginfo;
CHECK(sigemptyset(&sigset) == 0);
CHECK(sigaddset(&sigset, SIGWINCH) == 0);
CHECK(sigprocmask(SIG_BLOCK, &sigset, NULL) == 0);
print_tty_size("Initial");
CHECK(sigwaitinfo(&sigset, &siginfo) == SIGWINCH);
print_tty_size("After WINCH");
while (1) sleep(1);
} I can confirm that bug (2) was indeed a bug in zsh as it never happens with this simple program. I've written the following test case to demonstrate the bug in zsh. In some cases the following piece of code can print () {
emulate -L zsh
setopt trapsasync
TRAPWINCH() { echo WINCH }
local size=${LINES}x${COLUMNS}
while /bin/true; do
if [[ $size != ${LINES}x${COLUMNS} ]]; then
echo 'tty size changed'
size=${LINES}x${COLUMNS}
fi
done
} I haven't found a way to reproduce it consistently but the following procedure will sometimes trigger the bug:
I think, In theory, bug (3) must still trigger on my C program because I don't see how zsh can break that one. I cannot verify it though because I cannot trigger (3) on my machine even with zsh. My bug report for it was based on the results of testing done by my friend on a slower machine. It's possible that I messed something up but I'd rather not spend time investigating it (WINCH is useless to me either way).
On a reasonable laptop with idle CPU the median time to acquire correct tty size is around 60ms, which means that Note that I wrote that workaround code before I realized that () {
emulate zsh
if [[ -z $VTE_VERSION || $LINES != 24 || $COLUMNS != 80 ]]; then
echo "Real terminal size: $LINES $COLUMNS"
return
fi
zmodload zsh/datetime
local -F deadline=$((EPOCHREALTIME+0.1))
local size
while (( EPOCHREALTIME < deadline )) && size="$(/bin/stty size 2>/dev/null)"; do
if [[ $size != "24 80" ]]; then
echo "Real terminal size: $size"
return
fi
done
echo "Unknown terminal size."
} (Those The complexity of this code doesn't bother me at all. What does bother me is that it's slow when the real terminal size is 24x80 (the loop terminates by timeout in this case). And when that happens, I have to do something custom because I don't know if the timeout was hit due to the machine being slow or the real terminal size being 24x80. |
At this point I can't answer your questions. Please be patient until I find time to thoroughly investigate the case and come up with actual observations and answers, rather than wild assumptions. It might take a couple of days. |
Could you please test this Tilix patch, together with an updated (v2) VTE patch from VTE issue 188? |
I've never built Tilix. Given that it uses a built toolchain I'm unfamiliar with, it would require a non-trivial time commitment (it took me over an hour to test the VTE patch). Since you've implemented a path (thank you!), I suppose you are able to build Tilix on your machine, in which case it should be relatively straightforward for you to check whether the bug is fixed either by starting my C program (#1777 (comment)) through Tilix, or by using |
I couldn't reproduce any of the zsh issues on my computer (with unpatched VTE and Tilix), I guess the timings are different. My own methods of debugging suggest that the bug is fixed in Tilix with these two patches, but it's always good to have some confirmation. Hopefully someone else will provide it, then. Thanks anyways for the efforts you put into chasing down this issue (in other terminals as well)! |
I was under the impression that you've reproduced it, based on this comment of yours:
I must have misunderstood what you meant there. Since I'm apparently the only one who can reproduce the bug I've reported, this changes things. I went ahead and attempted to verify your patch. First, I built and installed VTE 0.56 with your patch (the same as before). Then I built Tilix from HEAD (without your patch). curl -fsSLo /tmp/dmd.deb http://downloads.dlang.org/releases/2.x/2.088.1/dmd_2.088.1-0_amd64.deb
sudo dpkg -i /tmp/dmd.deb
git clone https://github.com/gnunn1/tilix.git /tmp/tilix
cd /tmp/tilix
dub build --build=release
sudo ./install.sh /usr (Side note: I sent #1783 to fix a bug in I verified that:
Then I've applied your Tilix patch and rebuilt Tilix. curl -fsSL 'https://github.com/gnunn1/tilix/files/3784639/tilix-1777-vte-188-delayed-resize-workaround.patch.txt' | git apply
dub build --build=release
sudo ./install.sh /usr I verified that:
So the patch is an improvement but not a complete fix. By the way, it was apparent even before your patch that Tilix does something different when creating a new terminal with "Create new session" vs "Add terminal right". If I busy-wait for correct terminal dimensions, it takes 10 times longer for the terminal created with "Create new session" (10ms vs 1ms). |
@egmontkob I've bumped into another bug. When clicking "New Tab" in GNOME Terminal, on rare occasions a newly created terminal can report transposed TTY dimensions for a short time. Here's how I can reproduce this:
Expected: Prints "45 175" (the real dimensions of my GNOME Terminal). I've tried replacing Is this a known issue? |
Yes I messed it up in the original patch in vte/188, please use the updated patch from there. |
Thanks for the quick response. The updated patch (v2) fixes the problem indeed. |
So, regarding the pending issue ("create new session", bullet point 2 a few comments ago): I can confirm that this is still broken. In the mean time, there's still a bug with gnome-terminal, too. The first ideas of the main VTE developer plus me didn't fix it. Seems we'll have to dig deeper to understand what is going on. I have no idea when we'll have time for this, so I can't promise anything. |
Understood. Thanks for the update. I appreciate your work on this tricky issue. |
When a new terminal tab is created by clicking "Add terminal right" or "Create new session", it often reports incorrect TTY size of 24 x 80. When this happens, SIGWINCH is delivered a few milliseconds later. After SIGWINCH the TTY size is reported correctly.
I'm writing a piece of code that prints formatted text to the TTY during shell initialization. To print it well, I need to know the terminal width. The problem is that I cannot figure out how to get it. If the TTY size is reported as 24 x 80, should I wait for SIGWINCH or not? Most of the time 24 x 80 is fake data but sometimes it's real (people do have terminals of this size). When it's real, no SIGWINCH will come, so I shouldn't wait for it. If it's fake, I must not print anything until I find out the terminal size, so I must wait for SIGWINCH.
It would be perfect if the TTY was always created with the right size to begin with. I will also be happy with a workaround if you can offer one.
I'm querying TTY size through builtin
COLUMNS
andLINES
parameters inzsh
. However,stty size
also sometimes reports 24 x 80, although it happens rarely simply because it takes a long time to execute. Even running/bin/true
is often enough of a delay to get correct TTY size.The text was updated successfully, but these errors were encountered: