[PATCH] libmisc/shell: Work around tmux bug in row and column

chrisj at rtems.org chrisj at rtems.org
Wed Dec 13 02:47:48 UTC 2023


From: Chris Johns <chrisj at rtems.org>

- Extend the timeout to 150 msec for long remote sessions

- Improve the performance of the detection

Closes #4975
Closes #4977
---
 cpukit/libmisc/shell/shell.c | 141 ++++++++++++++++++++++++++++-------
 1 file changed, 116 insertions(+), 25 deletions(-)

diff --git a/cpukit/libmisc/shell/shell.c b/cpukit/libmisc/shell/shell.c
index cd83aa56d1..3aded5440f 100644
--- a/cpukit/libmisc/shell/shell.c
+++ b/cpukit/libmisc/shell/shell.c
@@ -805,6 +805,69 @@ void rtems_shell_print_env(
 }
 #endif
 
+/*
+ * Wait for the string to return or timeout.
+ */
+static bool rtems_shell_term_wait_for(const int fd, const char* str, const int timeout)
+{
+  int msec = timeout;
+  int i = 0;
+  while (msec-- > 0 && str[i] != '\0') {
+    char ch[2];
+    if (read(fd, &ch[0], 1) == 1) {
+      fflush(stdout);
+      if (ch[0] != str[i++]) {
+        return false;
+      }
+      msec = timeout;
+    } else {
+      usleep(1000);
+    }
+  }
+  if (msec == 0) {
+    return false;
+  }
+  return true;
+}
+
+/*
+ * Buffer a string up to the end string
+ */
+static int rtems_shell_term_buffer_until(const int fd,
+                                         char* buf,
+                                         const int size,
+                                         const char* end,
+                                         const int timeout)
+{
+  int msec = timeout;
+  int i = 0;
+  int e = 0;
+  memset(&buf[0], 0, size);
+  while (msec-- > 0 && i < size && end[e] != '\0') {
+    char ch[2];
+    if (read(fd, &ch[0], 1) == 1) {
+      fflush(stdout);
+      buf[i++] = ch[0];
+      if (ch[0] == end[e]) {
+        e++;
+      } else {
+        e = 0;
+      }
+      msec = timeout;
+    } else {
+      usleep(1000);
+    }
+  }
+  if (msec == 0 || end[e] != '\0') {
+    return -1;
+  }
+  i -= e;
+  if (i < size) {
+    buf[i] = '\0';
+  }
+  return i;
+}
+
 /*
  * Direct method to get the size of an XTERM window.
  *
@@ -831,10 +894,39 @@ static void rtems_shell_winsize( void )
       term.c_cc[VMIN] = 0;
       term.c_cc[VTIME] = 0;
       if (tcsetattr (fd, TCSADRAIN, &term) >= 0) {
-        int msec = 50;
-        int len = 0;
-        int i = 0;
+        const int timeout = 150;
+        bool row_cols_swapped = false;
         memset(&buf[0], 0, sizeof(buf));
+        /*
+         * https://github.com/tmux/tmux/issues/3457
+         *
+         * Tmux has a bug where the lines and cols are swapped. There is a lag
+         * in the time it takes to get the fix into code so see if tmux is
+         * running and which version and work around the bug.
+         *
+         * CSI > Ps q
+         *    Ps = 0   =>   DCS > | text ST
+         */
+        fputs("\033[>0q", stdout);
+        fflush(stdout);
+        if (rtems_shell_term_wait_for(fd, "\033P>|", timeout)) {
+          int len = rtems_shell_term_buffer_until(fd, buf, sizeof(buf), "\033\\", timeout);
+          if (len > 0) {
+            if (memcmp(buf, "tmux ", 5) == 0) {
+              static const char* bad_versions[] = {
+                "3.2", "3.2a", "3.3", "3.3a"
+              };
+              #define bad_versions_num (sizeof(bad_versions) / sizeof(bad_versions[0]))
+              size_t i;
+              for (i = 0; i < bad_versions_num; ++i) {
+                if (strcmp(bad_versions[i], buf + 5) == 0) {
+                  row_cols_swapped = true;
+                  break;
+                }
+              }
+            }
+          }
+        }
         /*
          * https://invisible-island.net/xterm/ctlseqs/ctlseqs.html#h3-Miscellaneous
          *
@@ -842,33 +934,32 @@ static void rtems_shell_winsize( void )
          */
         fputs("\033[18t", stdout);
         fflush(stdout);
-        while (msec-- > 0 && len < sizeof(buf)) {
-          char ch[2];
-          if (read(fd, &ch[0], 1) == 1) {
-            buf[len++] = ch[0];
-            msec = 50;
-          } else {
-            usleep(1000);
-          }
-        }
-        while (i < len) {
-          static const char resp[] = "\033[8;";
-          if (memcmp(resp, &buf[i], sizeof(resp) - 1) == 0) {
-            i += sizeof(resp) - 1;
-            while (i < len && buf[i] != ';') {
+        if (rtems_shell_term_wait_for(fd, "\033[8;", timeout)) {
+          int len = rtems_shell_term_buffer_until(fd, buf, sizeof(buf), ";", timeout);
+          if (len > 0) {
+            int i;
+            lines = 0;
+            i = 0;
+            while (i < len) {
               lines *= 10;
               lines += buf[i++] - '0';
             }
-            cols = 0;
-            ++i;
-            while (i < len && buf[i] != 't') {
-              cols *= 10;
-              cols += buf[i++] - '0';
+            len = rtems_shell_term_buffer_until(fd, buf, sizeof(buf), "t", timeout);
+            if (len > 0) {
+              cols = 0;
+              i = 0;
+              while (i < len) {
+                cols *= 10;
+                cols += buf[i++] - '0';
+              }
+              ok = true;
             }
-          } else {
-            i++;
           }
-          ok = true;
+        }
+        if (ok && row_cols_swapped) {
+          int tmp = lines;
+          lines = cols;
+          cols = tmp;
         }
       }
       tcsetattr (fd, TCSADRAIN, &cterm);
-- 
2.42.0



More information about the devel mailing list