//===-- GDBRemoteCommunication.h --------------------------------*- C++ -*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #ifndef LLDB_SOURCE_PLUGINS_PROCESS_GDB_REMOTE_GDBREMOTECOMMUNICATION_H #define LLDB_SOURCE_PLUGINS_PROCESS_GDB_REMOTE_GDBREMOTECOMMUNICATION_H #include "GDBRemoteCommunicationHistory.h" #include #include #include #include #include #include #include "lldb/Core/Communication.h" #include "lldb/Host/Config.h" #include "lldb/Host/HostThread.h" #include "lldb/Utility/Args.h" #include "lldb/Utility/Listener.h" #include "lldb/Utility/Predicate.h" #include "lldb/Utility/StringExtractorGDBRemote.h" #include "lldb/lldb-public.h" namespace lldb_private { namespace repro { class PacketRecorder; } namespace process_gdb_remote { enum GDBStoppointType { eStoppointInvalid = -1, eBreakpointSoftware = 0, eBreakpointHardware, eWatchpointWrite, eWatchpointRead, eWatchpointReadWrite }; enum class CompressionType { None = 0, // no compression ZlibDeflate, // zlib's deflate compression scheme, requires zlib or Apple's // libcompression LZFSE, // an Apple compression scheme, requires Apple's libcompression LZ4, // lz compression - called "lz4 raw" in libcompression terms, compat with // https://code.google.com/p/lz4/ LZMA, // Lempel–Ziv–Markov chain algorithm }; // Data included in the vFile:fstat packet. // https://sourceware.org/gdb/onlinedocs/gdb/struct-stat.html#struct-stat struct GDBRemoteFStatData { llvm::support::ubig32_t gdb_st_dev; llvm::support::ubig32_t gdb_st_ino; llvm::support::ubig32_t gdb_st_mode; llvm::support::ubig32_t gdb_st_nlink; llvm::support::ubig32_t gdb_st_uid; llvm::support::ubig32_t gdb_st_gid; llvm::support::ubig32_t gdb_st_rdev; llvm::support::ubig64_t gdb_st_size; llvm::support::ubig64_t gdb_st_blksize; llvm::support::ubig64_t gdb_st_blocks; llvm::support::ubig32_t gdb_st_atime; llvm::support::ubig32_t gdb_st_mtime; llvm::support::ubig32_t gdb_st_ctime; }; static_assert(sizeof(GDBRemoteFStatData) == 64, "size of GDBRemoteFStatData is not 64"); enum GDBErrno { #define HANDLE_ERRNO(name, value) GDB_##name = value, #include "Plugins/Process/gdb-remote/GDBRemoteErrno.def" GDB_EUNKNOWN = 9999 }; class ProcessGDBRemote; class GDBRemoteCommunication : public Communication { public: enum class PacketType { Invalid = 0, Standard, Notify }; enum class PacketResult { Success = 0, // Success ErrorSendFailed, // Status sending the packet ErrorSendAck, // Didn't get an ack back after sending a packet ErrorReplyFailed, // Status getting the reply ErrorReplyTimeout, // Timed out waiting for reply ErrorReplyInvalid, // Got a reply but it wasn't valid for the packet that // was sent ErrorReplyAck, // Sending reply ack failed ErrorDisconnected, // We were disconnected ErrorNoSequenceLock // We couldn't get the sequence lock for a multi-packet // request }; // Class to change the timeout for a given scope and restore it to the // original value when the // created ScopedTimeout object got out of scope class ScopedTimeout { public: ScopedTimeout(GDBRemoteCommunication &gdb_comm, std::chrono::seconds timeout); ~ScopedTimeout(); private: GDBRemoteCommunication &m_gdb_comm; std::chrono::seconds m_saved_timeout; // Don't ever reduce the timeout for a packet, only increase it. If the // requested timeout if less than the current timeout, we don't set it // and won't need to restore it. bool m_timeout_modified; }; GDBRemoteCommunication(); ~GDBRemoteCommunication() override; PacketResult GetAck(); size_t SendAck(); size_t SendNack(); char CalculcateChecksum(llvm::StringRef payload); PacketType CheckForPacket(const uint8_t *src, size_t src_len, StringExtractorGDBRemote &packet); bool GetSendAcks() { return m_send_acks; } // Set the global packet timeout. // // For clients, this is the timeout that gets used when sending // packets and waiting for responses. For servers, this is used when waiting // for ACKs. std::chrono::seconds SetPacketTimeout(std::chrono::seconds packet_timeout) { const auto old_packet_timeout = m_packet_timeout; m_packet_timeout = packet_timeout; return old_packet_timeout; } std::chrono::seconds GetPacketTimeout() const { return m_packet_timeout; } // Start a debugserver instance on the current host using the // supplied connection URL. Status StartDebugserverProcess( const char *url, Platform *platform, // If non nullptr, then check with the platform for // the GDB server binary if it can't be located ProcessLaunchInfo &launch_info, uint16_t *port, const Args *inferior_args, int pass_comm_fd); // Communication file descriptor to pass during // fork/exec to avoid having to connect/accept void DumpHistory(Stream &strm); void SetPacketRecorder(repro::PacketRecorder *recorder); static llvm::Error ConnectLocally(GDBRemoteCommunication &client, GDBRemoteCommunication &server); /// Expand GDB run-length encoding. static std::string ExpandRLE(std::string); protected: std::chrono::seconds m_packet_timeout; uint32_t m_echo_number; LazyBool m_supports_qEcho; GDBRemoteCommunicationHistory m_history; bool m_send_acks; bool m_is_platform; // Set to true if this class represents a platform, // false if this class represents a debug session for // a single process std::string m_bytes; std::recursive_mutex m_bytes_mutex; CompressionType m_compression_type; PacketResult SendPacketNoLock(llvm::StringRef payload); PacketResult SendNotificationPacketNoLock(llvm::StringRef notify_type, std::deque& queue, llvm::StringRef payload); PacketResult SendRawPacketNoLock(llvm::StringRef payload, bool skip_ack = false); PacketResult ReadPacket(StringExtractorGDBRemote &response, Timeout timeout, bool sync_on_timeout); PacketResult WaitForPacketNoLock(StringExtractorGDBRemote &response, Timeout timeout, bool sync_on_timeout); bool CompressionIsEnabled() { return m_compression_type != CompressionType::None; } // If compression is enabled, decompress the packet in m_bytes and update // m_bytes with the uncompressed version. // Returns 'true' packet was decompressed and m_bytes is the now-decompressed // text. // Returns 'false' if unable to decompress or if the checksum was invalid. // // NB: Once the packet has been decompressed, checksum cannot be computed // based // on m_bytes. The checksum was for the compressed packet. bool DecompressPacket(); Status StartListenThread(const char *hostname = "127.0.0.1", uint16_t port = 0); bool JoinListenThread(); lldb::thread_result_t ListenThread(); private: // Promise used to grab the port number from listening thread std::promise m_port_promise; HostThread m_listen_thread; std::string m_listen_url; #if defined(HAVE_LIBCOMPRESSION) CompressionType m_decompression_scratch_type = CompressionType::None; void *m_decompression_scratch = nullptr; #endif GDBRemoteCommunication(const GDBRemoteCommunication &) = delete; const GDBRemoteCommunication & operator=(const GDBRemoteCommunication &) = delete; }; } // namespace process_gdb_remote } // namespace lldb_private namespace llvm { template <> struct format_provider< lldb_private::process_gdb_remote::GDBRemoteCommunication::PacketResult> { static void format(const lldb_private::process_gdb_remote:: GDBRemoteCommunication::PacketResult &state, raw_ostream &Stream, StringRef Style); }; } // namespace llvm #endif // LLDB_SOURCE_PLUGINS_PROCESS_GDB_REMOTE_GDBREMOTECOMMUNICATION_H