//===-- ProcessGDBRemote.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_ProcessGDBRemote_h_ #define liblldb_ProcessGDBRemote_h_ // C Includes // C++ Includes #include #include // Other libraries and framework includes #include "lldb/Core/ArchSpec.h" #include "lldb/Core/Broadcaster.h" #include "lldb/Core/ConstString.h" #include "lldb/Core/Error.h" #include "lldb/Core/StreamString.h" #include "lldb/Core/StringList.h" #include "lldb/Core/ThreadSafeValue.h" #include "lldb/Target/Process.h" #include "lldb/Target/Thread.h" #include "GDBRemoteCommunicationClient.h" #include "Utility/StringExtractor.h" #include "GDBRemoteRegisterContext.h" class ThreadGDBRemote; class ProcessGDBRemote : public lldb_private::Process { public: //------------------------------------------------------------------ // Constructors and Destructors //------------------------------------------------------------------ static lldb::ProcessSP CreateInstance (lldb_private::Target& target, lldb_private::Listener &listener, const lldb_private::FileSpec *crash_file_path); static void Initialize(); static void DebuggerInitialize (lldb_private::Debugger &debugger); static void Terminate(); static lldb_private::ConstString GetPluginNameStatic(); static const char * GetPluginDescriptionStatic(); //------------------------------------------------------------------ // Constructors and Destructors //------------------------------------------------------------------ ProcessGDBRemote(lldb_private::Target& target, lldb_private::Listener &listener); virtual ~ProcessGDBRemote(); //------------------------------------------------------------------ // Check if a given Process //------------------------------------------------------------------ virtual bool CanDebug (lldb_private::Target &target, bool plugin_specified_by_name); virtual lldb_private::CommandObject * GetPluginCommandObject(); //------------------------------------------------------------------ // Creating a new process, or attaching to an existing one //------------------------------------------------------------------ virtual lldb_private::Error WillLaunch (lldb_private::Module* module); virtual lldb_private::Error DoLaunch (lldb_private::Module *exe_module, lldb_private::ProcessLaunchInfo &launch_info); virtual void DidLaunch (); virtual lldb_private::Error WillAttachToProcessWithID (lldb::pid_t pid); virtual lldb_private::Error WillAttachToProcessWithName (const char *process_name, bool wait_for_launch); virtual lldb_private::Error DoConnectRemote (lldb_private::Stream *strm, const char *remote_url); lldb_private::Error WillLaunchOrAttach (); virtual lldb_private::Error DoAttachToProcessWithID (lldb::pid_t pid); virtual lldb_private::Error DoAttachToProcessWithID (lldb::pid_t pid, const lldb_private::ProcessAttachInfo &attach_info); virtual lldb_private::Error DoAttachToProcessWithName (const char *process_name, const lldb_private::ProcessAttachInfo &attach_info); virtual void DidAttach (); //------------------------------------------------------------------ // PluginInterface protocol //------------------------------------------------------------------ virtual lldb_private::ConstString GetPluginName(); virtual uint32_t GetPluginVersion(); //------------------------------------------------------------------ // Process Control //------------------------------------------------------------------ virtual lldb_private::Error WillResume (); virtual lldb_private::Error DoResume (); virtual lldb_private::Error DoHalt (bool &caused_stop); virtual lldb_private::Error DoDetach (bool keep_stopped); virtual bool DetachRequiresHalt() { return true; } virtual lldb_private::Error DoSignal (int signal); virtual lldb_private::Error DoDestroy (); virtual void RefreshStateAfterStop(); //------------------------------------------------------------------ // Process Queries //------------------------------------------------------------------ virtual bool IsAlive (); virtual lldb::addr_t GetImageInfoAddress(); //------------------------------------------------------------------ // Process Memory //------------------------------------------------------------------ virtual size_t DoReadMemory (lldb::addr_t addr, void *buf, size_t size, lldb_private::Error &error); virtual size_t DoWriteMemory (lldb::addr_t addr, const void *buf, size_t size, lldb_private::Error &error); virtual lldb::addr_t DoAllocateMemory (size_t size, uint32_t permissions, lldb_private::Error &error); virtual lldb_private::Error GetMemoryRegionInfo (lldb::addr_t load_addr, lldb_private::MemoryRegionInfo ®ion_info); virtual lldb_private::Error DoDeallocateMemory (lldb::addr_t ptr); //------------------------------------------------------------------ // Process STDIO //------------------------------------------------------------------ virtual size_t PutSTDIN (const char *buf, size_t buf_size, lldb_private::Error &error); //---------------------------------------------------------------------- // Process Breakpoints //---------------------------------------------------------------------- virtual lldb_private::Error EnableBreakpointSite (lldb_private::BreakpointSite *bp_site); virtual lldb_private::Error DisableBreakpointSite (lldb_private::BreakpointSite *bp_site); //---------------------------------------------------------------------- // Process Watchpoints //---------------------------------------------------------------------- virtual lldb_private::Error EnableWatchpoint (lldb_private::Watchpoint *wp, bool notify = true); virtual lldb_private::Error DisableWatchpoint (lldb_private::Watchpoint *wp, bool notify = true); virtual lldb_private::Error GetWatchpointSupportInfo (uint32_t &num); virtual lldb_private::Error GetWatchpointSupportInfo (uint32_t &num, bool& after); virtual bool StartNoticingNewThreads(); virtual bool StopNoticingNewThreads(); GDBRemoteCommunicationClient & GetGDBRemote() { return m_gdb_comm; } //---------------------------------------------------------------------- // Override SetExitStatus so we can disconnect from the remote GDB server //---------------------------------------------------------------------- virtual bool SetExitStatus (int exit_status, const char *cstr); protected: friend class ThreadGDBRemote; friend class GDBRemoteCommunicationClient; friend class GDBRemoteRegisterContext; //---------------------------------------------------------------------- // Accessors //---------------------------------------------------------------------- bool IsRunning ( lldb::StateType state ) { return state == lldb::eStateRunning || IsStepping(state); } bool IsStepping ( lldb::StateType state) { return state == lldb::eStateStepping; } bool CanResume ( lldb::StateType state) { return state == lldb::eStateStopped; } bool HasExited (lldb::StateType state) { return state == lldb::eStateExited; } bool ProcessIDIsValid ( ) const; void Clear ( ); lldb_private::Flags & GetFlags () { return m_flags; } const lldb_private::Flags & GetFlags () const { return m_flags; } virtual bool UpdateThreadList (lldb_private::ThreadList &old_thread_list, lldb_private::ThreadList &new_thread_list); lldb_private::Error LaunchAndConnectToDebugserver (const lldb_private::ProcessInfo &process_info); void KillDebugserverProcess (); void BuildDynamicRegisterInfo (bool force); void SetLastStopPacket (const StringExtractorGDBRemote &response); bool ParsePythonTargetDefinition(const lldb_private::FileSpec &target_definition_fspec); bool ParseRegisters(lldb_private::ScriptInterpreterObject *registers_array); //------------------------------------------------------------------ /// Broadcaster event bits definitions. //------------------------------------------------------------------ enum { eBroadcastBitAsyncContinue = (1 << 0), eBroadcastBitAsyncThreadShouldExit = (1 << 1), eBroadcastBitAsyncThreadDidExit = (1 << 2) }; typedef enum AsyncThreadState { eAsyncThreadNotStarted, eAsyncThreadRunning, eAsyncThreadDone } AsyncThreadState; lldb_private::Flags m_flags; // Process specific flags (see eFlags enums) GDBRemoteCommunicationClient m_gdb_comm; lldb::pid_t m_debugserver_pid; StringExtractorGDBRemote m_last_stop_packet; lldb_private::Mutex m_last_stop_packet_mutex; GDBRemoteDynamicRegisterInfo m_register_info; lldb_private::Broadcaster m_async_broadcaster; lldb::thread_t m_async_thread; AsyncThreadState m_async_thread_state; lldb_private::Mutex m_async_thread_state_mutex; typedef std::vector tid_collection; typedef std::vector< std::pair > tid_sig_collection; typedef std::map MMapMap; tid_collection m_thread_ids; // Thread IDs for all threads. This list gets updated after stopping tid_collection m_continue_c_tids; // 'c' for continue tid_sig_collection m_continue_C_tids; // 'C' for continue with signal tid_collection m_continue_s_tids; // 's' for step tid_sig_collection m_continue_S_tids; // 'S' for step with signal size_t m_max_memory_size; // The maximum number of bytes to read/write when reading and writing memory MMapMap m_addr_to_mmap_size; lldb::BreakpointSP m_thread_create_bp_sp; bool m_waiting_for_attach; bool m_destroy_tried_resuming; lldb::CommandObjectSP m_command_sp; int64_t m_breakpoint_pc_offset; bool StartAsyncThread (); void StopAsyncThread (); static lldb::thread_result_t AsyncThread (void *arg); static bool MonitorDebugserverProcess (void *callback_baton, lldb::pid_t pid, bool exited, int signo, int exit_status); lldb::StateType SetThreadStopInfo (StringExtractor& stop_packet); void ClearThreadIDList (); bool UpdateThreadIDList (); void DidLaunchOrAttach (); lldb_private::Error ConnectToDebugserver (const char *host_port); const char * GetDispatchQueueNameForThread (lldb::addr_t thread_dispatch_qaddr, std::string &dispatch_queue_name); lldb_private::DynamicLoader * GetDynamicLoader (); private: //------------------------------------------------------------------ // For ProcessGDBRemote only //------------------------------------------------------------------ static bool NewThreadNotifyBreakpointHit (void *baton, lldb_private::StoppointCallbackContext *context, lldb::user_id_t break_id, lldb::user_id_t break_loc_id); DISALLOW_COPY_AND_ASSIGN (ProcessGDBRemote); }; #endif // liblldb_ProcessGDBRemote_h_