#ifndef liblldb_NativeProcessLinux_H_
#define liblldb_NativeProcessLinux_H_
#include <csignal>
#include <unordered_set>
#include "lldb/Host/Debug.h"
#include "lldb/Host/HostThread.h"
#include "lldb/Host/linux/Support.h"
#include "lldb/Target/MemoryRegionInfo.h"
#include "lldb/Utility/ArchSpec.h"
#include "lldb/Utility/FileSpec.h"
#include "lldb/lldb-types.h"
#include "llvm/ADT/SmallPtrSet.h"
#include "IntelPTCollector.h"
#include "NativeThreadLinux.h"
#include "Plugins/Process/POSIX/NativeProcessELF.h"
#include "Plugins/Process/Utility/NativeProcessSoftwareSingleStep.h"
namespace lldb_private {
class Status;
class Scalar;
namespace process_linux {
class NativeProcessLinux : public NativeProcessELF,
private NativeProcessSoftwareSingleStep {
#ifdef MS_DEBUGGER
friend class AscendThreadLinux;
#endif
public:
class Manager : public NativeProcessProtocol::Manager {
public:
Manager(MainLoop &mainloop);
llvm::Expected<std::unique_ptr<NativeProcessProtocol>>
Launch(ProcessLaunchInfo &launch_info,
NativeDelegate &native_delegate) override;
llvm::Expected<std::unique_ptr<NativeProcessProtocol>>
Attach(lldb::pid_t pid, NativeDelegate &native_delegate) override;
Extension GetSupportedExtensions() const override;
void AddProcess(NativeProcessLinux &process) {
m_processes.insert(&process);
}
void RemoveProcess(NativeProcessLinux &process) {
m_processes.erase(&process);
}
void CollectThread(::pid_t tid);
private:
MainLoop::SignalHandleUP m_sigchld_handle;
llvm::SmallPtrSet<NativeProcessLinux *, 2> m_processes;
llvm::DenseSet<::pid_t> m_unowned_threads;
void SigchldHandler();
};
~NativeProcessLinux() override { m_manager.RemoveProcess(*this); }
Status Resume(const ResumeActionList &resume_actions) override;
Status Halt() override;
Status Detach() override;
Status Signal(int signo) override;
Status Interrupt() override;
Status Kill() override;
Status GetMemoryRegionInfo(lldb::addr_t load_addr,
MemoryRegionInfo &range_info) override;
Status ReadMemory(lldb::addr_t addr, void *buf, size_t size,
size_t &bytes_read) override;
Status WriteMemory(lldb::addr_t addr, const void *buf, size_t size,
size_t &bytes_written) override;
llvm::Expected<lldb::addr_t> AllocateMemory(size_t size,
uint32_t permissions) override;
llvm::Error DeallocateMemory(lldb::addr_t addr) override;
Status ReadMemoryTags(int32_t type, lldb::addr_t addr, size_t len,
std::vector<uint8_t> &tags) override;
Status WriteMemoryTags(int32_t type, lldb::addr_t addr, size_t len,
const std::vector<uint8_t> &tags) override;
size_t UpdateThreads() override;
const ArchSpec &GetArchitecture() const override { return m_arch; }
Status SetBreakpoint(lldb::addr_t addr, uint32_t size,
bool hardware) override;
Status RemoveBreakpoint(lldb::addr_t addr, bool hardware = false) override;
void DoStopIDBumped(uint32_t newBumpId) override;
Status GetLoadedModuleFileSpec(const char *module_path,
FileSpec &file_spec) override;
Status GetFileLoadAddress(const llvm::StringRef &file_name,
lldb::addr_t &load_addr) override;
NativeThreadLinux *GetThreadByID(lldb::tid_t id);
NativeThreadLinux *GetCurrentThread();
llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>>
GetAuxvData() const override {
return getProcFile(GetID(), "auxv");
}
llvm::Error TraceStart(llvm::StringRef json_request,
llvm::StringRef type) override;
llvm::Error TraceStop(const TraceStopRequest &request) override;
llvm::Expected<llvm::json::Value>
TraceGetState(llvm::StringRef type) override;
llvm::Expected<std::vector<uint8_t>>
TraceGetBinaryData(const TraceGetBinaryDataRequest &request) override;
llvm::Expected<TraceSupportedResponse> TraceSupported() override;
static Status PtraceWrapper(int req, lldb::pid_t pid, void *addr = nullptr,
void *data = nullptr, size_t data_size = 0,
long *result = nullptr);
bool SupportHardwareSingleStepping() const;
Status GetSignalInfo(lldb::tid_t tid, void *siginfo) const;
#ifdef MS_DEBUGGER
Status ReadDeviceRegisterValue(const RegisterInfo *reg_info,
RegisterValue &value) override {
return Status();
}
Status ReadDeviceRegisterList(std::vector<std::string> ®_list) override {
return Status();
}
std::map<lldb::tid_t, lldb::addr_t>& GetThreadsSteppingWithBreakpoint() {
return m_threads_stepping_with_breakpoint;
}
void SetAicOnFocus(const uint32_t &core_id) override {
return;
}
void SetAivOnFocus(const uint32_t &core_id) override {
return;
}
void SetThreadOnFocus(const uint32_t &linear_id) override {
return;
}
void SetSingleCoreRunFlag(bool isSingleCoreRun) override {
return;
}
bool GetSingleCoreRunFlag() override {
return false;
}
Status GetDeviceInfo(DeviceInfo &info) override {
return Status();
}
Status GetCoresInfo(std::vector<CoreInfo> &cores_info) override {
return Status();
}
Status GetWarpsInfo(std::vector<WarpInfo> &warps_info) override {
return Status();
}
Status GetKernelInfo(KernelInfo &info) override {
return Status();
}
void SetClientDeviceId(const int32_t device_id) override {
return;
}
void SetVFStartPC(uint64_t device_id) override { return; }
#endif
protected:
llvm::Expected<llvm::ArrayRef<uint8_t>>
GetSoftwareBreakpointTrapOpcode(size_t size_hint) override;
llvm::Expected<uint64_t> Syscall(llvm::ArrayRef<uint64_t> args);
#ifndef MS_DEBUGGER
private:
#endif
Manager &m_manager;
ArchSpec m_arch;
LazyBool m_supports_mem_region = eLazyBoolCalculate;
std::vector<std::pair<MemoryRegionInfo, FileSpec>> m_mem_region_cache;
lldb::tid_t m_pending_notification_tid = LLDB_INVALID_THREAD_ID;
llvm::DenseMap<lldb::addr_t, lldb::addr_t> m_allocated_memory;
NativeProcessLinux(::pid_t pid, int terminal_fd, NativeDelegate &delegate,
const ArchSpec &arch, Manager &manager,
llvm::ArrayRef<::pid_t> tids);
static llvm::Expected<std::vector<::pid_t>> Attach(::pid_t pid);
static Status SetDefaultPtraceOpts(const lldb::pid_t);
bool TryHandleWaitStatus(lldb::pid_t pid, WaitStatus status);
void MonitorCallback(NativeThreadLinux &thread, WaitStatus status);
void MonitorSIGTRAP(const siginfo_t &info, NativeThreadLinux &thread);
void MonitorTrace(NativeThreadLinux &thread);
#ifdef MS_DEBUGGER
virtual void MonitorBreakpoint(NativeThreadLinux &thread);
#else
void MonitorBreakpoint(NativeThreadLinux &thread);
#endif
void MonitorWatchpoint(NativeThreadLinux &thread, uint32_t wp_index);
void MonitorSignal(const siginfo_t &info, NativeThreadLinux &thread);
bool HasThreadNoLock(lldb::tid_t thread_id);
void StopTrackingThread(NativeThreadLinux &thread);
NativeThreadLinux &AddThread(lldb::tid_t thread_id, bool resume);
Status NotifyTracersOfNewThread(lldb::tid_t tid);
Status NotifyTracersOfThreadDestroyed(lldb::tid_t tid);
void NotifyTracersProcessWillResume() override;
void NotifyTracersProcessDidStop() override;
Status GetEventMessage(lldb::tid_t tid, unsigned long *message);
void NotifyThreadDeath(lldb::tid_t tid);
Status Detach(lldb::tid_t tid);
void StopRunningThreads(lldb::tid_t triggering_tid);
void SignalIfAllThreadsStopped();
Status ResumeThread(NativeThreadLinux &thread, lldb::StateType state,
int signo);
void ThreadWasCreated(NativeThreadLinux &thread);
void SigchldHandler();
Status PopulateMemoryRegionCache();
IntelPTCollector m_intel_pt_collector;
bool MonitorClone(NativeThreadLinux &parent, lldb::pid_t child_pid,
int event);
};
}
}
#endif