1 //===-- Terminal.cpp ------------------------------------------------------===// 2 // 3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4 // See https://llvm.org/LICENSE.txt for license information. 5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6 // 7 //===----------------------------------------------------------------------===// 8 9 #include "lldb/Host/Terminal.h" 10 11 #include "lldb/Host/Config.h" 12 #include "lldb/Host/PosixApi.h" 13 #include "llvm/ADT/STLExtras.h" 14 15 #include <fcntl.h> 16 #include <signal.h> 17 18 #if LLDB_ENABLE_TERMIOS 19 #include <termios.h> 20 #endif 21 22 using namespace lldb_private; 23 24 bool Terminal::IsATerminal() const { return m_fd >= 0 && ::isatty(m_fd); } 25 26 bool Terminal::SetEcho(bool enabled) { 27 if (FileDescriptorIsValid()) { 28 #if LLDB_ENABLE_TERMIOS 29 if (IsATerminal()) { 30 struct termios fd_termios; 31 if (::tcgetattr(m_fd, &fd_termios) == 0) { 32 bool set_corectly = false; 33 if (enabled) { 34 if (fd_termios.c_lflag & ECHO) 35 set_corectly = true; 36 else 37 fd_termios.c_lflag |= ECHO; 38 } else { 39 if (fd_termios.c_lflag & ECHO) 40 fd_termios.c_lflag &= ~ECHO; 41 else 42 set_corectly = true; 43 } 44 45 if (set_corectly) 46 return true; 47 return ::tcsetattr(m_fd, TCSANOW, &fd_termios) == 0; 48 } 49 } 50 #endif // #if LLDB_ENABLE_TERMIOS 51 } 52 return false; 53 } 54 55 bool Terminal::SetCanonical(bool enabled) { 56 if (FileDescriptorIsValid()) { 57 #if LLDB_ENABLE_TERMIOS 58 if (IsATerminal()) { 59 struct termios fd_termios; 60 if (::tcgetattr(m_fd, &fd_termios) == 0) { 61 bool set_corectly = false; 62 if (enabled) { 63 if (fd_termios.c_lflag & ICANON) 64 set_corectly = true; 65 else 66 fd_termios.c_lflag |= ICANON; 67 } else { 68 if (fd_termios.c_lflag & ICANON) 69 fd_termios.c_lflag &= ~ICANON; 70 else 71 set_corectly = true; 72 } 73 74 if (set_corectly) 75 return true; 76 return ::tcsetattr(m_fd, TCSANOW, &fd_termios) == 0; 77 } 78 } 79 #endif // #if LLDB_ENABLE_TERMIOS 80 } 81 return false; 82 } 83 84 // Default constructor 85 TerminalState::TerminalState() 86 : m_tty(), m_tflags(-1), 87 #if LLDB_ENABLE_TERMIOS 88 m_termios_up(), 89 #endif 90 m_process_group(-1) { 91 } 92 93 // Destructor 94 TerminalState::~TerminalState() {} 95 96 void TerminalState::Clear() { 97 m_tty.Clear(); 98 m_tflags = -1; 99 #if LLDB_ENABLE_TERMIOS 100 m_termios_up.reset(); 101 #endif 102 m_process_group = -1; 103 } 104 105 // Save the current state of the TTY for the file descriptor "fd" and if 106 // "save_process_group" is true, attempt to save the process group info for the 107 // TTY. 108 bool TerminalState::Save(int fd, bool save_process_group) { 109 m_tty.SetFileDescriptor(fd); 110 if (m_tty.IsATerminal()) { 111 #if LLDB_ENABLE_POSIX 112 m_tflags = ::fcntl(fd, F_GETFL, 0); 113 #endif 114 #if LLDB_ENABLE_TERMIOS 115 if (m_termios_up == nullptr) 116 m_termios_up.reset(new struct termios); 117 int err = ::tcgetattr(fd, m_termios_up.get()); 118 if (err != 0) 119 m_termios_up.reset(); 120 #endif // #if LLDB_ENABLE_TERMIOS 121 #if LLDB_ENABLE_POSIX 122 if (save_process_group) 123 m_process_group = ::tcgetpgrp(0); 124 else 125 m_process_group = -1; 126 #endif 127 } else { 128 m_tty.Clear(); 129 m_tflags = -1; 130 #if LLDB_ENABLE_TERMIOS 131 m_termios_up.reset(); 132 #endif 133 m_process_group = -1; 134 } 135 return IsValid(); 136 } 137 138 // Restore the state of the TTY using the cached values from a previous call to 139 // Save(). 140 bool TerminalState::Restore() const { 141 #if LLDB_ENABLE_POSIX 142 if (IsValid()) { 143 const int fd = m_tty.GetFileDescriptor(); 144 if (TFlagsIsValid()) 145 fcntl(fd, F_SETFL, m_tflags); 146 147 #if LLDB_ENABLE_TERMIOS 148 if (TTYStateIsValid()) 149 tcsetattr(fd, TCSANOW, m_termios_up.get()); 150 #endif // #if LLDB_ENABLE_TERMIOS 151 152 if (ProcessGroupIsValid()) { 153 // Save the original signal handler. 154 void (*saved_sigttou_callback)(int) = nullptr; 155 saved_sigttou_callback = (void (*)(int))signal(SIGTTOU, SIG_IGN); 156 // Set the process group 157 tcsetpgrp(fd, m_process_group); 158 // Restore the original signal handler. 159 signal(SIGTTOU, saved_sigttou_callback); 160 } 161 return true; 162 } 163 #endif 164 return false; 165 } 166 167 // Returns true if this object has valid saved TTY state settings that can be 168 // used to restore a previous state. 169 bool TerminalState::IsValid() const { 170 return m_tty.FileDescriptorIsValid() && 171 (TFlagsIsValid() || TTYStateIsValid()); 172 } 173 174 // Returns true if m_tflags is valid 175 bool TerminalState::TFlagsIsValid() const { return m_tflags != -1; } 176 177 // Returns true if m_ttystate is valid 178 bool TerminalState::TTYStateIsValid() const { 179 #if LLDB_ENABLE_TERMIOS 180 return m_termios_up != nullptr; 181 #else 182 return false; 183 #endif 184 } 185 186 // Returns true if m_process_group is valid 187 bool TerminalState::ProcessGroupIsValid() const { 188 return static_cast<int32_t>(m_process_group) != -1; 189 } 190 191 // Constructor 192 TerminalStateSwitcher::TerminalStateSwitcher() : m_currentState(UINT32_MAX) {} 193 194 // Destructor 195 TerminalStateSwitcher::~TerminalStateSwitcher() {} 196 197 // Returns the number of states that this switcher contains 198 uint32_t TerminalStateSwitcher::GetNumberOfStates() const { 199 return llvm::array_lengthof(m_ttystates); 200 } 201 202 // Restore the state at index "idx". 203 // 204 // Returns true if the restore was successful, false otherwise. 205 bool TerminalStateSwitcher::Restore(uint32_t idx) const { 206 const uint32_t num_states = GetNumberOfStates(); 207 if (idx >= num_states) 208 return false; 209 210 // See if we already are in this state? 211 if (m_currentState < num_states && (idx == m_currentState) && 212 m_ttystates[idx].IsValid()) 213 return true; 214 215 // Set the state to match the index passed in and only update the current 216 // state if there are no errors. 217 if (m_ttystates[idx].Restore()) { 218 m_currentState = idx; 219 return true; 220 } 221 222 // We failed to set the state. The tty state was invalid or not initialized. 223 return false; 224 } 225 226 // Save the state at index "idx" for file descriptor "fd" and save the process 227 // group if requested. 228 // 229 // Returns true if the restore was successful, false otherwise. 230 bool TerminalStateSwitcher::Save(uint32_t idx, int fd, 231 bool save_process_group) { 232 const uint32_t num_states = GetNumberOfStates(); 233 if (idx < num_states) 234 return m_ttystates[idx].Save(fd, save_process_group); 235 return false; 236 } 237