forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			251 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			251 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			C++
		
	
	
	
//===-- ProcessMonitor.h -------------------------------------- -*- C++ -*-===//
 | 
						|
//
 | 
						|
//                     The LLVM Compiler Infrastructure
 | 
						|
//
 | 
						|
// This file is distributed under the University of Illinois Open Source
 | 
						|
// License. See LICENSE.TXT for details.
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
 | 
						|
#ifndef liblldb_ProcessMonitor_H_
 | 
						|
#define liblldb_ProcessMonitor_H_
 | 
						|
 | 
						|
// C Includes
 | 
						|
#include <semaphore.h>
 | 
						|
#include <signal.h>
 | 
						|
 | 
						|
// C++ Includes
 | 
						|
// Other libraries and framework includes
 | 
						|
#include "lldb/lldb-types.h"
 | 
						|
#include "lldb/Host/Mutex.h"
 | 
						|
 | 
						|
namespace lldb_private
 | 
						|
{
 | 
						|
class Error;
 | 
						|
class Module;
 | 
						|
class Scalar;
 | 
						|
} // End lldb_private namespace.
 | 
						|
 | 
						|
class ProcessLinux;
 | 
						|
class Operation;
 | 
						|
 | 
						|
/// @class ProcessMonitor
 | 
						|
/// @brief Manages communication with the inferior (debugee) process.
 | 
						|
///
 | 
						|
/// Upon construction, this class prepares and launches an inferior process for
 | 
						|
/// debugging.
 | 
						|
///
 | 
						|
/// Changes in the inferior process state are propagated to the associated
 | 
						|
/// ProcessLinux instance by calling ProcessLinux::SendMessage with the
 | 
						|
/// appropriate ProcessMessage events.
 | 
						|
///
 | 
						|
/// A purposely minimal set of operations are provided to interrogate and change
 | 
						|
/// the inferior process state.
 | 
						|
