diff options
-rw-r--r-- | lldb/source/Plugins/Process/Linux/ProcessLinux.cpp | 36 | ||||
-rw-r--r-- | lldb/source/Plugins/Process/Linux/ProcessLinux.h | 9 | ||||
-rw-r--r-- | lldb/source/Plugins/Process/Linux/ProcessMonitor.cpp | 27 |
3 files changed, 72 insertions, 0 deletions
diff --git a/lldb/source/Plugins/Process/Linux/ProcessLinux.cpp b/lldb/source/Plugins/Process/Linux/ProcessLinux.cpp index f62a303505a..b3405dda944 100644 --- a/lldb/source/Plugins/Process/Linux/ProcessLinux.cpp +++ b/lldb/source/Plugins/Process/Linux/ProcessLinux.cpp @@ -8,6 +8,8 @@ //===----------------------------------------------------------------------===// // C Includes +#include <errno.h> + // C++ Includes // Other libraries and framework includes #include "lldb/Core/PluginManager.h" @@ -397,6 +399,40 @@ ProcessLinux::GetByteOrder() const return m_byte_order; } +size_t +ProcessLinux::PutSTDIN(const char *buf, size_t len, Error &error) +{ + ssize_t status; + if ((status = write(m_monitor->GetTerminalFD(), buf, len)) < 0) + { + error.SetErrorToErrno(); + return 0; + } + return status; +} + +size_t +ProcessLinux::GetSTDOUT(char *buf, size_t len, Error &error) +{ + ssize_t bytes_read; + + // The terminal file descriptor is always in non-block mode. + if ((bytes_read = read(m_monitor->GetTerminalFD(), buf, len)) < 0) + { + if (errno != EAGAIN) + error.SetErrorToErrno(); + return 0; + } + return bytes_read; +} + +size_t +ProcessLinux::GetSTDERR(char *buf, size_t len, Error &error) +{ + return GetSTDOUT(buf, len, error); +} + + //------------------------------------------------------------------------------ // ProcessInterface protocol. diff --git a/lldb/source/Plugins/Process/Linux/ProcessLinux.h b/lldb/source/Plugins/Process/Linux/ProcessLinux.h index 18e5aa712d0..733561195de 100644 --- a/lldb/source/Plugins/Process/Linux/ProcessLinux.h +++ b/lldb/source/Plugins/Process/Linux/ProcessLinux.h @@ -138,6 +138,15 @@ public: virtual lldb::addr_t GetImageInfoAddress(); + virtual size_t + PutSTDIN(const char *buf, size_t len, lldb_private::Error &error); + + virtual size_t + GetSTDOUT(char *buf, size_t len, lldb_private::Error &error); + + virtual size_t + GetSTDERR(char *buf, size_t len, lldb_private::Error &error); + //------------------------------------------------------------------ // PluginInterface protocol //------------------------------------------------------------------ diff --git a/lldb/source/Plugins/Process/Linux/ProcessMonitor.cpp b/lldb/source/Plugins/Process/Linux/ProcessMonitor.cpp index 490ca6f60a3..c5909c85a6d 100644 --- a/lldb/source/Plugins/Process/Linux/ProcessMonitor.cpp +++ b/lldb/source/Plugins/Process/Linux/ProcessMonitor.cpp @@ -114,6 +114,27 @@ DoWriteMemory(lldb::pid_t pid, unsigned word_size, return bytes_written; } +// Simple helper function to ensure flags are enabled on the given file +// descriptor. +static bool +EnsureFDFlags(int fd, int flags, Error &error) +{ + int status; + + if ((status = fcntl(fd, F_GETFL)) == -1) + { + error.SetErrorToErrno(); + return false; + } + + if (fcntl(fd, F_SETFL, status | flags) == -1) + { + error.SetErrorToErrno(); + return false; + } + + return true; +} //------------------------------------------------------------------------------ /// @class Operation @@ -706,6 +727,12 @@ ProcessMonitor::Launch(LaunchArgs *args) monitor->m_terminal_fd = terminal.ReleaseMasterFileDescriptor(); monitor->m_pid = pid; + // Set the terminal fd to be in non blocking mode (it simplifies the + // implementation of ProcessLinux::GetSTDOUT to have a non-blocking + // descriptor to read from). + if (!EnsureFDFlags(monitor->m_terminal_fd, O_NONBLOCK, args->m_error)) + goto FINISH; + // Update the process thread list with this new thread and mark it as // current. inferior.reset(new LinuxThread(process, pid)); |