forked from OSchip/llvm-project
161 lines
5.0 KiB
C++
161 lines
5.0 KiB
C++
//===-- GDBRemote.cpp -----------------------------------------------------===//
|
|
//
|
|
// 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "lldb/Utility/GDBRemote.h"
|
|
|
|
#include "lldb/Utility/Flags.h"
|
|
#include "lldb/Utility/Stream.h"
|
|
|
|
#include <cstdio>
|
|
|
|
using namespace lldb;
|
|
using namespace lldb_private::repro;
|
|
using namespace lldb_private;
|
|
using namespace llvm;
|
|
|
|
StreamGDBRemote::StreamGDBRemote() : StreamString() {}
|
|
|
|
StreamGDBRemote::StreamGDBRemote(uint32_t flags, uint32_t addr_size,
|
|
ByteOrder byte_order)
|
|
: StreamString(flags, addr_size, byte_order) {}
|
|
|
|
StreamGDBRemote::~StreamGDBRemote() = default;
|
|
|
|
int StreamGDBRemote::PutEscapedBytes(const void *s, size_t src_len) {
|
|
int bytes_written = 0;
|
|
const uint8_t *src = static_cast<const uint8_t *>(s);
|
|
bool binary_is_set = m_flags.Test(eBinary);
|
|
m_flags.Clear(eBinary);
|
|
while (src_len) {
|
|
uint8_t byte = *src;
|
|
src++;
|
|
src_len--;
|
|
if (byte == 0x23 || byte == 0x24 || byte == 0x7d || byte == 0x2a) {
|
|
bytes_written += PutChar(0x7d);
|
|
byte ^= 0x20;
|
|
}
|
|
bytes_written += PutChar(byte);
|
|
};
|
|
if (binary_is_set)
|
|
m_flags.Set(eBinary);
|
|
return bytes_written;
|
|
}
|
|
|
|
llvm::StringRef GDBRemotePacket::GetTypeStr() const {
|
|
switch (type) {
|
|
case GDBRemotePacket::ePacketTypeSend:
|
|
return "send";
|
|
case GDBRemotePacket::ePacketTypeRecv:
|
|
return "read";
|
|
case GDBRemotePacket::ePacketTypeInvalid:
|
|
return "invalid";
|
|
}
|
|
llvm_unreachable("All enum cases should be handled");
|
|
}
|
|
|
|
void GDBRemotePacket::Dump(Stream &strm) const {
|
|
strm.Printf("tid=0x%4.4" PRIx64 " <%4u> %s packet: %s\n", tid,
|
|
bytes_transmitted, GetTypeStr().data(), packet.data.c_str());
|
|
}
|
|
|
|
void yaml::ScalarEnumerationTraits<GDBRemotePacket::Type>::enumeration(
|
|
IO &io, GDBRemotePacket::Type &value) {
|
|
io.enumCase(value, "Invalid", GDBRemotePacket::ePacketTypeInvalid);
|
|
io.enumCase(value, "Send", GDBRemotePacket::ePacketTypeSend);
|
|
io.enumCase(value, "Recv", GDBRemotePacket::ePacketTypeRecv);
|
|
}
|
|
|
|
void yaml::ScalarTraits<GDBRemotePacket::BinaryData>::output(
|
|
const GDBRemotePacket::BinaryData &Val, void *, raw_ostream &Out) {
|
|
Out << toHex(Val.data);
|
|
}
|
|
|
|
StringRef yaml::ScalarTraits<GDBRemotePacket::BinaryData>::input(
|
|
StringRef Scalar, void *, GDBRemotePacket::BinaryData &Val) {
|
|
Val.data = fromHex(Scalar);
|
|
return {};
|
|
}
|
|
|
|
void yaml::MappingTraits<GDBRemotePacket>::mapping(IO &io,
|
|
GDBRemotePacket &Packet) {
|
|
io.mapRequired("packet", Packet.packet);
|
|
io.mapRequired("type", Packet.type);
|
|
io.mapRequired("bytes", Packet.bytes_transmitted);
|
|
io.mapRequired("index", Packet.packet_idx);
|
|
io.mapRequired("tid", Packet.tid);
|
|
}
|
|
|
|
StringRef
|
|
yaml::MappingTraits<GDBRemotePacket>::validate(IO &io,
|
|
GDBRemotePacket &Packet) {
|
|
return {};
|
|
}
|
|
|
|
void GDBRemoteProvider::Keep() {
|
|
std::vector<std::string> files;
|
|
for (auto &recorder : m_packet_recorders) {
|
|
files.push_back(recorder->GetFilename().GetPath());
|
|
}
|
|
|
|
FileSpec file = GetRoot().CopyByAppendingPathComponent(Info::file);
|
|
std::error_code ec;
|
|
llvm::raw_fd_ostream os(file.GetPath(), ec, llvm::sys::fs::OF_TextWithCRLF);
|
|
if (ec)
|
|
return;
|
|
yaml::Output yout(os);
|
|
yout << files;
|
|
}
|
|
|
|
void GDBRemoteProvider::Discard() { m_packet_recorders.clear(); }
|
|
|
|
llvm::Expected<std::unique_ptr<PacketRecorder>>
|
|
PacketRecorder::Create(const FileSpec &filename) {
|
|
std::error_code ec;
|
|
auto recorder = std::make_unique<PacketRecorder>(std::move(filename), ec);
|
|
if (ec)
|
|
return llvm::errorCodeToError(ec);
|
|
return std::move(recorder);
|
|
}
|
|
|
|
PacketRecorder *GDBRemoteProvider::GetNewPacketRecorder() {
|
|
std::size_t i = m_packet_recorders.size() + 1;
|
|
std::string filename = (llvm::Twine(Info::name) + llvm::Twine("-") +
|
|
llvm::Twine(i) + llvm::Twine(".yaml"))
|
|
.str();
|
|
auto recorder_or_error =
|
|
PacketRecorder::Create(GetRoot().CopyByAppendingPathComponent(filename));
|
|
if (!recorder_or_error) {
|
|
llvm::consumeError(recorder_or_error.takeError());
|
|
return nullptr;
|
|
}
|
|
|
|
m_packet_recorders.push_back(std::move(*recorder_or_error));
|
|
return m_packet_recorders.back().get();
|
|
}
|
|
|
|
void PacketRecorder::Record(const GDBRemotePacket &packet) {
|
|
if (!m_record)
|
|
return;
|
|
yaml::Output yout(m_os);
|
|
yout << const_cast<GDBRemotePacket &>(packet);
|
|
m_os.flush();
|
|
}
|
|
|
|
llvm::raw_ostream *GDBRemoteProvider::GetHistoryStream() {
|
|
FileSpec history_file = GetRoot().CopyByAppendingPathComponent(Info::file);
|
|
|
|
std::error_code EC;
|
|
m_stream_up = std::make_unique<raw_fd_ostream>(
|
|
history_file.GetPath(), EC, sys::fs::OpenFlags::OF_TextWithCRLF);
|
|
return m_stream_up.get();
|
|
}
|
|
|
|
char GDBRemoteProvider::ID = 0;
|
|
const char *GDBRemoteProvider::Info::file = "gdb-remote.yaml";
|
|
const char *GDBRemoteProvider::Info::name = "gdb-remote";
|