12#include "lldb/Host/Config.h"
26#include "llvm/ADT/SmallString.h"
27#include "llvm/ADT/StringRef.h"
28#include "llvm/Config/llvm-config.h"
29#include "llvm/Support/Error.h"
30#include "llvm/Support/ScopedPrinter.h"
37#if HAVE_LIBCOMPRESSION
38#include <compression.h>
52#ifdef LLDB_CONFIGURATION_DEBUG
68#if HAVE_LIBCOMPRESSION
69 if (m_decompression_scratch)
70 free (m_decompression_scratch);
77 for (
char c : payload)
80 return checksum & 255;
87 const size_t bytes_written =
WriteAll(&ch, 1, status,
nullptr);
88 LLDB_LOGF(log,
"<%4" PRIu64
"> send packet: %c", (uint64_t)bytes_written, ch);
97 const size_t bytes_written =
WriteAll(&ch, 1, status,
nullptr);
98 LLDB_LOGF(log,
"<%4" PRIu64
"> send packet: %c", (uint64_t)bytes_written, ch);
100 return bytes_written;
107 packet.
Write(payload.data(), payload.size());
110 std::string packet_str = std::string(packet.
GetString());
117 llvm::StringRef notify_type, std::deque<std::string> &queue,
118 llvm::StringRef payload) {
126 packet.
Write(notify_type.data(), notify_type.size());
128 packet.
Write(payload.data(), payload.size());
134 queue.push_back(payload.str());
142 if (delay.count() > 0)
143 std::this_thread::sleep_for(delay);
148 const char *packet_data = packet.data();
149 const size_t packet_length = packet.size();
150 size_t bytes_written =
WriteAll(packet_data, packet_length, status,
nullptr);
152 size_t binary_start_offset = 0;
153 if (strncmp(packet_data,
"$vFile:pwrite:", strlen(
"$vFile:pwrite:")) ==
155 const char *first_comma = strchr(packet_data,
',');
157 const char *second_comma = strchr(first_comma + 1,
',');
159 binary_start_offset = second_comma - packet_data + 1;
170 if (binary_start_offset) {
173 strm.
Printf(
"<%4" PRIu64
"> send packet: %.*s", (uint64_t)bytes_written,
174 (
int)binary_start_offset, packet_data);
177 for (p = (
const uint8_t *)packet_data + binary_start_offset; *p !=
'#';
179 strm.
Printf(
"\\x%2.2x", *p);
181 strm.
Printf(
"%*s", (
int)3, p);
184 LLDB_LOGF(log,
"<%4" PRIu64
"> send packet: %.*s",
185 (uint64_t)bytes_written, (
int)packet_length, packet_data);
188 m_history.AddPacket(packet.str(), packet_length,
191 if (bytes_written == packet_length) {
197 LLDB_LOGF(log,
"error: failed to send packet: %.*s", (
int)packet_length,
220 bool sync_on_timeout) {
238 bool sync_on_timeout) {
239 uint8_t buffer[8192];
248 bool timed_out =
false;
249 bool disconnected =
false;
252 size_t bytes_read =
Read(buffer,
sizeof(buffer), timeout, status, &
error);
255 "Read(buffer, sizeof(buffer), timeout = {0}, "
256 "status = {1}, error = {2}) => bytes_read = {3}",
260 if (bytes_read > 0) {
267 if (sync_on_timeout) {
290 bool sync_success =
false;
291 bool got_actual_response =
false;
293 char echo_packet[32];
294 int echo_packet_len = 0;
298 echo_packet_len = ::snprintf(echo_packet,
sizeof(echo_packet),
300 std::string regex_str =
"^";
301 regex_str += echo_packet;
306 ::snprintf(echo_packet,
sizeof(echo_packet),
"qC");
314 const uint32_t max_retries = 3;
315 uint32_t successful_responses = 0;
316 for (uint32_t i = 0; i < max_retries; ++i) {
321 ++successful_responses;
325 }
else if (successful_responses == 1) {
330 packet = echo_response;
331 got_actual_response =
true;
345 if (got_actual_response) {
389 size_t pkt_size =
m_bytes.size();
402 size_t hash_mark_idx =
m_bytes.find(
'#');
403 if (hash_mark_idx == std::string::npos)
405 if (hash_mark_idx + 2 >=
m_bytes.size())
408 if (!::isxdigit(
m_bytes[hash_mark_idx + 1]) ||
409 !::isxdigit(
m_bytes[hash_mark_idx + 2]))
412 size_t content_length =
415 size_t content_start = 2;
417 size_t checksum_idx =
425 size_t size_of_first_packet = hash_mark_idx + 3;
432 uint64_t decompressed_bufsize = ULONG_MAX;
434 size_t i = content_start;
435 while (i < hash_mark_idx && isdigit(
m_bytes[i]))
437 if (i < hash_mark_idx &&
m_bytes[i] ==
':') {
440 content_length = hash_mark_idx - content_start;
441 std::string bufsize_str(
m_bytes.data() + 2, i - 2 - 1);
443 decompressed_bufsize = ::strtoul(bufsize_str.c_str(),
nullptr, 10);
444 if (errno != 0 || decompressed_bufsize == ULONG_MAX) {
445 m_bytes.erase(0, size_of_first_packet);
452 char packet_checksum_cstr[3];
453 packet_checksum_cstr[0] =
m_bytes[checksum_idx];
454 packet_checksum_cstr[1] =
m_bytes[checksum_idx + 1];
455 packet_checksum_cstr[2] =
'\0';
456 long packet_checksum = strtol(packet_checksum_cstr,
nullptr, 16);
459 llvm::StringRef(
m_bytes).substr(1, hash_mark_idx - 1));
460 bool success = packet_checksum == actual_checksum;
463 "error: checksum mismatch: %.*s expected 0x%2.2x, got 0x%2.2x",
464 (
int)(pkt_size),
m_bytes.c_str(), (uint8_t)packet_checksum,
465 (uint8_t)actual_checksum);
470 m_bytes.erase(0, size_of_first_packet);
486 std::vector<uint8_t> unescaped_content;
487 unescaped_content.reserve(content_length);
488 size_t i = content_start;
489 while (i < hash_mark_idx) {
492 unescaped_content.push_back(
m_bytes[i] ^ 0x20);
494 unescaped_content.push_back(
m_bytes[i]);
499 uint8_t *decompressed_buffer =
nullptr;
500 size_t decompressed_bytes = 0;
502 if (decompressed_bufsize != ULONG_MAX) {
503 decompressed_buffer = (uint8_t *)malloc(decompressed_bufsize);
504 if (decompressed_buffer ==
nullptr) {
505 m_bytes.erase(0, size_of_first_packet);
510#if HAVE_LIBCOMPRESSION
515 compression_algorithm compression_type;
517 compression_type = COMPRESSION_LZFSE;
519 compression_type = COMPRESSION_ZLIB;
521 compression_type = COMPRESSION_LZ4_RAW;
523 compression_type = COMPRESSION_LZMA;
526 if (m_decompression_scratch) {
527 free (m_decompression_scratch);
528 m_decompression_scratch =
nullptr;
530 size_t scratchbuf_size = 0;
532 scratchbuf_size = compression_decode_scratch_buffer_size (COMPRESSION_LZFSE);
534 scratchbuf_size = compression_decode_scratch_buffer_size (COMPRESSION_LZ4_RAW);
536 scratchbuf_size = compression_decode_scratch_buffer_size (COMPRESSION_ZLIB);
539 compression_decode_scratch_buffer_size(COMPRESSION_LZMA);
540 if (scratchbuf_size > 0) {
541 m_decompression_scratch = (
void*) malloc (scratchbuf_size);
546 if (decompressed_bufsize != ULONG_MAX && decompressed_buffer !=
nullptr) {
547 decompressed_bytes = compression_decode_buffer(
548 decompressed_buffer, decompressed_bufsize,
549 (uint8_t *)unescaped_content.data(), unescaped_content.size(),
550 m_decompression_scratch, compression_type);
556 if (decompressed_bytes == 0 && decompressed_bufsize != ULONG_MAX &&
557 decompressed_buffer !=
nullptr &&
560 memset(&stream, 0,
sizeof(z_stream));
561 stream.next_in = (Bytef *)unescaped_content.data();
562 stream.avail_in = (uInt)unescaped_content.size();
564 stream.next_out = (Bytef *)decompressed_buffer;
565 stream.avail_out = decompressed_bufsize;
566 stream.total_out = 0;
567 stream.zalloc = Z_NULL;
568 stream.zfree = Z_NULL;
569 stream.opaque = Z_NULL;
571 if (inflateInit2(&stream, -15) == Z_OK) {
572 int status = inflate(&stream, Z_NO_FLUSH);
574 if (status == Z_STREAM_END) {
575 decompressed_bytes = stream.total_out;
581 if (decompressed_bytes == 0 || decompressed_buffer ==
nullptr) {
582 if (decompressed_buffer)
583 free(decompressed_buffer);
584 m_bytes.erase(0, size_of_first_packet);
588 std::string new_packet;
589 new_packet.reserve(decompressed_bytes + 6);
590 new_packet.push_back(
m_bytes[0]);
591 new_packet.append((
const char *)decompressed_buffer, decompressed_bytes);
592 new_packet.push_back(
'#');
595 llvm::StringRef((
const char *)decompressed_buffer, decompressed_bytes));
596 char decompressed_checksum_str[3];
597 snprintf(decompressed_checksum_str, 3,
"%02x", decompressed_checksum);
598 new_packet.append(decompressed_checksum_str);
600 new_packet.push_back(
'0');
601 new_packet.push_back(
'0');
604 m_bytes.replace(0, size_of_first_packet, new_packet.data(),
607 free(decompressed_buffer);
619 if (src && src_len > 0) {
621 LLDB_LOGF(log,
"GDBRemoteCommunication::%s adding %u bytes: %.*s",
622 __FUNCTION__, (uint32_t)src_len, (uint32_t)src_len, src);
624 m_bytes.append((
const char *)src, src_len);
627 bool isNotifyPacket =
false;
633 size_t content_start = 0;
634 size_t content_length = 0;
635 size_t total_length = 0;
636 size_t checksum_idx = std::string::npos;
639 size_t original_packet_size =
m_bytes.size();
651 content_length = total_length = 1;
655 isNotifyPacket =
true;
661 size_t hash_pos =
m_bytes.find(
'#');
662 if (hash_pos != std::string::npos) {
663 if (hash_pos + 2 <
m_bytes.size()) {
664 checksum_idx = hash_pos + 1;
669 content_length = hash_pos - 1;
675 content_length = std::string::npos;
686 const size_t bytes_len =
m_bytes.size();
689 for (idx = 1; !done && idx < bytes_len; ++idx) {
703 LLDB_LOGF(log,
"GDBRemoteCommunication::%s tossing %u junk bytes: '%.*s'",
704 __FUNCTION__, idx - 1, idx - 1,
m_bytes.c_str());
709 if (content_length == std::string::npos) {
712 }
else if (total_length > 0) {
715 assert(content_length <=
m_bytes.size());
716 assert(total_length <=
m_bytes.size());
717 assert(content_length <= total_length);
718 size_t content_end = content_start + content_length;
732 if (
m_bytes[0] ==
'$' && total_length > 4) {
733 for (
size_t i = 0; !binary && i < total_length; ++i) {
735 if (!llvm::isPrint(c) && !llvm::isSpace(c)) {
744 strm.
Printf(
"<%4" PRIu64
":%" PRIu64
"> read packet: %c",
745 (uint64_t)original_packet_size, (uint64_t)total_length,
748 strm.
Printf(
"<%4" PRIu64
"> read packet: %c",
749 (uint64_t)total_length,
m_bytes[0]);
750 for (
size_t i = content_start; i < content_end; ++i) {
756 const char escapee =
m_bytes[++i] ^ 0x20;
757 strm.
Printf(
"%2.2x", escapee);
759 strm.
Printf(
"%2.2x", (uint8_t)ch);
768 LLDB_LOGF(log,
"<%4" PRIu64
":%" PRIu64
"> read packet: %.*s",
769 (uint64_t)original_packet_size, (uint64_t)total_length,
770 (
int)(total_length),
m_bytes.c_str());
772 LLDB_LOGF(log,
"<%4" PRIu64
"> read packet: %.*s",
773 (uint64_t)total_length, (
int)(total_length),
783 auto maybe_packet_str =
785 if (!maybe_packet_str) {
786 m_bytes.erase(0, total_length);
793 assert(checksum_idx <
m_bytes.size());
794 if (::isxdigit(
m_bytes[checksum_idx + 0]) ||
795 ::isxdigit(
m_bytes[checksum_idx + 1])) {
797 const char *packet_checksum_cstr = &
m_bytes[checksum_idx];
798 char packet_checksum = strtol(packet_checksum_cstr,
nullptr, 16);
800 llvm::StringRef(
m_bytes).slice(content_start, content_end));
801 success = packet_checksum == actual_checksum;
804 "error: checksum mismatch: %.*s expected 0x%2.2x, "
806 (
int)(total_length),
m_bytes.c_str(),
807 (uint8_t)packet_checksum, (uint8_t)actual_checksum);
817 LLDB_LOGF(log,
"error: invalid checksum in packet: '%s'\n",
822 m_bytes.erase(0, total_length);
836 std::variant<llvm::StringRef, shared_fd_t> comm,
842#if !defined(__APPLE__)
853 llvm::SmallString<128> named_pipe_path;
862 if (
shared_fd_t *comm_fd = std::get_if<shared_fd_t>(&comm)) {
863 LLDB_LOG(log,
"debugserver communicates over fd {0}", comm_fd);
865 debugserver_args.
AppendArgument(llvm::formatv(
"--fd={0}", *comm_fd).str());
870 llvm::StringRef url = std::get<llvm::StringRef>(comm);
871 LLDB_LOG(log,
"debugserver listens on: {0}", url);
874#if defined(__APPLE__)
899 std::string env_debugserver_log_file =
900 host_env.lookup(
"LLDB_DEBUGSERVER_LOG_FILE");
901 if (!env_debugserver_log_file.empty()) {
903 llvm::formatv(
"--log-file={0}", env_debugserver_log_file).str());
906#if defined(__APPLE__)
907 const char *env_debugserver_log_flags = getenv(
"LLDB_DEBUGSERVER_LOG_FLAGS");
908 if (env_debugserver_log_flags) {
910 llvm::formatv(
"--log-flags={0}", env_debugserver_log_flags).str());
913 std::string env_debugserver_log_channels =
914 host_env.lookup(
"LLDB_SERVER_LOG_CHANNELS");
915 if (!env_debugserver_log_channels.empty()) {
917 llvm::formatv(
"--log-channels={0}", env_debugserver_log_channels)
924 uint32_t env_var_index = 1;
927 char env_var_name[64];
928 snprintf(env_var_name,
sizeof(env_var_name),
929 "LLDB_DEBUGSERVER_EXTRA_ARG_%" PRIu32, env_var_index++);
930 std::string extra_arg = host_env.lookup(env_var_name);
931 has_env_var = !extra_arg.empty();
936 "GDBRemoteCommunication::%s adding env var %s contents "
937 "to stub command line (%s)",
938 __FUNCTION__, env_var_name, extra_arg.c_str());
940 }
while (has_env_var);
963 launch_info.
Dump(string_stream, platform);
964 LLDB_LOG(log,
"launch info for gdb-remote stub:\n{0}",
972 if (std::holds_alternative<shared_fd_t>(comm))
976 if (named_pipe_path.size() > 0) {
979 LLDB_LOG(log,
"failed to open named pipe {0} for reading: {1}",
980 named_pipe_path,
error);
989 while (
error.Success()) {
991 if (llvm::Expected<size_t> num_bytes =
992 socket_pipe.
Read(buf, std::size(buf), std::chrono::seconds(10))) {
1000 LLDB_LOG(log,
"failed to synchronize on pipe {0}: {1}", named_pipe_path,
1003 socket_pipe.
Close();
1005 if (named_pipe_path.size() > 0) {
1007 LLDB_LOG(log,
"failed to delete pipe {0}: {1}", named_pipe_path, err);
1021 if (curr_timeout < timeout) {
1033void llvm::format_provider<GDBRemoteCommunication::PacketResult>::format(
1043 Stream <<
"ErrorSendFailed";
1046 Stream <<
"ErrorSendAck";
1049 Stream <<
"ErrorReplyFailed";
1052 Stream <<
"ErrorReplyTimeout";
1055 Stream <<
"ErrorReplyInvalid";
1058 Stream <<
"ErrorReplyAck";
1061 Stream <<
"ErrorDisconnected";
1064 Stream <<
"ErrorNoSequenceLock";
1069std::optional<std::string>
1072 std::string decoded;
1073 decoded.reserve(packet.size());
1074 for (std::string::const_iterator c = packet.begin(); c != packet.end(); ++c) {
1076 if (decoded.empty())
1077 return std::nullopt;
1080 char char_to_repeat = decoded.back();
1082 if (++c == packet.end())
1083 return std::nullopt;
1084 int repeat_count = *c + 3 -
' ';
1087 for (
int i = 0; i < repeat_count; ++i)
1088 decoded.push_back(char_to_repeat);
1089 }
else if (*c == 0x7d) {
1092 if (++c == packet.end())
1093 return std::nullopt;
1094 char escapee = *c ^ 0x20;
1095 decoded.push_back(escapee);
1097 decoded.push_back(*c);
static llvm::raw_ostream & error(Stream &strm)
#define LLDB_LOG(log,...)
The LLDB_LOG* macros defined below are the way to emit log messages.
#define LLDB_LOGF(log,...)
#define LLDB_LOG_VERBOSE(log,...)
A command line argument class.
void AppendArguments(const Args &rhs)
size_t GetArgumentCount() const
Gets the number of arguments left in this command object.
void AppendArgument(llvm::StringRef arg_str, char quote_char='\0')
Appends a new argument to the end of the list argument list.
virtual size_t Read(void *dst, size_t dst_len, const Timeout< std::micro > &timeout, lldb::ConnectionStatus &status, Status *error_ptr)
Read bytes from the current connection.
bool IsConnected() const
Check if the connection is valid.
size_t WriteAll(const void *src, size_t src_len, lldb::ConnectionStatus &status, Status *error_ptr)
Repeatedly attempt writing until either src_len bytes are written or a permanent failure occurs.
Communication()
Construct the Communication object.
virtual lldb::ConnectionStatus Disconnect(Status *error_ptr=nullptr)
Disconnect the communications connection if one is currently connected.
static std::string ConnectionStatusAsString(lldb::ConnectionStatus status)
static Status LaunchProcess(ProcessLaunchInfo &launch_info)
Launch the process specified in launch_info.
static Environment GetEnvironment()
void PutString(llvm::StringRef str)
lldb::pipe_t GetWritePipe() const override
void CloseWriteFileDescriptor() override
Status CreateNew() override
bool CanWrite() const override
bool CanRead() const override
llvm::Expected< size_t > Read(void *buf, size_t size, const Timeout< std::micro > &timeout=std::nullopt) override
Status Delete(llvm::StringRef name) override
Status OpenAsReader(llvm::StringRef name) override
Status CreateWithUniqueName(llvm::StringRef prefix, llvm::SmallVectorImpl< char > &name) override
void Dump(Stream &s, Platform *platform) const
Environment & GetEnvironment()
bool AppendSuppressFileAction(int fd, bool read, bool write)
bool AppendCloseFileAction(int fd)
bool AppendDuplicateFileAction(int fd, int dup_fd)
bool GetLaunchInSeparateProcessGroup() const
bool Execute(llvm::StringRef string, llvm::SmallVectorImpl< llvm::StringRef > *matches=nullptr) const
Execute a regular expression match using the compiled regular expression that is already in this obje...
static const shared_fd_t kInvalidFD
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.
const char * GetData() const
llvm::StringRef GetString() const
A stream class that can stream formatted output to a file.
size_t Write(const void *src, size_t src_len)
Output character bytes to the stream.
size_t size_t PutHex8(uint8_t uvalue)
Append an uint8_t value in the hexadecimal format to the stream.
size_t Printf(const char *format,...) __attribute__((format(printf
Output printf formatted output to the stream.
std::chrono::seconds m_saved_timeout
GDBRemoteCommunication & m_gdb_comm
ScopedTimeout(GDBRemoteCommunication &gdb_comm, std::chrono::seconds timeout)
PacketResult ReadPacket(StringExtractorGDBRemote &response, Timeout< std::micro > timeout, bool sync_on_timeout)
char CalculcateChecksum(llvm::StringRef payload)
~GDBRemoteCommunication() override
PacketResult SendNotificationPacketNoLock(llvm::StringRef notify_type, std::deque< std::string > &queue, llvm::StringRef payload)
std::recursive_mutex m_bytes_mutex
PacketResult WaitForPacketNoLock(StringExtractorGDBRemote &response, Timeout< std::micro > timeout, bool sync_on_timeout)
void DumpHistory(Stream &strm)
bool CompressionIsEnabled()
LazyBool m_supports_qEcho
PacketResult SendRawPacketNoLock(llvm::StringRef payload, bool skip_ack=false)
CompressionType m_compression_type
static Status StartDebugserverProcess(std::variant< llvm::StringRef, shared_fd_t > comm, ProcessLaunchInfo &launch_info, const Args *inferior_args)
PacketResult SendPacketNoLock(llvm::StringRef payload)
static std::optional< std::string > ExpandRLE(std::string)
Expand GDB run-length encoding.
std::chrono::seconds GetPacketTimeout() const
std::chrono::seconds m_packet_timeout
PacketType CheckForPacket(const uint8_t *src, size_t src_len, StringExtractorGDBRemote &packet)
GDBRemoteCommunicationHistory m_history
static std::chrono::milliseconds GetPacketTestDelay()
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.
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.