11#include "lldb/Host/Config.h"
45#include "llvm/Support/ErrorExtras.h"
46#include "llvm/Support/ErrorHandling.h"
47#include "llvm/Support/JSON.h"
48#include "llvm/Support/ScopedPrinter.h"
49#include "llvm/TargetParser/Triple.h"
63enum GDBRemoteServerError {
66 eErrorNoProcess = eErrorFirst,
238 bool &interrupt,
bool &quit) {
274 "%s: no process command line specified to launch", __FUNCTION__);
276 const bool should_forward_stdio =
283 if (should_forward_stdio) {
296 "process but one already exists");
306 SetEnabledExtensions(*m_current_process);
313 if (should_forward_stdio) {
318 "pid = {0}: setting up stdout/stderr redirection via $O "
319 "gdb-remote commands",
320 m_current_process->GetID());
323 auto terminal_fd = m_current_process->GetTerminalFileDescriptor();
324 if (terminal_fd >= 0) {
326 "ProcessGDBRemoteCommunicationServerLLGS::%s setting "
327 "inferior STDIO fd to %d",
328 __FUNCTION__, terminal_fd);
329 Status status = SetSTDIOFileDescriptor(terminal_fd);
334 "ProcessGDBRemoteCommunicationServerLLGS::%s ignoring "
335 "inferior STDIO since terminal fd reported as %d",
336 __FUNCTION__, terminal_fd);
340 "pid = {0} skipping stdout/stderr redirection via $O: inferior "
341 "will communicate over client-provided file descriptors",
342 m_current_process->GetID());
345 printf(
"Launched '%s' as process %" PRIu64
"...\n",
346 m_process_launch_info.GetArguments().GetArgumentAtIndex(0),
347 m_current_process->GetID());
354 LLDB_LOGF(log,
"GDBRemoteCommunicationServerLLGS::%s pid %" PRIu64,
361 "cannot attach to process %" PRIu64
362 " when another process with pid %" PRIu64
" is being debugged.",
369 llvm::errs() << llvm::formatv(
"failed to attach to process {0}: {1}\n", pid,
377 SetEnabledExtensions(*m_current_process);
380 auto terminal_fd = m_current_process->GetTerminalFileDescriptor();
381 if (terminal_fd >= 0) {
383 "ProcessGDBRemoteCommunicationServerLLGS::%s setting "
384 "inferior STDIO fd to %d",
385 __FUNCTION__, terminal_fd);
386 Status status = SetSTDIOFileDescriptor(terminal_fd);
391 "ProcessGDBRemoteCommunicationServerLLGS::%s ignoring "
392 "inferior STDIO since terminal fd reported as %d",
393 __FUNCTION__, terminal_fd);
396 printf(
"Attached to process %" PRIu64
"...\n", pid);
401 llvm::StringRef process_name,
bool include_existing) {
404 std::chrono::milliseconds polling_interval = std::chrono::milliseconds(1);
410 process_name, llvm::sys::path::Style::native);
413 if (include_existing) {
414 LLDB_LOG(log,
"including existing processes in search");
418 LLDB_LOG(log,
"placed '{0}' processes in the exclusion list.",
419 exclusion_list.size());
422 LLDB_LOG(log,
"waiting for '{0}' to appear", process_name);
424 auto is_in_exclusion_list =
426 for (
auto &excluded : exclusion_list) {
427 if (excluded.GetProcessID() == info.GetProcessID())
435 loop_process_list.clear();
438 llvm::erase_if(loop_process_list, is_in_exclusion_list);
441 if (loop_process_list.size() == 1) {
442 auto matching_process_pid = loop_process_list[0].GetProcessID();
443 LLDB_LOG(log,
"found pid {0}", matching_process_pid);
448 if (loop_process_list.size() > 1) {
451 "Multiple executables with name: '{0}' found. Pids: ",
453 for (
size_t i = 0; i < loop_process_list.size() - 1; ++i) {
454 error_stream.
Format(
"{0}, ", loop_process_list[i].GetProcessID());
456 error_stream.
Format(
"{0}.", loop_process_list.back().GetProcessID());
464 LLDB_LOG(log,
"sleep {0} seconds", polling_interval);
465 std::this_thread::sleep_for(polling_interval);
471 assert(process &&
"process cannot be NULL");
474 "GDBRemoteCommunicationServerLLGS::%s called with "
475 "NativeProcessProtocol pid %" PRIu64
", current state: %s",
476 __FUNCTION__, process->
GetID(),
483 assert(process &&
"process cannot be NULL");
489 LLDB_LOG(log,
"pid = {0}, failed to retrieve process exit status",
494 response.
PutHex8(GDBRemoteServerError::eErrorExitStatus);
498 LLDB_LOG(log,
"pid = {0}, returning exit type {1}", process->
GetID(),
507 response.
Format(
"{0:g}", *wait_status);
510 response.
Format(
";process:{0:x-}", process->
GetID());
518 uint32_t buf_size,
bool swap) {
521 for (i = buf_size - 1; i >= 0; i--)
524 for (i = 0; i < buf_size; i++)
545 switch (reg_info.
format) {
555 return "bytes-with-ascii";
559 return "char-printable";
571 return "hex-uppercase";
587 return "vector-char";
589 return "vector-sint64";
591 return "vector-float16";
593 return "vector-float64";
595 return "vector-sint8";
597 return "vector-uint8";
599 return "vector-sint16";
601 return "vector-uint16";
603 return "vector-sint32";
605 return "vector-uint32";
607 return "vector-float32";
609 return "vector-uint64";
611 return "vector-uint128";
613 return "complex-integer";
617 return "address-info";
621 return "instruction";
629 llvm_unreachable(
"Unknown register format");
674 response.
Printf(
"%" PRIx32, *reg_num);
676 response.
Printf(
"%" PRIu32, *reg_num);
688 reg_value_p = ®_value;
699 std::vector<uint8_t> zeros(reg_info.
byte_size,
'\0');
705static std::optional<json::Object>
711 json::Object register_object;
713#ifdef LLDB_JTHREADSINFO_FULL_REGISTER_SET
714 const auto expedited_regs =
717 const auto expedited_regs =
720 if (expedited_regs.empty())
723 for (
auto ®_num : expedited_regs) {
726 if (reg_info_p ==
nullptr) {
728 "%s failed to get register info for register index %" PRIu32,
729 __FUNCTION__, reg_num);
740 LLDB_LOGF(log,
"%s failed to read register '%s' index %" PRIu32
": %s",
742 reg_info_p->
name ? reg_info_p->
name :
"<unnamed-register>",
743 reg_num,
error.AsCString());
751 register_object.try_emplace(llvm::to_string(reg_num),
755 return register_object;
759 switch (stop_reason) {
773 return "processor trace";
781 return "async interrupt";
793static llvm::Expected<json::Array>
797 json::Array threads_array;
804 std::string description;
805 if (!thread.GetStopReason(tid_stop_info, description))
806 return llvm::createStringError(
"failed to get stop reason");
808 const int signum = tid_stop_info.
signo;
810 "GDBRemoteCommunicationServerLLGS::%s pid %" PRIu64
812 " got signal signo = %d, reason = %d, exc_type = %" PRIu64,
813 __FUNCTION__, process.
GetID(), tid, signum, tid_stop_info.
reason,
816 json::Object thread_obj;
820 thread_obj.try_emplace(
"registers", std::move(*registers));
823 thread_obj.try_emplace(
"tid",
static_cast<int64_t
>(tid));
826 thread_obj.try_emplace(
"signal", signum);
828 const std::string thread_name = thread.GetName();
829 if (!thread_name.empty())
830 thread_obj.try_emplace(
"name", thread_name);
834 thread_obj.try_emplace(
"reason", stop_reason);
836 if (!description.empty())
837 thread_obj.try_emplace(
"description", description);
841 thread_obj.try_emplace(
844 json::Array medata_array;
847 medata_array.push_back(
850 thread_obj.try_emplace(
"medata", std::move(medata_array));
852 threads_array.push_back(std::move(thread_obj));
854 return threads_array;
864 LLDB_LOG(log,
"preparing packet for pid {0} tid {1}", process.
GetID(),
870 std::string description;
871 if (!thread.GetStopReason(tid_stop_info, description))
882 int signum = tid_stop_info.
signo;
885 "pid {0}, tid {1}, got signal signo = {2}, reason = {3}, exc_type = {4}",
886 process.
GetID(), thread.GetID(), signum,
int(tid_stop_info.
reason),
890 response.
PutHex8(signum & 0xff);
898 const std::string thread_name = thread.GetName();
899 if (!thread_name.empty()) {
900 size_t thread_name_len = thread_name.length();
902 if (::strcspn(thread_name.c_str(),
"$#+-;:") == thread_name_len) {
924 uint32_t thread_num = 0;
928 response.
Printf(
"%" PRIx64, listed_thread.GetID());
938 if (thread_num > 1) {
939 const bool threads_with_valid_stop_info_only =
true;
945 unescaped_response.
AsRawOstream() << std::move(*threads_info);
950 "failed to prepare a jstopinfo field for pid {1}: {0}",
956 char delimiter =
':';
968 LLDB_LOGF(log,
"%s failed to read register '%s' index %" PRIu32
": %s",
970 reg_info_p->
name ? reg_info_p->
name :
"<unnamed-register>",
971 reg_to_read,
error.AsCString());
990 const auto expedited_regs =
993 for (
auto ®_num : expedited_regs) {
997 if (reg_info_p !=
nullptr && reg_info_p->
value_regs ==
nullptr) {
1000 if (
error.Success()) {
1001 response.
Printf(
"%.02x:", reg_num);
1007 "GDBRemoteCommunicationServerLLGS::%s failed to read "
1008 "register '%s' index %" PRIu32
": %s",
1010 reg_info_p->
name ? reg_info_p->
name :
"<unnamed-register>",
1011 reg_num,
error.AsCString());
1017 if (reason_str !=
nullptr) {
1018 response.
Printf(
"reason:%s;", reason_str);
1021 if (!description.empty()) {
1052 response.
Printf(
"%s:p%" PRIx64
".%" PRIx64
";", reason_str,
1069 if (response.
Empty())
1089 if (listed_thread.GetID() != thread_to_skip) {
1091 if (!stop_reply.
Empty())
1099 assert(process &&
"process cannot be NULL");
1102 LLDB_LOGF(log,
"GDBRemoteCommunicationServerLLGS::%s called", __FUNCTION__);
1108 "GDBRemoteCommunicationServerLLGS::%s failed to send stop "
1109 "notification for PID %" PRIu64
", state: eStateExited",
1110 __FUNCTION__, process->
GetID());
1141 assert(process &&
"process cannot be NULL");
1144 LLDB_LOGF(log,
"GDBRemoteCommunicationServerLLGS::%s called", __FUNCTION__);
1150 "GDBRemoteCommunicationServerLLGS::%s failed to send stop "
1151 "notification for PID %" PRIu64
", state: eStateExited",
1152 __FUNCTION__, process->
GetID());
1158 assert(process &&
"process cannot be NULL");
1161 "GDBRemoteCommunicationServerLLGS::%s called with "
1162 "NativeProcessProtocol pid %" PRIu64
", state: %s",
1190 "GDBRemoteCommunicationServerLLGS::%s didn't handle state "
1191 "change for pid %" PRIu64
", new state: %s",
1203 std::unique_ptr<NativeProcessProtocol> child_process) {
1215 bool interrupt =
false;
1220 std::chrono::microseconds(0),
error, interrupt, done);
1226 "GDBRemoteCommunicationServerLLGS::%s processing a packet "
1228 __FUNCTION__,
error.AsCString());
1236 std::unique_ptr<Connection> connection) {
1237 IOObjectSP read_object_sp = connection->GetReadObject();
1249 const llvm::json::Value &value) {
1250 std::string json_string;
1251 raw_string_ostream os(json_string);
1256 escaped_response.
PutEscapedBytes(json_string.c_str(), json_string.size());
1263 if ((buffer ==
nullptr) || (len == 0)) {
1282 std::unique_ptr<ConnectionFileDescriptor> conn_up(
1294 "failed to set connection for inferior I/O communication");
1316 LLDB_LOG(log,
"Failed to set up stdio forwarding: {0}",
error);
1330 buffer,
sizeof buffer, std::chrono::microseconds(0), status, &
error);
1341 "GDBRemoteCommunicationServerLLGS::%s Stopping stdio "
1342 "forwarding as communication returned status %d (error: "
1344 __FUNCTION__, status,
error.AsCString());
1376 Expected<TraceStopRequest> stop_request =
1377 json::parse<TraceStopRequest>(packet.
Peek(),
"TraceStopRequest");
1397 Expected<TraceStartRequest> request =
1398 json::parse<TraceStartRequest>(packet.
Peek(),
"TraceStartRequest");
1418 Expected<TraceGetStateRequest> request =
1419 json::parse<TraceGetStateRequest>(packet.
Peek(),
"TraceGetStateRequest");
1436 llvm::Expected<TraceGetBinaryDataRequest> request =
1437 llvm::json::parse<TraceGetBinaryDataRequest>(packet.
Peek(),
1438 "TraceGetBinaryDataRequest");
1442 if (Expected<std::vector<uint8_t>> bytes =
1459 std::vector<std::string> structured_data_plugins =
1463 llvm::json::Value(llvm::json::Array(structured_data_plugins)));
1520 LLDB_LOG(log,
"No debugged process found.");
1526 LLDB_LOG(log,
"Killing process {0}", it->first);
1529 LLDB_LOG(log,
"Failed to kill debugged process {0}: {1}", it->first,
1550 "vKill failed to parse the process id");
1568 packet.
SetFilePos(::strlen(
"QSetDisableASLR:"));
1579 packet.
SetFilePos(::strlen(
"QSetWorkingDir:"));
1621 LLDB_LOG(log,
"process {0} cannot be resumed (state={1})", process.
GetID(),
1640 LLDB_LOGF(log,
"GDBRemoteCommunicationServerLLGS::%s called", __FUNCTION__);
1645 "GDBRemoteCommunicationServerLLGS::%s no debugged process "
1657 const uint32_t signo =
1658 packet.
GetHexMaxU32(
false, std::numeric_limits<uint32_t>::max());
1659 if (signo == std::numeric_limits<uint32_t>::max())
1665 if (*packet.
Peek() ==
';')
1669 packet,
"unexpected content after $C{signal-number}");
1675 LLDB_LOG(log,
"process cannot be resumed (state={0})",
1696 static_cast<int>(signo)};
1700 resume_actions.
Append(action);
1706 LLDB_LOG(log,
"failed to send signal for process {0}: {1}",
1727 LLDB_LOGF(log,
"GDBRemoteCommunicationServerLLGS::%s called", __FUNCTION__);
1732 const bool has_continue_address = (packet.
GetBytesLeft() > 0);
1733 if (has_continue_address) {
1734 LLDB_LOG(log,
"not implemented for c[address] variant [{0} remains]",
1742 "GDBRemoteCommunicationServerLLGS::%s no debugged process "
1763 response.
Printf(
"vCont;c;C;s;S;t");
1784 LLDB_LOGF(log,
"GDBRemoteCommunicationServerLLGS::%s handling vCont packet",
1791 "GDBRemoteCommunicationServerLLGS::%s missing action from "
1797 if (::strcmp(packet.
Peek(),
";s") == 0) {
1803 std::unordered_map<lldb::pid_t, ResumeActionList> thread_actions;
1815 const char action = packet.
GetChar();
1819 if (thread_action.
signal == 0)
1821 packet,
"Could not parse signal in vCont packet C action");
1831 if (thread_action.
signal == 0)
1833 packet,
"Could not parse signal in vCont packet S action");
1864 pid = pid_tid->first;
1865 tid = pid_tid->second;
1871 packet,
"'t' action not supported for individual threads");
1877 LLDB_LOG(log,
"no process selected via Hc");
1886 thread_action.
tid = tid;
1891 packet,
"vCont: p-1 is not valid with a specific tid");
1893 thread_actions[process_it.first].Append(thread_action);
1895 thread_actions[pid].Append(thread_action);
1898 assert(thread_actions.size() >= 1);
1899 if (thread_actions.size() > 1 && !
m_non_stop)
1902 "Resuming multiple processes is supported in non-stop mode only");
1904 for (std::pair<lldb::pid_t, ResumeActionList> x : thread_actions) {
1907 LLDB_LOG(log,
"vCont failed for process {0}: process not debugged",
1924 assert(process_it->second.process_up);
1926 if (process_it->second.process_up->IsRunning()) {
1929 Status error = process_it->second.process_up->Interrupt();
1931 LLDB_LOG(log,
"vCont failed to halt process {0}: {1}", x.first,
1936 LLDB_LOG(log,
"halted process {0}", x.first);
1940 assert(thread_actions.size() == 1);
1956 LLDB_LOG(log,
"setting current thread id to {0}", tid);
1965 LLDB_LOG(log,
"setting continue thread id to {0}", tid);
1979 return x.front() !=
'W' && x.front() !=
'X';
1989 if (!stop_reply.
Empty())
2016 bool force_synchronous) {
2023 if (it.second.process_up->IsRunning())
2033 switch (process_state) {
2059 LLDB_LOG(log,
"pid {0}, current state reporting not handled: {1}",
2060 process.
GetID(), process_state);
2085 const uint32_t reg_index =
2086 packet.
GetHexMaxU32(
false, std::numeric_limits<uint32_t>::max());
2087 if (reg_index == std::numeric_limits<uint32_t>::max())
2118 if (!encoding.empty())
2119 response <<
"encoding:" << encoding <<
';';
2122 if (!format.empty())
2123 response <<
"format:" << format <<
';';
2125 const char *
const register_set_name =
2127 if (register_set_name)
2128 response <<
"set:" << register_set_name <<
';';
2132 response.
Printf(
"ehframe:%" PRIu32
";",
2136 response.
Printf(
"dwarf:%" PRIu32
";",
2140 if (!kind_generic.empty())
2141 response <<
"generic:" << kind_generic <<
';';
2166 LLDB_LOG(log,
"iterating over threads of process {0}", process.
GetID());
2168 LLDB_LOG(log,
"iterated thread tid={0}", thread.GetID());
2169 response.
PutChar(had_any ?
',' :
'm');
2182 bool had_any =
false;
2211 LLDB_LOG(log,
"failed, no thread available");
2218 std::vector<uint8_t> regs_buffer;
2223 if (reg_info ==
nullptr) {
2224 LLDB_LOG(log,
"failed to get register info for register index {0}",
2235 LLDB_LOG(log,
"failed to read register at index {0}", reg_num);
2262 const uint32_t reg_index =
2263 packet.
GetHexMaxU32(
false, std::numeric_limits<uint32_t>::max());
2264 if (reg_index == std::numeric_limits<uint32_t>::max()) {
2266 "GDBRemoteCommunicationServerLLGS::%s failed, could not "
2267 "parse register number from request \"%s\"",
2275 LLDB_LOG(log,
"failed, no thread available");
2286 "GDBRemoteCommunicationServerLLGS::%s failed, requested "
2287 "register %" PRIu32
" beyond register count %" PRIu32,
2295 "GDBRemoteCommunicationServerLLGS::%s failed, requested "
2296 "register %" PRIu32
" returned NULL",
2297 __FUNCTION__, reg_index);
2309 "GDBRemoteCommunicationServerLLGS::%s failed, read of "
2310 "requested register %" PRIu32
" (%s) failed: %s",
2311 __FUNCTION__, reg_index, reg_info->
name,
error.AsCString());
2315 const uint8_t *
const data =
2316 static_cast<const uint8_t *
>(reg_value.
GetBytes());
2319 "GDBRemoteCommunicationServerLLGS::%s failed to get data "
2320 "bytes from requested register %" PRIu32,
2321 __FUNCTION__, reg_index);
2326 for (uint32_t i = 0; i < reg_value.
GetByteSize(); ++i)
2342 const uint32_t reg_index =
2343 packet.
GetHexMaxU32(
false, std::numeric_limits<uint32_t>::max());
2344 if (reg_index == std::numeric_limits<uint32_t>::max()) {
2346 "GDBRemoteCommunicationServerLLGS::%s failed, could not "
2347 "parse register number from request \"%s\"",
2355 packet,
"P packet missing '=' char after register number");
2364 "GDBRemoteCommunicationServerLLGS::%s failed, no thread "
2365 "available (thread index 0)",
2375 "GDBRemoteCommunicationServerLLGS::%s failed, requested "
2376 "register %" PRIu32
" returned NULL",
2377 __FUNCTION__, reg_index);
2385 "GDBRemoteCommunicationServerLLGS::%s failed, requested "
2386 "register %" PRIu32
" beyond register count %" PRIu32,
2402 "GDBRemoteCommunicationServerLLGS::%s failed, write of "
2403 "requested register %" PRIu32
" (%s) failed: %s",
2404 __FUNCTION__, reg_index, reg_info->
name,
error.AsCString());
2419 "GDBRemoteCommunicationServerLLGS::%s failed, H command "
2420 "missing {g,c} variant",
2425 const char h_variant = packet.
GetChar();
2427 switch (h_variant) {
2439 "GDBRemoteCommunicationServerLLGS::%s failed, invalid $H variant %c",
2440 __FUNCTION__, h_variant);
2442 "H variant unsupported, should be c or g");
2446 auto pid_tid = packet.
GetPidTid(default_process ? default_process->
GetID()
2458 llvm::createStringError(
"no current process and no PID provided"));
2464 llvm::createStringErrorV(
"no process with PID {0} debugged", pid));
2470 new_process_it->second.process_up->GetThreadByID(tid);
2473 "GDBRemoteCommunicationServerLLGS::%s failed, tid %" PRIu64
2481 switch (h_variant) {
2493 assert(
false &&
"unsupported $H variant - shouldn't get here");
2495 "H variant unsupported, should be c or g");
2510 "GDBRemoteCommunicationServerLLGS::%s failed, no process available",
2544 LLDB_LOG(log,
"failed, no process available");
2571 "GDBRemoteCommunicationServerLLGS::%s failed, no process available",
2594 const uint64_t byte_count = packet.
GetHexMaxU64(
false, 0);
2595 if (byte_count == 0) {
2597 "GDBRemoteCommunicationServerLLGS::%s nothing to read: "
2598 "zero-length packet",
2604 std::string buf(byte_count,
'\0');
2609 size_t bytes_read = 0;
2611 read_addr, &buf[0], byte_count, bytes_read);
2614 "ReadMemoryWithoutTrap({0}) read {1} of {2} requested bytes (error: {3})",
2615 read_addr, byte_count, bytes_read,
error);
2616 if (bytes_read == 0)
2621 char kind = packet.
GetChar(
'?');
2625 assert(kind ==
'm');
2626 for (
size_t i = 0; i < bytes_read; ++i)
2641 "GDBRemoteCommunicationServerLLGS::%s failed, no process available",
2656 Permissions perms = {};
2660 perms |= ePermissionsReadable;
2663 perms |= ePermissionsWritable;
2666 perms |= ePermissionsExecutable;
2690 "GDBRemoteCommunicationServerLLGS::%s failed, no process available",
2718 "GDBRemoteCommunicationServerLLGS::%s failed, no process available",
2741 const uint64_t byte_count = packet.
GetHexMaxU64(
false, 0);
2742 if (byte_count == 0) {
2743 LLDB_LOG(log,
"nothing to write: zero-length packet");
2750 packet,
"Comma sep missing in M packet after byte length");
2753 std::vector<uint8_t> buf(byte_count, 0);
2759 const uint64_t convert_count = packet.
GetHexBytes(buf, 0);
2760 if (convert_count != byte_count) {
2762 "pid {0} mem {1:x}: asked to write {2} bytes, but only found {3} "
2766 "not match hex-encoded content "
2771 size_t bytes_written = 0;
2775 LLDB_LOG(log,
"pid {0} mem {1:x}: failed to write. Error: {2}",
2780 if (bytes_written == 0) {
2781 LLDB_LOG(log,
"pid {0} mem {1:x}: wrote 0 of {2} requested bytes",
2805 "GDBRemoteCommunicationServerLLGS::%s failed, no process available",
2832 "GDBRemoteCommunicationServerLLGS::%s failed, no process available",
2838 packet.
SetFilePos(strlen(
"qMemoryRegionInfo:"));
2859 response.
Printf(
"start:%" PRIx64
";size:%" PRIx64
";",
2903 if (std::optional<unsigned> protection_key = region_info.
GetProtectionKey())
2904 response.
Printf(
"protection-key:%" PRIu32
";", *protection_key);
2911struct UseBreakpoint {
2912 bool want_hardware =
false;
2914struct UseWatchpoint {
2916 static constexpr bool want_hardware =
true;
2918struct InvalidStoppoint {};
2920std::variant<UseBreakpoint, UseWatchpoint, InvalidStoppoint>
2922 switch (stoppoint_type) {
2924 return UseBreakpoint{
false};
2926 return UseBreakpoint{
true};
2928 return UseWatchpoint{ 1};
2930 return UseWatchpoint{ 2};
2932 return UseWatchpoint{ 3};
2934 return InvalidStoppoint();
2936 llvm_unreachable(
"unhandled GDBStoppointType");
2942 llvm::StringRef packet_str) {
2947 LLDB_LOG(log,
"failed, no process available");
2957 "Too short Z packet, missing software/hardware specifier"};
2961 std::variant<UseBreakpoint, UseWatchpoint, InvalidStoppoint> bp_variant =
2962 getBreakpointKind(stoppoint_type);
2963 if (std::holds_alternative<InvalidStoppoint>(bp_variant))
2965 "Z packet had invalid software/hardware specifier"};
2969 "Malformed Z packet, expecting comma after stoppoint type"};
2978 "Malformed Z packet, expecting comma after address"};
2981 const uint32_t size =
2982 packet.
GetHexMaxU32(
false, std::numeric_limits<uint32_t>::max());
2983 if (size == std::numeric_limits<uint32_t>::max())
2985 "Malformed Z packet, failed to parse size argument"};
2988 if (
auto *bp_kind = std::get_if<UseBreakpoint>(&bp_variant)) {
2991 if (
error.Success())
2994 LLDB_LOG(log,
"pid {0} failed to set breakpoint: {1}",
3000 auto wp_kind = std::get<UseWatchpoint>(bp_variant);
3002 addr, size, wp_kind.flags, wp_kind.want_hardware);
3003 if (
error.Success())
3006 LLDB_LOG(log,
"pid {0} failed to set watchpoint: {1}",
3013 llvm::StringRef packet_str) {
3018 LLDB_LOG(log,
"failed, no process available");
3028 "Too short z packet, missing software/hardware specifier"};
3032 std::variant<UseBreakpoint, UseWatchpoint, InvalidStoppoint> bp_variant =
3033 getBreakpointKind(stoppoint_type);
3034 if (std::holds_alternative<InvalidStoppoint>(bp_variant))
3036 "z packet had invalid software/hardware specifier"};
3040 "Malformed z packet, expecting comma after stoppoint type"};
3049 "Malformed z packet, expecting comma after address"};
3061 if (
auto *bp_kind = std::get_if<UseBreakpoint>(&bp_variant)) {
3064 if (
error.Success())
3067 LLDB_LOG(log,
"pid {0} failed to remove breakpoint: {1}",
3073 if (
error.Success())
3076 LLDB_LOG(log,
"pid {0} failed to remove watchpoint: {1}",
3086 using T = std::decay_t<
decltype(arg)>;
3087 static_assert(std::is_same_v<T, BreakpointOK> ||
3088 std::is_same_v<T, BreakpointError> ||
3089 std::is_same_v<T, BreakpointIllFormed>,
3090 "non-exhaustive visitor!");
3091 if constexpr (std::is_same_v<T, BreakpointOK>)
3093 else if constexpr (std::is_same_v<T, BreakpointError>)
3117 if (!packet_str.consume_front(
"jMultiBreakpoint:"))
3119 "Invalid jMultiBreakpoint packet prefix");
3121 llvm::Expected<llvm::json::Value> parsed = llvm::json::parse(packet_str);
3123 llvm::consumeError(parsed.takeError());
3125 "jMultiBreakpoint did not contain valid JSON");
3127 llvm::json::Object *request_dict = parsed->getAsObject();
3130 packet,
"jMultiBreakpoint did not contain a JSON dictionary");
3132 llvm::json::Array *request_array =
3133 request_dict->getArray(
"breakpoint_requests");
3137 "jMultiBreakpoint did not contain a valid 'breakpoint_requests' field");
3139 llvm::json::Array reply_array;
3140 for (
const llvm::json::Value &value : *request_array) {
3141 std::optional<llvm::StringRef> request = value.getAsString();
3144 "jMultiBreakpoint had a non-string entry");
3149 [&](
const auto &arg) {
3150 using T = std::decay_t<
decltype(arg)>;
3151 static_assert(std::is_same_v<T, BreakpointOK> ||
3152 std::is_same_v<T, BreakpointError> ||
3153 std::is_same_v<T, BreakpointIllFormed>,
3154 "non-exhaustive visitor!");
3155 if constexpr (std::is_same_v<T, BreakpointOK>)
3156 reply_array.push_back(
"OK");
3157 else if constexpr (std::is_same_v<T, BreakpointError>)
3158 reply_array.push_back(
3159 llvm::formatv(
"E{0:X-2}", arg.error_code).str());
3161 reply_array.push_back(
"E03");
3166 llvm::json::Object dict;
3167 dict.try_emplace(
"results", std::move(reply_array));
3170 stream.
AsRawOstream() << llvm::json::Value(std::move(dict));
3171 StringRef response_str = stream.
GetString();
3186 "GDBRemoteCommunicationServerLLGS::%s failed, no process available",
3224llvm::Expected<std::unique_ptr<llvm::MemoryBuffer>>
3229 return llvm::createStringError(llvm::inconvertibleErrorCode(),
3230 "No thread available");
3238 response.
Printf(
"<?xml version=\"1.0\"?>\n");
3239 response.
Printf(
"<target version=\"1.0\">\n");
3243 response.
Printf(
"<architecture>%s</architecture>\n",
3250 response.
Indent(
"<feature>\n");
3253 if (registers_count)
3256 llvm::StringSet<> field_enums_seen;
3257 for (
int reg_index = 0; reg_index < registers_count; reg_index++) {
3263 "%s failed to get register info for register index %" PRIu32,
3264 "target.xml", reg_index);
3276 response.
Printf(
"<reg name=\"%s\" bitsize=\"%" PRIu32
3277 "\" regnum=\"%d\" ",
3287 if (!encoding.empty())
3288 response <<
"encoding=\"" << encoding <<
"\" ";
3291 if (!format.empty())
3292 response <<
"format=\"" << format <<
"\" ";
3297 const char *
const register_set_name =
3299 if (register_set_name)
3300 response <<
"group=\"" << register_set_name <<
"\" ";
3304 response.
Printf(
"ehframe_regnum=\"%" PRIu32
"\" ",
3309 response.
Printf(
"dwarf_regnum=\"%" PRIu32
"\" ",
3313 if (!kind_generic.empty())
3314 response <<
"generic=\"" << kind_generic <<
"\" ";
3324 response.
PutCString(
"invalidate_regnums=\"");
3332 if (registers_count)
3335 response.
Indent(
"</feature>\n");
3337 response.
Indent(
"</target>\n");
3338 return MemoryBuffer::getMemBufferCopy(response.
GetString(),
"target.xml");
3341llvm::Expected<std::unique_ptr<llvm::MemoryBuffer>>
3343 llvm::StringRef annex) {
3347 return llvm::createStringError(llvm::inconvertibleErrorCode(),
3348 "No process available");
3351 if (
object ==
"auxv") {
3354 if (!buffer_or_error)
3355 return llvm::errorCodeToError(buffer_or_error.getError());
3356 return std::move(*buffer_or_error);
3359 if (
object ==
"siginfo") {
3362 return llvm::createStringError(llvm::inconvertibleErrorCode(),
3363 "no current thread");
3365 auto buffer_or_error = thread->GetSiginfo();
3366 if (!buffer_or_error)
3367 return buffer_or_error.takeError();
3368 return std::move(*buffer_or_error);
3371 if (
object ==
"libraries-svr4") {
3374 return library_list.takeError();
3377 response.
Printf(
"<library-list-svr4 version=\"1.0\">");
3378 for (
auto const &library : *library_list) {
3379 response.
Printf(
"<library name=\"%s\" ",
3381 response.
Printf(
"lm=\"0x%" PRIx64
"\" ", library.link_map);
3382 response.
Printf(
"l_addr=\"0x%" PRIx64
"\" ", library.base_addr);
3383 response.
Printf(
"l_ld=\"0x%" PRIx64
"\" />", library.ld_addr);
3385 response.
Printf(
"</library-list-svr4>");
3386 return MemoryBuffer::getMemBufferCopy(response.
GetString(), __FUNCTION__);
3389 if (
object ==
"features" && annex ==
"target.xml")
3392 return llvm::make_error<UnimplementedError>();
3398 SmallVector<StringRef, 5> fields;
3400 StringRef(packet.
GetStringRef()).split(fields,
':', 4);
3401 if (fields.size() != 5)
3403 StringRef &xfer_object = fields[1];
3404 StringRef &xfer_action = fields[2];
3405 StringRef &xfer_annex = fields[3];
3407 if (xfer_action !=
"read")
3410 const uint64_t xfer_offset =
3411 offset_data.
GetHexMaxU64(
false, std::numeric_limits<uint64_t>::max());
3412 if (xfer_offset == std::numeric_limits<uint64_t>::max())
3415 if (offset_data.
GetChar() !=
',')
3417 "qXfer packet missing comma after offset");
3419 const uint64_t xfer_length =
3420 offset_data.
GetHexMaxU64(
false, std::numeric_limits<uint64_t>::max());
3421 if (xfer_length == std::numeric_limits<uint64_t>::max())
3425 std::string buffer_key = (xfer_object + xfer_action + xfer_annex).str();
3432 .insert(std::make_pair(buffer_key, std::move(*buffer_up)))
3438 bool done_with_buffer =
false;
3439 llvm::StringRef buffer = buffer_it->second->getBuffer();
3440 if (xfer_offset >= buffer.size()) {
3443 done_with_buffer =
true;
3446 buffer = buffer.drop_front(xfer_offset);
3449 if (xfer_length >= buffer.size()) {
3452 done_with_buffer =
true;
3456 buffer = buffer.take_front(xfer_length);
3462 if (done_with_buffer)
3474 packet.
SetFilePos(strlen(
"QSaveRegisterState"));
3481 packet,
"No thread specified in QSaveRegisterState packet");
3484 "No thread was is set with the Hg packet");
3494 LLDB_LOG(log,
"pid {0} failed to save all register values: {1}",
3502 "GetNextRegisterSaveID() returned an existing register save id");
3512 response.
Printf(
"%" PRIu32, save_id);
3522 packet.
SetFilePos(strlen(
"QRestoreRegisterState:"));
3525 packet,
"QRestoreRegisterState packet missing register save id");
3527 const uint32_t save_id = packet.
GetU32(0);
3529 LLDB_LOG(log,
"QRestoreRegisterState packet has malformed save id, "
3530 "expecting decimal uint32_t");
3539 packet,
"No thread specified in QRestoreRegisterState packet");
3542 "No thread was is set with the Hg packet");
3557 "pid {0} does not have a register set save buffer for id {1}",
3561 register_data_sp = it->second;
3569 LLDB_LOG(log,
"pid {0} failed to restore all register values: {1}",
3591 "vAttach failed to parse the process id");
3595 "GDBRemoteCommunicationServerLLGS::%s attempting to attach to "
3603 "GDBRemoteCommunicationServerLLGS::%s failed to attach to "
3604 "pid %" PRIu64
": %s\n",
3605 __FUNCTION__, pid,
error.AsCString());
3628 std::string process_name;
3631 "vAttachWait failed to parse process name");
3633 LLDB_LOG(log,
"attempting to attach to process named '{0}'", process_name);
3637 LLDB_LOG(log,
"failed to attach to process named '{0}': {1}", process_name,
3667 std::string process_name;
3670 "vAttachOrWait failed to parse process name");
3672 LLDB_LOG(log,
"attempting to attach to process named '{0}'", process_name);
3676 LLDB_LOG(log,
"failed to attach to process named '{0}': {1}", process_name,
3694 if (!s.consume_front(
"vRun;"))
3697 llvm::SmallVector<llvm::StringRef, 16> argv;
3700 for (llvm::StringRef hex_arg : argv) {
3705 LLDB_LOGF(log,
"LLGSPacketHandler::%s added arg: \"%s\"", __FUNCTION__,
3744 llvm::Error detach_error = llvm::Error::success();
3745 bool detached =
false;
3750 "GDBRemoteCommunicationServerLLGS::%s detaching %" PRId64,
3751 __FUNCTION__, it->first);
3752 if (llvm::Error e = it->second.process_up->Detach().ToError())
3753 detach_error = llvm::joinErrors(std::move(detach_error), std::move(e));
3784 packet.
SetFilePos(strlen(
"qThreadStopInfo"));
3788 "GDBRemoteCommunicationServerLLGS::%s failed, could not "
3789 "parse thread id from request \"%s\"",
3809 const bool threads_with_valid_stop_info_only =
false;
3810 llvm::Expected<json::Value> threads_info =
3812 if (!threads_info) {
3814 "failed to prepare a packet for pid {1}: {0}",
3833 packet.
SetFilePos(strlen(
"qWatchpointSupportInfo"));
3842 if (hw_debug_cap == std::nullopt)
3843 response.
Printf(
"num:0;");
3845 response.
Printf(
"num:%d;", hw_debug_cap->second);
3858 packet.
SetFilePos(strlen(
"qFileLoadAddress:"));
3862 std::string file_name;
3875 response.
PutHex64(file_load_address);
3882 std::vector<int> signals;
3888 int signal = packet.
GetS32(-1, 16);
3891 signals.push_back(signal);
3894 char separator = packet.
GetChar();
3895 if (separator ==
'\0')
3897 if (separator !=
';')
3899 " expected semicolon.");
3923 "GDBRemoteCommunicationServerLLGS::%s failed, no process available",
3933 const char *current_char = packet.
Peek();
3934 if (!current_char || *current_char ==
',')
3939 char previous_char = packet.
GetChar();
3940 current_char = packet.
Peek();
3942 if (previous_char !=
',' || (current_char && *current_char ==
':'))
3944 "Invalid addr,length pair in qMemTags packet");
3948 packet,
"Too short qMemtags: packet (looking for length)");
3952 const char *invalid_type_err =
"Invalid type field in qMemTags: packet";
3958 const char *first_type_char = packet.
Peek();
3959 if (first_type_char && (*first_type_char ==
'+' || *first_type_char ==
'-'))
3966 packet.
GetU64(std::numeric_limits<uint64_t>::max(), 16);
3969 raw_type > std::numeric_limits<uint32_t>::max() ||
3978 uint32_t raw_type_32 = raw_type;
3979 int32_t type =
reinterpret_cast<int32_t &
>(raw_type_32);
3982 std::vector<uint8_t> tags;
4004 "GDBRemoteCommunicationServerLLGS::%s failed, no process available",
4014 const char *current_char = packet.
Peek();
4015 if (!current_char || *current_char ==
',')
4020 char previous_char = packet.
GetChar();
4021 current_char = packet.
Peek();
4023 if (previous_char !=
',' || (current_char && *current_char ==
':'))
4025 "Invalid addr,length pair in QMemTags packet");
4029 packet,
"Too short QMemtags: packet (looking for length)");
4033 const char *invalid_type_err =
"Invalid type field in QMemTags: packet";
4038 const char *first_type_char = packet.
Peek();
4039 if (first_type_char && (*first_type_char ==
'+' || *first_type_char ==
'-'))
4047 packet.
GetU64(std::numeric_limits<uint64_t>::max(), 16);
4048 if (raw_type > std::numeric_limits<uint32_t>::max())
4053 uint32_t raw_type_32 = raw_type;
4054 int32_t type =
reinterpret_cast<int32_t &
>(raw_type_32);
4059 "Missing tag data in QMemTags: packet");
4062 const char *invalid_data_err =
"Invalid tag data in QMemTags: packet";
4069 std::vector<uint8_t> tag_data;
4074 tag_data.resize(byte_count);
4075 size_t converted_bytes = packet.
GetHexBytes(tag_data, 0);
4076 if (converted_bytes != byte_count) {
4094 std::string path_hint;
4097 assert(packet_str.starts_with(
"qSaveCore"));
4098 if (packet_str.consume_front(
"qSaveCore;")) {
4099 for (
auto x : llvm::split(packet_str,
';')) {
4100 if (x.consume_front(
"path-hint:"))
4124 assert(packet_str.starts_with(
"QNonStop:"));
4125 packet_str.consume_front(
"QNonStop:");
4126 if (packet_str ==
"0") {
4130 if (process_it.second.process_up->IsRunning()) {
4132 Status error = process_it.second.process_up->Interrupt();
4135 "while disabling nonstop, failed to halt process {0}: {1}",
4136 process_it.first,
error);
4150 }
else if (packet_str ==
"1") {
4162 std::deque<std::string> &queue) {
4206 return interrupt_res;
4226 llvm::createStringError(
"no current process and no PID provided"));
4234 if (!new_process_it->second.process_up->GetThreadByID(tid))
4248 connection->Disconnect(&
error);
4250 if (
error.Success()) {
4252 "GDBRemoteCommunicationServerLLGS::%s disconnect process "
4253 "terminal stdio - SUCCESS",
4257 "GDBRemoteCommunicationServerLLGS::%s disconnect process "
4258 "terminal stdio - FAIL: %s",
4259 __FUNCTION__,
error.AsCString());
4278 else if (current_tid == 0) {
4290 "GDBRemoteCommunicationServerLLGS::%s gdb-remote parse "
4291 "error: expected ';' prior to start of thread suffix: packet "
4301 if (strncmp(packet.
Peek(),
"thread:", strlen(
"thread:")) != 0) {
4303 "GDBRemoteCommunicationServerLLGS::%s gdb-remote parse "
4304 "error: expected 'thread:' but not found, packet contents = "
4348 ->GetLoadedModuleFileSpec(module_path.c_str(), file_spec)
4359 llvm::StringRef value) {
4361 for (
const char &c : value) {
4384 const llvm::ArrayRef<llvm::StringRef> client_features) {
4385 std::vector<std::string> ret =
4387 ret.insert(ret.end(), {
4388 "QThreadSuffixSupported+",
4389 "QListThreadsInStopReply+",
4390 "qXfer:features:read+",
4392 "jMultiBreakpoint+",
4398 if (
bool(plugin_features & Extension::pass_signals))
4399 ret.push_back(
"QPassSignals+");
4400 if (
bool(plugin_features & Extension::auxv))
4401 ret.push_back(
"qXfer:auxv:read+");
4402 if (
bool(plugin_features & Extension::libraries_svr4))
4403 ret.push_back(
"qXfer:libraries-svr4:read+");
4404 if (
bool(plugin_features & Extension::siginfo_read))
4405 ret.push_back(
"qXfer:siginfo:read+");
4406 if (
bool(plugin_features & Extension::memory_tagging))
4407 ret.push_back(
"memory-tagging+");
4408 if (
bool(plugin_features & Extension::savecore))
4409 ret.push_back(
"qSaveCore+");
4413 for (llvm::StringRef x : client_features)
4415 llvm::StringSwitch<Extension>(x)
4416 .Case(
"multiprocess+", Extension::multiprocess)
4417 .Case(
"fork-events+", Extension::fork)
4418 .Case(
"vfork-events+", Extension::vfork)
4433 ret.push_back(
"multiprocess+");
4435 ret.push_back(
"fork-events+");
4437 ret.push_back(
"vfork-events+");
4463 response.
Format(
"p{0:x-}.", pid);
4464 response.
Format(
"{0:x-}", tid);
4469 bool reverse_connect) {
4471 if (std::optional<URI> url =
URI::Parse(url_arg)) {
4472 if (reverse_connect)
4473 return url_arg.str();
4478 std::string new_url = llvm::StringSwitch<std::string>(url->scheme)
4479 .Case(
"tcp",
"listen")
4480 .Case(
"unix",
"unix-accept")
4481 .Case(
"unix-abstract",
"unix-abstract-accept")
4482 .Default(url->scheme.str());
4483 llvm::append_range(new_url, url_arg.substr(url->scheme.size()));
4487 std::string host_port = url_arg.str();
4490 if (url_arg.starts_with(
":"))
4491 host_port.insert(0,
"localhost");
4495 return (reverse_connect ?
"connect://" :
"listen://") + host_port;
4498 return (reverse_connect ?
"unix-connect://" :
"unix-accept://") +
static const size_t reg_size
static llvm::raw_ostream & error(Stream &strm)
static llvm::StringRef GetEncodingNameOrEmpty(const RegisterInfo ®_info)
static llvm::StringRef GetFormatNameOrEmpty(const RegisterInfo ®_info)
static void WriteRegisterValueInHexFixedWidth(StreamString &response, NativeRegisterContext ®_ctx, const RegisterInfo ®_info, const RegisterValue *reg_value_p, lldb::ByteOrder byte_order)
static void AppendHexValue(StreamString &response, const uint8_t *buf, uint32_t buf_size, bool swap)
static std::optional< json::Object > GetRegistersAsJSON(NativeThreadProtocol &thread)
static const char * GetStopReasonString(StopReason stop_reason)
static void CollectRegNums(const uint32_t *reg_num, StreamString &response, bool usehex)
static bool ResumeActionListStopsAllThreads(ResumeActionList &actions)
static llvm::StringRef GetKindGenericOrEmpty(const RegisterInfo ®_info)
static llvm::Expected< json::Array > GetJSONThreadsInfo(NativeProcessProtocol &process, bool abridged)
#define LLDB_LOG(log,...)
The LLDB_LOG* macros defined below are the way to emit log messages.
#define LLDB_LOGF(log,...)
#define LLDB_LOG_ERROR(log, error,...)
void swap(lldb_private::NonNullSharedPtr< T > &lhs, lldb_private::NonNullSharedPtr< T > &rhs)
Specialized swap function for NonNullSharedPtr to enable argument-dependent lookup (ADL) and efficien...
An architecture specification class.
virtual void SetConnection(std::unique_ptr< Connection > connection)
Sets the connection that it to be used by this class.
A uniqued constant string class.
llvm::StringRef GetStringRef() const
Get the string value as a llvm::StringRef.
void SetFile(llvm::StringRef path, Style style)
Change the file specified with a new path.
size_t GetPath(char *path, size_t max_path_length, bool denormalize=true) const
Extract the full path to the file.
static FileSystem & Instance()
static bool GetProcessInfo(lldb::pid_t pid, ProcessInstanceInfo &proc_info)
static uint32_t FindProcesses(const ProcessInstanceInfoMatch &match_info, ProcessInstanceInfoList &proc_infos)
virtual void RequestTermination()
LazyBool GetReadable() const
std::optional< unsigned > GetProtectionKey() const
ConstString GetName() const
LazyBool GetExecutable() const
LazyBool IsShadowStack() const
LazyBool GetWritable() const
LazyBool GetMemoryTagged() const
virtual std::optional< WaitStatus > GetExitStatus()
virtual void SetEnabledExtensions(Extension flags)
Method called in order to propagate the bitmap of protocol extensions supported by the client.
lldb::pid_t GetID() const
lldb::StateType GetState() const
NativeThreadProtocol * GetThreadByID(lldb::tid_t tid)
virtual Status Resume(const ResumeActionList &resume_actions)=0
lldb::tid_t GetCurrentThreadID() const
Extension
Extension flag constants, returned by Manager::GetSupportedExtensions() and passed to SetEnabledExten...
ThreadIterable Threads() const
uint32_t ConvertRegisterKindToRegisterNumber(uint32_t kind, uint32_t num) const
virtual uint32_t GetUserRegisterCount() const =0
virtual const RegisterInfo * GetRegisterInfoAtIndex(uint32_t reg) const =0
const char * GetRegisterSetNameForRegisterAtIndex(uint32_t reg_index) const
virtual Status WriteAllRegisterValues(const lldb::DataBufferSP &data_sp)=0
virtual Status ReadRegister(const RegisterInfo *reg_info, RegisterValue ®_value)=0
virtual bool RegisterOffsetIsDynamic() const
virtual Status ReadAllRegisterValues(lldb::WritableDataBufferSP &data_sp)=0
virtual std::vector< uint32_t > GetExpeditedRegisters(ExpeditedRegs expType) const
virtual Status WriteRegister(const RegisterInfo *reg_info, const RegisterValue ®_value)=0
FileSpec & GetExecutableFile()
void SetNameMatchType(NameMatch name_match_type)
ProcessInstanceInfo & GetProcessInfo()
void EnumsToXML(Stream &strm, llvm::StringSet<> &seen) const
Enum types must be defined before use, and GDBRemoteCommunicationServerLLGS view of the register type...
const std::string & GetID() const
void ToXML(Stream &strm) const
Output XML that describes this set of flags.
const void * GetBytes() const
uint32_t GetByteSize() const
const ResumeAction * GetActionForThread(lldb::tid_t tid, bool default_ok) const
void Append(const ResumeAction &action)
bool SetDefaultThreadActionIfNeeded(lldb::StateType action, int signal)
static llvm::Expected< HostAndPort > DecodeHostAndPort(llvm::StringRef host_and_port)
static Status FromErrorStringWithFormat(const char *format,...) __attribute__((format(printf
static Status FromErrorString(const char *str)
bool Fail() const
Test for error condition.
static Status FromError(llvm::Error error)
Avoid using this in new code. Migrate APIs to llvm::Expected instead.
bool Success() const
Test for success condition.
int PutEscapedBytes(const void *s, size_t src_len)
Output a block of data to the stream performing GDB-remote escaping.
const char * GetData() const
llvm::StringRef GetString() const
A stream class that can stream formatted output to a file.
void Format(const char *format, Args &&... args)
Forwards the arguments to llvm::formatv and writes to the stream.
llvm::raw_ostream & AsRawOstream()
Returns a raw_ostream that forwards the data to this Stream object.
size_t Indent(llvm::StringRef s="")
Indent the current line in the stream.
size_t size_t PutHex8(uint8_t uvalue)
Append an uint8_t value in the hexadecimal format to the stream.
size_t PutStringAsRawHex8(llvm::StringRef s)
size_t PutHex64(uint64_t uvalue, lldb::ByteOrder byte_order=lldb::eByteOrderInvalid)
size_t Printf(const char *format,...) __attribute__((format(printf
Output printf formatted output to the stream.
size_t PutCString(llvm::StringRef cstr)
Output a C string to the stream.
size_t PutHex32(uint32_t uvalue, lldb::ByteOrder byte_order=lldb::eByteOrderInvalid)
void IndentLess(unsigned amount=2)
Decrement the current indentation level.
size_t PutBytesAsRawHex8(const void *src, size_t src_len, lldb::ByteOrder src_byte_order=lldb::eByteOrderInvalid, lldb::ByteOrder dst_byte_order=lldb::eByteOrderInvalid)
void IndentMore(unsigned amount=2)
Increment the current indentation level.
GDBRemoteCommunicationServerCommon()
virtual FileSpec FindModuleFile(const std::string &module_path, const ArchSpec &arch)
void RegisterMemberFunctionHandler(StringExtractorGDBRemote::ServerPacketType packet_type, PacketResult(T::*handler)(StringExtractorGDBRemote &packet))
static void CreateProcessInfoResponse_DebugServerStyle(const ProcessInstanceInfo &proc_info, StreamString &response)
Status m_process_launch_error
ProcessLaunchInfo m_process_launch_info
virtual std::vector< std::string > HandleFeatures(llvm::ArrayRef< llvm::StringRef > client_features)
PacketResult Handle_QMemTags(StringExtractorGDBRemote &packet)
PacketResult Handle_qVAttachOrWaitSupported(StringExtractorGDBRemote &packet)
PacketResult Handle_vStdio(StringExtractorGDBRemote &packet)
PacketResult HandleNotificationAck(std::deque< std::string > &queue)
PacketResult Handle_QSetDisableASLR(StringExtractorGDBRemote &packet)
void SetLaunchInfo(const ProcessLaunchInfo &info)
PacketResult SendONotification(const char *buffer, uint32_t len)
PacketResult Handle_D(StringExtractorGDBRemote &packet)
NativeProcessProtocol::Extension m_extensions_supported
PacketResult Handle_qMemTags(StringExtractorGDBRemote &packet)
void AddProcessThreads(StreamGDBRemote &response, NativeProcessProtocol &process, bool &had_any)
llvm::StringMap< std::unique_ptr< llvm::MemoryBuffer > > m_xfer_buffer_map
PacketResult SendStopReasonForState(NativeProcessProtocol &process, lldb::StateType process_state, bool force_synchronous)
void EnqueueStopReplyPackets(lldb::tid_t thread_to_skip)
PacketResult Handle_jLLDBTraceGetBinaryData(StringExtractorGDBRemote &packet)
PacketResult Handle_qfThreadInfo(StringExtractorGDBRemote &packet)
StreamString PrepareStopReplyPacketForThread(NativeThreadProtocol &thread)
uint8_t m_reg_bytes[RegisterValue::kMaxRegisterByteSize]
PacketResult Handle__m(StringExtractorGDBRemote &packet)
GDBRemoteCommunication::PacketResult SendStructuredDataPacket(const llvm::json::Value &value)
PacketResult Handle_g(StringExtractorGDBRemote &packet)
std::variant< BreakpointOK, BreakpointIllFormed, BreakpointError > BreakpointResult
uint32_t m_next_saved_registers_id
void RegisterPacketHandlers()
PacketResult Handle_qMemoryRegionInfoSupported(StringExtractorGDBRemote &packet)
PacketResult Handle_qThreadStopInfo(StringExtractorGDBRemote &packet)
BreakpointResult ExecuteRemoveBreakpoint(llvm::StringRef packet_str)
Core logic for a z (remove breakpoint/watchpoint) request.
PacketResult Handle_c(StringExtractorGDBRemote &packet)
PacketResult Handle_QThreadSuffixSupported(StringExtractorGDBRemote &packet)
PacketResult Handle_interrupt(StringExtractorGDBRemote &packet)
PacketResult Handle_vKill(StringExtractorGDBRemote &packet)
std::unordered_map< lldb::pid_t, DebuggedProcess > m_debugged_processes
void InitializeDelegate(NativeProcessProtocol *process) override
FileSpec FindModuleFile(const std::string &module_path, const ArchSpec &arch) override
PacketResult Handle_vAttachOrWait(StringExtractorGDBRemote &packet)
PacketResult Handle_vAttach(StringExtractorGDBRemote &packet)
PacketResult Handle_stop_reason(StringExtractorGDBRemote &packet)
PacketResult Handle_jLLDBTraceSupported(StringExtractorGDBRemote &packet)
std::deque< std::string > m_stop_notification_queue
PacketResult Handle_qsThreadInfo(StringExtractorGDBRemote &packet)
PacketResult Handle_vStopped(StringExtractorGDBRemote &packet)
PacketResult Handle_z(StringExtractorGDBRemote &packet)
bool m_disabling_non_stop
lldb::tid_t GetContinueThreadID() const
PacketResult Handle_H(StringExtractorGDBRemote &packet)
void StopSTDIOForwarding()
PacketResult SendContinueSuccessResponse()
PacketResult Handle_jLLDBTraceGetState(StringExtractorGDBRemote &packet)
std::deque< std::string > m_stdio_notification_queue
bool m_list_threads_in_stop_reply
PacketResult Handle_qFileLoadAddress(StringExtractorGDBRemote &packet)
void NewSubprocess(NativeProcessProtocol *parent_process, std::unique_ptr< NativeProcessProtocol > child_process) override
PacketResult Handle_qC(StringExtractorGDBRemote &packet)
static std::string XMLEncodeAttributeValue(llvm::StringRef value)
NativeThreadProtocol * GetThreadFromSuffix(StringExtractorGDBRemote &packet)
Status SetSTDIOFileDescriptor(int fd)
lldb::tid_t GetCurrentThreadID() const
PacketResult Handle_QNonStop(StringExtractorGDBRemote &packet)
Status AttachToProcess(lldb::pid_t pid)
Attach to a process.
PacketResult SendWResponse(NativeProcessProtocol *process)
PacketResult Handle_s(StringExtractorGDBRemote &packet)
Status InitializeConnection(std::unique_ptr< Connection > connection)
bool m_thread_suffix_supported
void HandleInferiorState_Exited(NativeProcessProtocol *process)
lldb::tid_t m_continue_tid
PacketResult Handle_P(StringExtractorGDBRemote &packet)
PacketResult SendBreakpointResponse(StringExtractorGDBRemote &packet, const BreakpointResult &result)
Convert a BreakpointResult into a PacketResult, sending the appropriate response.
PacketResult Handle_qSaveCore(StringExtractorGDBRemote &packet)
NativeProcessProtocol * m_current_process
void HandleInferiorState_Stopped(NativeProcessProtocol *process)
BreakpointResult ExecuteSetBreakpoint(llvm::StringRef packet_str)
Core logic for a Z (set breakpoint/watchpoint) request.
NativeProcessProtocol * m_continue_process
PacketResult Handle_vCont(StringExtractorGDBRemote &packet)
PacketResult Handle_vCont_actions(StringExtractorGDBRemote &packet)
PacketResult Handle_vCtrlC(StringExtractorGDBRemote &packet)
PacketResult Handle_jLLDBTraceStart(StringExtractorGDBRemote &packet)
Status LaunchProcess() override
Launch a process with the current launch settings.
PacketResult Handle_qGetWorkingDir(StringExtractorGDBRemote &packet)
void DataAvailableCallback()
PacketResult Handle_qStructuredDataPlugins(StringExtractorGDBRemote &packet)
PacketResult Handle_jThreadsInfo(StringExtractorGDBRemote &packet)
PacketResult Handle_M(StringExtractorGDBRemote &packet)
void DidExec(NativeProcessProtocol *process) override
PacketResult Handle_memory_read(StringExtractorGDBRemote &packet)
Status AttachWaitProcess(llvm::StringRef process_name, bool include_existing)
Wait to attach to a process with a given name.
PacketResult Handle_vAttachWait(StringExtractorGDBRemote &packet)
PacketResult ResumeProcess(NativeProcessProtocol &process, const ResumeActionList &actions)
PacketResult Handle_QListThreadsInStopReply(StringExtractorGDBRemote &packet)
PacketResult Handle_jLLDBTraceStop(StringExtractorGDBRemote &packet)
PacketResult Handle_T(StringExtractorGDBRemote &packet)
GDBRemoteCommunicationServerLLGS(MainLoop &mainloop, NativeProcessProtocol::Manager &process_manager)
PacketResult Handle_QSetWorkingDir(StringExtractorGDBRemote &packet)
PacketResult Handle_k(StringExtractorGDBRemote &packet)
PacketResult Handle_qXfer(StringExtractorGDBRemote &packet)
PacketResult Handle_jMultiBreakpoint(StringExtractorGDBRemote &packet)
PacketResult SendStopReplyPacketForThread(NativeProcessProtocol &process, lldb::tid_t tid, bool force_synchronous)
void StartSTDIOForwarding()
PacketResult Handle_C(StringExtractorGDBRemote &packet)
PacketResult Handle__M(StringExtractorGDBRemote &packet)
void AppendThreadIDToResponse(Stream &response, lldb::pid_t pid, lldb::tid_t tid)
std::vector< std::string > HandleFeatures(const llvm::ArrayRef< llvm::StringRef > client_features) override
void ClearProcessSpecificData()
PacketResult Handle_Z(StringExtractorGDBRemote &packet)
PacketResult Handle_QPassSignals(StringExtractorGDBRemote &packet)
PacketResult Handle_qWatchpointSupportInfo(StringExtractorGDBRemote &packet)
void ProcessStateChanged(NativeProcessProtocol *process, lldb::StateType state) override
lldb::tid_t m_current_tid
void SetEnabledExtensions(NativeProcessProtocol &process)
std::unordered_map< uint32_t, lldb::DataBufferSP > m_saved_registers_map
PacketResult Handle_QRestoreRegisterState(StringExtractorGDBRemote &packet)
NativeProcessProtocol::Manager & m_process_manager
void SetContinueThreadID(lldb::tid_t tid)
PacketResult Handle_p(StringExtractorGDBRemote &packet)
PacketResult Handle_qRegisterInfo(StringExtractorGDBRemote &packet)
void SetCurrentThreadID(lldb::tid_t tid)
std::mutex m_saved_registers_mutex
PacketResult Handle_qProcessInfo(StringExtractorGDBRemote &packet)
PacketResult Handle_vRun(StringExtractorGDBRemote &packet)
MainLoop::ReadHandleUP m_stdio_handle_up
void MaybeCloseInferiorTerminalConnection()
MainLoop::ReadHandleUP m_network_handle_up
PacketResult Handle_qMemoryRegionInfo(StringExtractorGDBRemote &packet)
llvm::Expected< std::unique_ptr< llvm::MemoryBuffer > > ReadXferObject(llvm::StringRef object, llvm::StringRef annex)
PacketResult Handle_QSaveRegisterState(StringExtractorGDBRemote &packet)
std::recursive_mutex m_debugged_process_mutex
uint32_t GetNextSavedRegistersID()
Communication m_stdio_communication
llvm::Expected< std::unique_ptr< llvm::MemoryBuffer > > BuildTargetXml()
PacketResult Handle_I(StringExtractorGDBRemote &packet)
void RegisterPacketHandler(StringExtractorGDBRemote::ServerPacketType packet_type, PacketHandler handler)
PacketResult SendErrorResponse(const Status &error)
PacketResult SendIllFormedResponse(const StringExtractorGDBRemote &packet, const char *error_message)
PacketResult GetPacketAndSendResponse(Timeout< std::micro > timeout, Status &error, bool &interrupt, bool &quit)
PacketResult SendJSONResponse(const llvm::json::Value &value)
Serialize and send a JSON object response.
PacketResult SendUnimplementedResponse(const char *packet)
PacketResult SendOKResponse()
PacketResult SendNotificationPacketNoLock(llvm::StringRef notify_type, std::deque< std::string > &queue, llvm::StringRef payload)
PacketResult SendPacketNoLock(llvm::StringRef payload)
#define LLDB_REGNUM_GENERIC_RA
#define LLDB_REGNUM_GENERIC_ARG8
#define LLDB_INVALID_SIGNAL_NUMBER
#define LLDB_INVALID_THREAD_ID
#define LLDB_REGNUM_GENERIC_ARG6
#define LLDB_REGNUM_GENERIC_SP
#define LLDB_REGNUM_GENERIC_ARG4
#define LLDB_REGNUM_GENERIC_ARG3
#define LLDB_REGNUM_GENERIC_ARG1
#define LLDB_REGNUM_GENERIC_ARG7
#define LLDB_REGNUM_GENERIC_FLAGS
#define LLDB_INVALID_ADDRESS
#define LLDB_INVALID_REGNUM
#define LLDB_REGNUM_GENERIC_TP
#define LLDB_INVALID_PROCESS_ID
#define LLDB_REGNUM_GENERIC_ARG2
#define LLDB_REGNUM_GENERIC_PC
#define LLDB_REGNUM_GENERIC_FP
#define LLDB_REGNUM_GENERIC_ARG5
lldb::ByteOrder InlHostByteOrder()
std::string LLGSArgToURL(llvm::StringRef url_arg, bool reverse_connect)
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.
const char * StateAsCString(lldb::StateType state)
Converts a StateType to a C string.
std::vector< ProcessInstanceInfo > ProcessInstanceInfoList
std::shared_ptr< lldb_private::IOObject > IOObjectSP
ConnectionStatus
Connection Status Types.
@ eConnectionStatusError
Check GetError() for details.
@ eConnectionStatusInterrupted
Interrupted read.
@ eConnectionStatusTimedOut
Request timed out.
@ eConnectionStatusEndOfFile
End-of-file encountered.
@ eConnectionStatusSuccess
Success.
@ eConnectionStatusLostConnection
Lost connection while connected to a valid connection.
@ eConnectionStatusNoConnection
No connection.
@ eFormatCString
NULL terminated C strings.
@ eFormatCharArray
Print characters with no single quotes, used for character arrays that can contain non printable char...
@ eFormatInstruction
Disassemble an opcode.
@ eFormatVoid
Do not print this.
@ eFormatComplex
Floating point complex type.
@ eFormatHexFloat
ISO C99 hex float string.
@ eFormatOSType
OS character codes encoded into an integer 'PICT' 'text' etc...
@ eFormatAddressInfo
Describe what an address points to (func + offset with file/line, symbol + offset,...
@ eFormatCharPrintable
Only printable characters, '.' if not printable.
@ eFormatComplexInteger
Integer complex type.
@ eFormatFloat128
Disambiguate between 128-bit long double (which uses eFormatFloat) and __float128 (which uses eFormat...
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.
@ eStateLaunching
Process is in the process of launching.
@ eStateAttaching
Process is currently trying to attach.
@ 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.
@ eEncodingVector
vector registers
@ eEncodingUint
unsigned integer
@ eEncodingSint
signed integer
ByteOrder
Byte ordering definitions.
std::shared_ptr< lldb_private::DataBuffer > DataBufferSP
std::shared_ptr< lldb_private::WritableDataBuffer > WritableDataBufferSP
StopReason
Thread stop reasons.
@ eStopReasonInstrumentation
@ eStopReasonPlanComplete
@ eStopReasonHistoryBoundary
@ eStopReasonExec
Program was re-exec'ed.
@ eStopReasonInterrupt
Thread requested interrupt.
@ eStopReasonProcessorTrace
@ eStopReasonThreadExiting
@ eRegisterKindGeneric
insn ptr reg, stack ptr reg, etc not specific to any particular target
@ eRegisterKindDWARF
the register numbers seen DWARF
@ eRegisterKindEHFrame
the register numbers seen in eh_frame
BaseType GetRangeBase() const
SizeType GetByteSize() const
Every register is described in detail including its name, alternate name (optional),...
lldb::Encoding encoding
Encoding of the register bits.
const char * alt_name
Alternate name of this register, can be NULL.
uint32_t * value_regs
List of registers (terminated with LLDB_INVALID_REGNUM).
uint32_t byte_offset
The byte offset in the register context data where this register's value is found.
uint32_t byte_size
Size in bytes of the register.
uint32_t kinds[lldb::kNumRegisterKinds]
Holds all of the various register numbers for all register kinds.
const RegisterFlags * flags_type
If not nullptr, a type defined by XML descriptions.
const char * name
Name of this register, can't be NULL.
lldb::Format format
Default display format.
uint32_t * invalidate_regs
List of registers (terminated with LLDB_INVALID_REGNUM).
struct lldb_private::ThreadStopInfo::@116236113001137253323017204263037302160273237376::@034237007264067231263360140073224264215170222231 exception
struct lldb_private::ThreadStopInfo::@116236113001137253323017204263037302160273237376::@075376350020015165106375110034031012204332263127 fork
union lldb_private::ThreadStopInfo::@116236113001137253323017204263037302160273237376 details
static std::optional< URI > Parse(llvm::StringRef uri)