#include "content/providers/gdb_provider.hpp" #include #include #include #include #include namespace hex::plugin::builtin::prv { using namespace std::chrono_literals; namespace gdb { namespace { u8 calculateChecksum(const std::string &data) { u64 checksum = 0; for (const auto &c : data) checksum += c; return checksum & 0xFF; } std::string createPacket(const std::string &data) { return hex::format("${}#{:02x}", data, calculateChecksum(data)); } std::optional parsePacket(const std::string &packet) { if (packet.length() < 4) return std::nullopt; if (!packet.starts_with('$')) return std::nullopt; if (packet[packet.length() - 3] != '#') return std::nullopt; std::string data = packet.substr(1, packet.length() - 4); std::string checksum = packet.substr(packet.length() - 2, 2); if (checksum.length() != 2 || crypt::decode16(checksum)[0] != calculateChecksum(data)) return std::nullopt; return data; } } void sendAck(Socket &socket) { socket.writeString("+"); } std::vector readMemory(Socket &socket, u64 address, size_t size) { std::string packet = createPacket(hex::format("m{:X},{:X}", address, size)); socket.writeString(packet); auto receivedPacket = socket.readString(size * 2 + 4); if (receivedPacket.empty()) return {}; auto receivedData = parsePacket(receivedPacket); if (!receivedData.has_value()) return {}; if (receivedData->size() == 3 && receivedData->starts_with("E")) return {}; auto data = crypt::decode16(receivedData.value()); data.resize(size); return data; } void writeMemory(Socket &socket, u64 address, const void *buffer, size_t size) { std::vector bytes(size); std::memcpy(bytes.data(), buffer, size); std::string byteString = crypt::encode16(bytes); std::string packet = createPacket(hex::format("M{:X},{:X}:{}", address, size, byteString)); socket.writeString(packet); auto receivedPacket = socket.readString(3); } bool enableNoAckMode(Socket &socket) { socket.writeString(createPacket("QStartNoAckMode")); auto ack = socket.readString(1); if (ack.empty() || ack[0] != '+') return {}; auto receivedPacket = socket.readString(6); auto receivedData = parsePacket(receivedPacket); if (receivedData && *receivedData == "OK") { sendAck(socket); return true; } else { return false; } } } GDBProvider::GDBProvider() : Provider(), m_size(0xFFFF'FFFF) { } GDBProvider::~GDBProvider() { this->close(); } bool GDBProvider::isAvailable() const { return this->m_socket.isConnected(); } bool GDBProvider::isReadable() const { return this->m_socket.isConnected(); } bool GDBProvider::isWritable() const { return false; } bool GDBProvider::isResizable() const { return false; } bool GDBProvider::isSavable() const { return false; } void GDBProvider::read(u64 offset, void *buffer, size_t size, bool overlays) { if ((offset - this->getBaseAddress()) > (this->getActualSize() - size) || buffer == nullptr || size == 0) return; offset -= this->getBaseAddress(); u64 alignedOffset = offset - (offset % CacheLineSize); if (size <= CacheLineSize) { std::scoped_lock lock(this->m_cacheLock); const auto &cacheLine = std::find_if(this->m_cache.begin(), this->m_cache.end(), [&](auto &line) { return line.address == alignedOffset; }); if (cacheLine != this->m_cache.end()) { // Cache hit } else { // Cache miss this->m_cache.push_back({ alignedOffset, { 0 } }); } if (cacheLine != this->m_cache.end()) std::memcpy(buffer, &cacheLine->data[0] + (offset % CacheLineSize), size); } else { while (size > 0) { size_t readSize = std::min(size, CacheLineSize); this->readRaw(offset, buffer, readSize); size -= readSize; offset += readSize; } } for (u64 i = 0; i < size; i++) if (getPatches().contains(offset + i)) reinterpret_cast(buffer)[i] = getPatches()[offset + PageSize * this->m_currPage + i]; if (overlays) this->applyOverlays(offset, buffer, size); } void GDBProvider::write(u64 offset, const void *buffer, size_t size) { if ((offset - this->getBaseAddress()) > (this->getActualSize() - size) || buffer == nullptr || size == 0) return; offset -= this->getBaseAddress(); gdb::writeMemory(this->m_socket, offset, buffer, size); } void GDBProvider::readRaw(u64 offset, void *buffer, size_t size) { if ((offset - this->getBaseAddress()) > (this->getActualSize() - size) || buffer == nullptr || size == 0) return; auto data = gdb::readMemory(this->m_socket, offset, size); std::memcpy(buffer, &data[0], data.size()); } void GDBProvider::writeRaw(u64 offset, const void *buffer, size_t size) { if ((offset - this->getBaseAddress()) > (this->getActualSize() - size) || buffer == nullptr || size == 0) return; gdb::writeMemory(this->m_socket, offset, buffer, size); } void GDBProvider::save() { this->applyPatches(); } void GDBProvider::saveAs(const fs::path &path) { } size_t GDBProvider::getActualSize() const { return this->m_size; } std::string GDBProvider::getName() const { std::string address, port; if (!this->isConnected()) { address = "-"; port = "-"; } else { address = this->m_ipAddress; port = std::to_string(this->m_port); } return hex::format("hex.builtin.provider.gdb.name"_lang, address, port); } std::vector> GDBProvider::getDataInformation() const { return { {"hex.builtin.provider.gdb.server"_lang, hex::format("{}:{}", this->m_ipAddress, this->m_port)}, }; } bool GDBProvider::open() { this->m_socket.connect(this->m_ipAddress, this->m_port); if (!gdb::enableNoAckMode(this->m_socket)) { this->m_socket.disconnect(); return false; } if (this->m_socket.isConnected()) { this->m_cacheUpdateThread = std::thread([this]() { auto cacheLine = this->m_cache.begin(); while (this->isConnected()) { { std::scoped_lock lock(this->m_cacheLock); if (cacheLine != this->m_cache.end()) { auto data = gdb::readMemory(this->m_socket, cacheLine->address, 0x1000); while (this->m_cache.size() > 10) { this->m_cache.pop_front(); cacheLine = this->m_cache.begin(); } std::memcpy(cacheLine->data.data(), data.data(), data.size()); } cacheLine++; if (cacheLine == this->m_cache.end()) cacheLine = this->m_cache.begin(); } std::this_thread::sleep_for(100ms); } }); return true; } else { return false; } } void GDBProvider::close() { this->m_socket.disconnect(); if (this->m_cacheUpdateThread.joinable()) { this->m_cacheUpdateThread.join(); } } bool GDBProvider::isConnected() const { return this->m_socket.isConnected(); } void GDBProvider::drawLoadInterface() { ImGui::InputText("hex.builtin.provider.gdb.ip"_lang, this->m_ipAddress.data(), this->m_ipAddress.capacity(), ImGuiInputTextFlags_CallbackEdit, ImGui::UpdateStringSizeCallback, &this->m_ipAddress); ImGui::InputInt("hex.builtin.provider.gdb.port"_lang, &this->m_port, 0, 0); ImGui::Separator(); ImGui::TextUnformatted("0x"); ImGui::SameLine(); ImGui::InputScalar("hex.common.size"_lang, ImGuiDataType_U64, &this->m_size, nullptr, nullptr, "%llx", ImGuiInputTextFlags_CharsHexadecimal); if (this->m_port < 0) this->m_port = 0; else if (this->m_port > 0xFFFF) this->m_port = 0xFFFF; } }