29#include "llvm/ADT/StringRef.h"
30#include "llvm/Support/ConvertUTF.h"
31#include "llvm/Support/Errc.h"
32#include "llvm/Support/Error.h"
33#include "llvm/Support/Format.h"
34#include "llvm/Support/Threading.h"
35#include "llvm/Support/raw_ostream.h"
43#pragma warning(disable : 4005)
62 ErrorAsOutParameter EOut(&E);
79 ErrorAsOutParameter EOut(&E);
91 E = createStringError(inconvertibleErrorCode(),
92 "Cannot get process information");
101 llvm::sys::ScopedLock lock(
m_mutex);
105 LLDB_LOG(log,
"process {0} is in state {1}. Resuming...",
112 for (uint32_t i = 0; i <
m_threads.size(); ++i) {
116 if (action ==
nullptr)
119 switch (action->
state) {
126 "Trying to resume thread at index {0}, but failed with "
138 "NativeProcessWindows::%s (): unexpected state %s specified "
139 "for pid %" PRIu64
", tid %" PRIu64,
154 if (active_exception) {
162 LLDB_LOG(log,
"error: process {0} is in state {1}. Returning...",
176 bool caused_stop =
false;
181 if (err.
Fail() || !caused_stop)
200 "error: process {0} in state = {1}, but "
201 "cannot detach it in this state.",
211 "Windows does not support sending signals to processes");
232 size_t size,
size_t &bytes_read) {
237 size_t size,
size_t &bytes_written) {
241llvm::Expected<lldb::addr_t>
272 std::string description) {
276 stop_info.
reason = reason;
285 thread.SetStopReason(stop_info, description);
290 std::string description) {
296 for (uint32_t i = 0; i <
m_threads.size(); ++i) {
299 LLDB_LOG(log,
"failed to stop thread {0}: {1}", t->GetID(),
error);
306llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>>
309 return llvm::errc::not_supported;
312llvm::Expected<llvm::ArrayRef<uint8_t>>
314 static const uint8_t g_aarch64_opcode[] = {0x00, 0x00, 0x3e,
316 static const uint8_t g_thumb_opcode[] = {0xfe, 0xde};
319 case llvm::Triple::aarch64:
320 return llvm::ArrayRef(g_aarch64_opcode);
322 case llvm::Triple::arm:
323 case llvm::Triple::thumb:
324 return llvm::ArrayRef(g_thumb_opcode);
361 AutoHandle snapshot(CreateToolhelp32Snapshot(TH32CS_SNAPMODULE,
GetID()));
364 me.dwSize =
sizeof(MODULEENTRY32W);
365 if (Module32FirstW(snapshot.
get(), &me)) {
368 if (!llvm::convertWideToUTF8(me.szExePath, path))
374 }
while (Module32Next(snapshot.
get(), &me));
391 FileSpec module_file_spec(module_path);
394 if (it.first == module_file_spec) {
395 file_spec = it.first;
400 "Module (%s) not found in process %" PRIu64
"!",
415 if (it.first == file_spec) {
416 load_addr = it.second;
421 "Can't get loaded address of file (%s) in process %" PRIu64
"!",
425llvm::Expected<std::vector<LoadedLibraryInfo>>
428 return error.ToError();
430 std::vector<LoadedLibraryInfo> libs;
434 info.
name = file_spec.GetPath();
436 libs.push_back(std::move(info));
447 LLDB_LOG(log,
"Process {0} exited with code {1}",
GetID(), exit_code);
465 LLDB_LOG(log,
"Debugger connected to process {0}. Image base = {1:x}",
476 LLDB_LOG(log,
"Cannot get process information during debugger connecting "
485 m_threads.push_back(std::make_unique<NativeThreadWindows>(
500 "received error while checking for watchpoint hits, pid = "
502 thread->GetID(),
error);
507 formatv(
"{0} {1} {2}", wp_addr, wp_id, wp_hit_addr).str();
526 auto ®_ctx = stop_thread->GetRegisterContext();
529 LLDB_LOG(log,
"Hit non-loader breakpoint at address {0:x}.",
546 if (args.size() >= 2) {
548 Status error = reg_ctx.GetWatchpointHitIndex(hw_id, args[1]);
551 "received error while checking for watchpoint hits, pid = "
557 formatv(
"{0} {1} {2}", reg_ctx.GetWatchpointAddress(hw_id), hw_id,
570 "Hit loader breakpoint at address {0:x}, setting initial stop "
593 "DebugBreakProcess injection treated as Halt SIGSTOP for tid "
599 signal_info.
signo = 19;
602 for (uint32_t i = 0; i <
m_threads.size(); ++i) {
605 LLDB_LOG(log,
"Failed to stop thread {1:x}: {0}", t->GetID(),
612 injected->SetStopReason(signal_info,
"interrupt");
617 std::string desc = formatv(
"Exception {0:x8} encountered at address {1:x8}",
630 "Debugger thread reported exception {0:x} at address {1:x} "
631 "(first_chance={2})",
639 llvm::raw_string_ostream desc_stream(desc);
641 <<
" encountered at address "
643 record.
Dump(desc_stream);
654 llvm::sys::ScopedLock lock(
m_mutex);
660 case DWORD(STATUS_SINGLE_STEP):
661 case STATUS_WX86_SINGLE_STEP:
663 case DWORD(STATUS_BREAKPOINT):
672 llvm::sys::ScopedLock lock(
m_mutex);
674 auto thread = std::make_unique<NativeThreadWindows>(*
this, new_thread);
675 thread->GetRegisterContext().ClearAllHardwareWatchpoints();
686 LLDB_LOG(log,
"failed to suspend newly-created thread {0}: {1}",
687 thread->GetID(),
error);
691 thread->SetStopReason(stop_info,
"");
698 uint32_t exit_code) {
700 llvm::erase_if(
m_threads, [thread_id](
const auto &t) {
701 return t->GetID() == thread_id;
718 uint16_t length_lower_word) {
721 llvm::SmallVector<char, 256> buffer;
723 debug_string_addr, is_unicode, length_lower_word, buffer)) {
724 std::string err_str = llvm::toString(std::move(err));
726 llvm::formatv(
"Failed to read debug string at {0:x} "
727 "(size & 0xffff={1}, unicode={2}): {3}\n",
728 debug_string_addr, length_lower_word, is_unicode, err_str)
731 m_delegate.NewProcessOutput(
this, llvm::StringRef(msg));
738 assert(buffer.size() % 2 == 0);
739 llvm::ArrayRef<unsigned short> utf16(
740 reinterpret_cast<const unsigned short *
>(buffer.data()),
743 if (!llvm::convertUTF16ToUTF8String(utf16, out)) {
744 LLDB_LOG(log,
"Debug string is not valid Utf 16");
747 m_delegate.NewProcessOutput(
this, llvm::StringRef(out.data(), out.size()));
750 llvm::StringRef(buffer.data(), buffer.size()));
754llvm::Expected<std::unique_ptr<NativeProcessProtocol>>
758 Error E = Error::success();
759 auto process_up = std::unique_ptr<NativeProcessWindows>(
763 return std::move(process_up);
766llvm::Expected<std::unique_ptr<NativeProcessProtocol>>
769 Error E = Error::success();
771 auto process_up = std::unique_ptr<NativeProcessWindows>(
775 return std::move(process_up);
785 std::make_unique<ConnectionConPTY>(
m_pty));
813 self->m_delegate.NewProcessOutput(
814 self, llvm::StringRef(
static_cast<const char *
>(src), src_len));
821 "no ConPTY connection on this NativeProcessWindows");
828 "ConPTY stdin write returned status {0}",
static_cast<int>(status));
static llvm::raw_ostream & error(Stream &strm)
#define STATUS_WX86_BREAKPOINT
#define LLDB_LOG(log,...)
The LLDB_LOG* macros defined below are the way to emit log messages.
lldb::tid_t GetThreadID() const
void Dump(llvm::raw_ostream &stream) const
unsigned long GetExceptionValue() const
llvm::ArrayRef< uint64_t > GetExceptionArguments() const
lldb::addr_t GetExceptionAddress() const
size_t GetPath(char *path, size_t max_path_length, bool denormalize=true) const
Extract the full path to the file.
static FileSystem & Instance()
void Resolve(llvm::SmallVectorImpl< char > &path, bool force_make_absolute=false)
Resolve path to make it canonical.
static bool GetProcessInfo(lldb::pid_t pid, ProcessInstanceInfo &proc_info)
NativeProcessProtocol(lldb::pid_t pid, int terminal_fd, NativeDelegate &delegate)
lldb::pid_t GetID() const
Status SetSoftwareBreakpoint(lldb::addr_t addr, uint32_t size_hint)
virtual const NativeWatchpointList::WatchpointMap & GetWatchpointMap() const
lldb::StateType GetState() const
void SetState(lldb::StateType state, bool notify_delegates=true)
NativeThreadProtocol * GetThreadByID(lldb::tid_t tid)
void SetCurrentThreadID(lldb::tid_t tid)
std::recursive_mutex m_threads_mutex
std::vector< std::unique_ptr< NativeThreadProtocol > > m_threads
virtual bool SetExitStatus(WaitStatus status, bool bNotifyStateChange)
void SetID(lldb::pid_t pid)
NativeDelegate & m_delegate
Status RemoveSoftwareBreakpoint(lldb::addr_t addr)
virtual Status SetHardwareBreakpoint(lldb::addr_t addr, size_t size)
virtual llvm::Expected< llvm::ArrayRef< uint8_t > > GetSoftwareBreakpointTrapOpcode(size_t size_hint)
virtual Status RemoveHardwareBreakpoint(lldb::addr_t addr)
std::unordered_map< lldb::addr_t, SoftwareBreakpoint > m_software_breakpoints
llvm::Expected< std::unique_ptr< NativeProcessProtocol > > Launch(ProcessLaunchInfo &launch_info, NativeDelegate &native_delegate) override
Launch a process for debugging.
llvm::Expected< std::unique_ptr< NativeProcessProtocol > > Attach(lldb::pid_t pid, NativeDelegate &native_delegate) override
Attach to an existing process.
Status GetMemoryRegionInfo(lldb::addr_t load_addr, MemoryRegionInfo &range_info) override
ExceptionResult HandleBreakpointException(const ExceptionRecord &record)
llvm::Error DeallocateMemory(lldb::addr_t addr) override
Status Resume(const ResumeActionList &resume_actions) override
void OnCreateThread(const HostThread &thread) override
void OnExitProcess(uint32_t exit_code) override
Status GetLoadedModuleFileSpec(const char *module_path, FileSpec &file_spec) override
void StartStdioForwarding()
Wire up m_stdio_communication on m_pty's STDOUT HANDLE.
static void STDIOReadThreadBytesReceived(void *baton, const void *src, size_t src_len)
Bridge between m_stdio_communication's read thread and NativeDelegate::NewProcessOutput.
llvm::Expected< llvm::ArrayRef< uint8_t > > GetSoftwareBreakpointTrapOpcode(size_t size_hint) override
void OnDebuggerConnected(lldb::addr_t image_base) override
Status ReadMemory(lldb::addr_t addr, void *buf, size_t size, size_t &bytes_read) override
~NativeProcessWindows() override
NativeProcessWindows(ProcessLaunchInfo &launch_info, NativeDelegate &delegate, llvm::Error &E)
bool IsAlive() const override
Status SetBreakpoint(lldb::addr_t addr, uint32_t size, bool hardware) override
Status CacheLoadedModules()
void StopStdioForwarding()
Tear down the read thread and disconnect m_stdio_communication.
size_t WriteStdin(const void *buf, size_t len, Status &error) override
Forward bytes from the gdb-remote I packet into the inferior's ConPTY-backed stdin via m_stdio_commun...
bool m_pending_halt
Set when Halt() / Interrupt() schedules a DebugBreakProcess injection.
ExceptionResult HandleGenericException(bool first_chance, const ExceptionRecord &record)
void SetArchitecture(const ArchSpec &arch_spec)
ExceptionResult OnDebugException(bool first_chance, const ExceptionRecord &record) override
void OnDebugString(lldb::addr_t debug_string_addr, bool is_unicode, uint16_t length_lower_word) override
size_t GetSoftwareBreakpointPCOffset() override
Return the offset of the PC relative to the software breakpoint that was hit.
std::shared_ptr< PseudoConsole > m_pty
PseudoConsole for the lldb-server stdio-forwarding path.
ThreadedCommunication m_stdio_communication
Wraps a ConnectionConPTY around the PTY's parent-side STDOUT HANDLE.
llvm::ErrorOr< std::unique_ptr< llvm::MemoryBuffer > > GetAuxvData() const override
void OnUnloadDll(lldb::addr_t module_addr) override
Status GetFileLoadAddress(const llvm::StringRef &file_name, lldb::addr_t &load_addr) override
NativeThreadWindows * GetThreadByID(lldb::tid_t thread_id)
Status WriteMemory(lldb::addr_t addr, const void *buf, size_t size, size_t &bytes_written) override
void OnExitThread(lldb::tid_t thread_id, uint32_t exit_code) override
llvm::Expected< lldb::addr_t > AllocateMemory(size_t size, uint32_t permissions) override
Status Signal(int signo) override
Sends a process a UNIX signal signal.
Status RemoveBreakpoint(lldb::addr_t addr, bool hardware=false) override
const ArchSpec & GetArchitecture() const override
void SetStopReasonForThread(NativeThreadWindows &thread, lldb::StopReason reason, std::string description="")
Status Interrupt() override
Tells a process to interrupt all operations as if by a Ctrl-C.
void OnLoadDll(const ModuleSpec &module_spec, lldb::addr_t module_addr) override
bool FindSoftwareBreakpoint(lldb::addr_t addr)
void StopThread(lldb::tid_t thread_id, lldb::StopReason reason, std::string description="")
size_t UpdateThreads() override
ExceptionResult HandleSingleStepException(const ExceptionRecord &record)
bool m_initial_stop_seen
Whether we've seen the loader breakpoint that fires once per process at launch / attach.
bool HasPendingLibraryEvents() override
bool m_pending_library_events
Set whenever an OS DLL load/unload event has been seen since the last stop reply.
llvm::Expected< std::vector< LoadedLibraryInfo > > GetLoadedLibraries() override
Return the currently loaded libraries of the target in the qXfer:libraries:read form (generic name + ...
lldb::addr_t GetSharedLibraryInfoAddress() override
Status IgnoreSignals(llvm::ArrayRef< int > signals) override
std::map< lldb_private::FileSpec, lldb::addr_t > m_loaded_modules
virtual Status GetWatchpointHitIndex(uint32_t &wp_index, lldb::addr_t trap_addr)
virtual lldb::addr_t GetWatchpointAddress(uint32_t wp_index)
virtual lldb::addr_t GetWatchpointHitAddress(uint32_t wp_index)
Status WriteMemory(lldb::addr_t addr, const void *buf, size_t size, size_t &bytes_written)
Status DestroyProcess(lldb::StateType process_state)
Status LaunchProcess(ProcessLaunchInfo &launch_info, DebugDelegateSP delegate)
Status GetMemoryRegionInfo(lldb::addr_t load_addr, MemoryRegionInfo &range_info)
std::unique_ptr< ProcessWindowsData > m_session_data
Status AllocateMemory(size_t size, uint32_t permissions, lldb::addr_t &addr)
virtual ExceptionResult OnDebugException(bool first_chance, const ExceptionRecord &record)
Status AttachProcess(lldb::pid_t pid, const ProcessAttachInfo &attach_info, DebugDelegateSP delegate)
lldb::pid_t GetDebuggedProcessId() const
Status ReadMemory(lldb::addr_t addr, void *buf, size_t size, size_t &bytes_read)
virtual void OnExitProcess(uint32_t exit_code)
llvm::Error ReadDebugString(lldb::addr_t debug_string_addr, bool is_unicode, uint16_t length_lower_word, llvm::SmallVectorImpl< char > &output)
Read an OUTPUT_DEBUG_STRING_INFO payload from the inferior.
Status HaltProcess(bool &caused_stop)
Status DeallocateMemory(lldb::addr_t addr)
void SetProcessID(lldb::pid_t pid)
ArchSpec & GetArchitecture()
std::shared_ptr< PTY > TakePTY()
A pseudo terminal helper class.
const ResumeAction * GetActionForThread(lldb::tid_t tid, bool default_ok) const
llvm::Error ToError() const
FIXME: Replace all uses with takeError() instead.
static Status FromErrorStringWithFormat(const char *format,...) __attribute__((format(printf
static Status FromErrorString(const char *str)
bool Fail() const
Test for error condition.
static Status static Status FromErrorStringWithFormatv(const char *format, Args &&...args)
bool Success() const
Test for success condition.
#define LLDB_INVALID_INDEX32
#define LLDB_INVALID_ADDRESS
#define LLDB_INVALID_PROCESS_ID
A class that represents a running process on the host machine.
Log * GetLog(Cat mask)
Retrieve the Log object for the channel associated with the given log enum.
std::shared_ptr< IDebugDelegate > DebugDelegateSP
std::shared_ptr< ExceptionRecord > ExceptionRecordSP
const char * StateAsCString(lldb::StateType state)
Converts a StateType to a C string.
ConnectionStatus
Connection Status Types.
@ eConnectionStatusSuccess
Success.
StateType
Process and Thread States.
@ eStateUnloaded
Process is object is valid, but not currently loaded.
@ eStateDetached
Process has been detached and can't be examined.
@ eStateStopped
Process or thread is stopped and can be examined.
@ eStateSuspended
Process or thread is in a suspended state as far as the debugger is concerned while other processes o...
@ eStateRunning
Process or thread is running and can't be examined.
@ eStateExited
Process has exited and can't be examined.
@ eStateStepping
Process or thread is in the process of stepping and can not be examined.
@ eStateCrashed
Process or thread has crashed and can be examined.
@ eErrorTypeWin32
Standard Win32 error codes.
StopReason
Thread stop reasons.
Generic loaded-library entry used by the non-SVR4 qXfer:libraries:read form of the GDB remote library...
struct lldb_private::ThreadStopInfo::@116236113001137253323017204263037302160273237376::@034237007264067231263360140073224264215170222231 exception
union lldb_private::ThreadStopInfo::@116236113001137253323017204263037302160273237376 details