class ProcessMonitor
 | 
						|
{
 | 
						|
public:
 | 
						|
 | 
						|
    /// Launches an inferior process ready for debugging.  Forms the
 | 
						|
    /// implementation of Process::DoLaunch.
 | 
						|
    ProcessMonitor(ProcessLinux *process,
 | 
						|
                   lldb_private::Module *module,
 | 
						|
                   char const *argv[],
 | 
						|
                   char const *envp[],
 | 
						|
                   const char *stdin_path,
 | 
						|
                   const char *stdout_path,
 | 
						|
                   const char *stderr_path,
 | 
						|
                   lldb_private::Error &error);
 | 
						|
 | 
						|
    ~ProcessMonitor();
 | 
						|
 | 
						|
    /// Provides the process number of debugee.
 | 
						|
    lldb::pid_t
 | 
						|
    GetPID() const { return m_pid; }
 | 
						|
 | 
						|
    /// Returns the process associated with this ProcessMonitor.
 | 
						|
    ProcessLinux &
 | 
						|
    GetProcess() { return *m_process; }
 | 
						|
 | 
						|
    /// Returns a file descriptor to the controlling terminal of the inferior
 | 
						|
    /// process.
 | 
						|
    ///
 | 
						|
    /// Reads from this file descriptor yield both the standard output and
 | 
						|
    /// standard error of this debugee.  Even if stderr and stdout were
 | 
						|
    /// redirected on launch it may still happen that data is available on this
 | 
						|
    /// descriptor (if the inferior process opens /dev/tty, for example).
 | 
						|
    ///
 | 
						|
    /// If this monitor was attached to an existing process this method returns
 | 
						|
    /// -1.
 | 
						|
    int
 | 
						|
    GetTerminalFD() const { return m_terminal_fd; }
 | 
						|
 | 
						|
    /// Reads @p size bytes from address @vm_adder in the inferior process
 | 
						|
    /// address space.
 | 
						|
    ///
 | 
						|
    /// This method is provided to implement Process::DoReadMemory.
 | 
						|
    size_t
 | 
						|
    ReadMemory(lldb::addr_t vm_addr, void *buf, size_t size,
 | 
						|
               lldb_private::Error &error);
 | 
						|
 | 
						|
    /// Writes @p size bytes from address @p vm_adder in the inferior process
 | 
						|
    /// address space.
 | 
						|
    ///
 | 
						|
    /// This method is provided to implement Process::DoWriteMemory.
 | 
						|
    size_t
 | 
						|
    WriteMemory(lldb::addr_t vm_addr, const void *buf, size_t size,
 | 
						|
                lldb_private::Error &error);
 | 
						|
 | 
						|
    /// Reads the contents from the register identified by the given (architecture
 | 
						|
    /// dependent) offset.
 | 
						|
    ///
 | 
						|
    /// This method is provided for use by RegisterContextLinux derivatives.
 | 
						|
    bool
 | 
						|
    ReadRegisterValue(unsigned offset, lldb_private::RegisterValue &value);
 | 
						|
 | 
						|
    /// Writes the given value to the register identified by the given
 | 
						|
    /// (architecture dependent) offset.
 | 
						|
    ///
 | 
						|
    /// This method is provided for use by RegisterContextLinux derivatives.
 | 
						|
    bool
 | 
						|
    WriteRegisterValue(unsigned offset, const lldb_private::RegisterValue &value);
 | 
						|
 | 
						|
    /// Reads all general purpose registers into the specified buffer.
 | 
						|
    bool
 | 
						|
    ReadGPR(void *buf);
 | 
						|
 | 
						|
    /// Reads all floating point registers into the specified buffer.
 | 
						|
    bool
 | 
						|
    ReadFPR(void *buf);
 | 
						|
 | 
						|
    /// Writes a siginfo_t structure corresponding to the given thread ID to the
 | 
						|
    /// memory region pointed to by @p siginfo.
 | 
						|
    bool
 | 
						|
    GetSignalInfo(lldb::tid_t tid, void *siginfo);
 | 
						|
 | 
						|
    /// Writes the raw event message code (vis-a-vis PTRACE_GETEVENTMSG)
 | 
						|
    /// corresponding to the given thread IDto the memory pointed to by @p
 | 
						|
    /// message.
 | 
						|
    bool
 | 
						|
    GetEventMessage(lldb::tid_t tid, unsigned long *message);
 | 
						|
 | 
						|
    /// Resumes the given thread.  If @p signo is anything but
 | 
						|
    /// LLDB_INVALID_SIGNAL_NUMBER, deliver that signal to the thread.
 | 
						|
    bool
 | 
						|
    Resume(lldb::tid_t tid, uint32_t signo);
 | 
						|
 | 
						|
    /// Single steps the given thread.  If @p signo is anything but
 | 
						|
    /// LLDB_INVALID_SIGNAL_NUMBER, deliver that signal to the thread.
 | 
						|
    bool
 | 
						|
    SingleStep(lldb::tid_t tid, uint32_t signo);
 | 
						|
 | 
						|
    /// Sends the inferior process a PTRACE_KILL signal.  The inferior will
 | 
						|
    /// still exists and can be interrogated.  Once resumed it will exit as
 | 
						|
    /// though it received a SIGKILL.
 | 
						|
    bool
 | 
						|
    BringProcessIntoLimbo();
 | 
						|
 | 
						|
    bool
 | 
						|
    Detach();
 | 
						|
 | 
						|
 | 
						|
private:
 | 
						|
    ProcessLinux *m_process;
 | 
						|
 | 
						|
    lldb::thread_t m_operation_thread;
 | 
						|
    lldb::pid_t m_pid;
 | 
						|
    int m_terminal_fd;
 | 
						|
 | 
						|
    lldb::thread_t m_monitor_thread;
 | 
						|
 | 
						|
    lldb_private::Mutex m_server_mutex;
 | 
						|
    int m_client_fd;
 | 
						|
    int m_server_fd;
 | 
						|
 | 
						|
    /// @class LauchArgs
 | 
						|
    ///
 | 
						|
    /// @brief Simple structure to pass data to the thread responsible for
 | 
						|
    /// launching a child process.
 | 
						|
    struct LaunchArgs
 | 
						|
    {
 | 
						|
        LaunchArgs(ProcessMonitor *monitor,
 | 
						|
                   lldb_private::Module *module,
 | 
						|
                   char const **argv,
 | 
						|
                   char const **envp,
 | 
						|
                   const char *stdin_path,
 | 
						|
                   const char *stdout_path,
 | 
						|
                   const char *stderr_path);
 | 
						|
 | 
						|
        ~LaunchArgs();
 | 
						|
 | 
						|
        ProcessMonitor *m_monitor;      // The monitor performing the launch.
 | 
						|
        lldb_private::Module *m_module; // The executable image to launch.
 | 
						|
        char const **m_argv;            // Process arguments.
 | 
						|
        char const **m_envp;            // Process environment.
 | 
						|
        const char *m_stdin_path;       // Redirect stdin or NULL.
 | 
						|
        const char *m_stdout_path;      // Redirect stdout or NULL.
 | 
						|
        const char *m_stderr_path;      // Redirect stderr or NULL.
 | 
						|
        sem_t m_semaphore;              // Posted to once launch complete.
 | 
						|
        lldb_private::Error m_error;    // Set if process launch failed.
 | 
						|
    };
 | 
						|
 | 
						|
    void
 | 
						|
    StartOperationThread(LaunchArgs *args, lldb_private::Error &error);
 | 
						|
 | 
						|
    void
 | 
						|
    StopOperationThread();
 | 
						|
 | 
						|
    static void *
 | 
						|
    OperationThread(void *arg);
 | 
						|
 | 
						|
    static bool
 | 
						|
    Launch(LaunchArgs *args);
 | 
						|
 | 
						|
    bool
 | 
						|
    EnableIPC();
 | 
						|
 | 
						|
    static void
 | 
						|
    ServeOperation(LaunchArgs *args);
 | 
						|
 | 
						|
    static bool
 | 
						|
    DupDescriptor(const char *path, int fd, int flags);
 | 
						|
 | 
						|
    static bool
 | 
						|
    MonitorCallback(void *callback_baton,
 | 
						|
                    lldb::pid_t pid, int signal, int status);
 | 
						|
 | 
						|
    static ProcessMessage
 | 
						|
    MonitorSIGTRAP(ProcessMonitor *monitor,
 | 
						|
                   const struct siginfo *info, lldb::pid_t pid);
 | 
						|
 | 
						|
    static ProcessMessage
 | 
						|
    MonitorSignal(ProcessMonitor *monitor, 
 | 
						|
                  const struct siginfo *info, lldb::pid_t pid);
 | 
						|
 | 
						|
    static ProcessMessage::CrashReason
 | 
						|
    GetCrashReasonForSIGSEGV(const struct siginfo *info);
 | 
						|
 | 
						|
    static ProcessMessage::CrashReason
 | 
						|
    GetCrashReasonForSIGILL(const struct siginfo *info);
 | 
						|
 | 
						|
    static ProcessMessage::CrashReason
 | 
						|
    GetCrashReasonForSIGFPE(const struct siginfo *info);
 | 
						|
 | 
						|
    static ProcessMessage::CrashReason
 | 
						|
    GetCrashReasonForSIGBUS(const struct siginfo *info);
 | 
						|
 | 
						|
    void
 | 
						|
    DoOperation(Operation *op);
 | 
						|
 | 
						|
    /// Stops the child monitor thread.
 | 
						|
    void
 | 
						|
    StopMonitoringChildProcess();
 | 
						|
 | 
						|
    void 
 | 
						|
    StopMonitor();
 | 
						|
 | 
						|
    void
 | 
						|
    CloseFD(int &fd);
 | 
						|
};
 | 
						|
 | 
						|
#endif // #ifndef liblldb_ProcessMonitor_H_
 |