Duckstation/src/duckstation-qt/gdbconnection.cpp

101 lines
3 KiB
C++
Raw Normal View History

// SPDX-FileCopyrightText: 2019-2022 Connor McLaughlin <stenzek@gmail.com> and contributors.
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
2020-12-17 17:32:29 +00:00
#include "gdbconnection.h"
#include "common/log.h"
#include "core/gdb_protocol.h"
#include "qthost.h"
2020-12-17 17:32:29 +00:00
Log_SetChannel(GDBConnection);
2023-02-25 16:27:03 +00:00
GDBConnection::GDBConnection(GDBServer* parent, intptr_t descriptor) : QTcpSocket(parent), m_descriptor(descriptor)
2020-12-17 17:32:29 +00:00
{
2023-02-25 16:27:03 +00:00
if (!setSocketDescriptor(descriptor))
{
2023-02-25 16:27:03 +00:00
Log_ErrorPrintf("(%" PRIdPTR ") Failed to set socket descriptor: %s", descriptor,
errorString().toUtf8().constData());
deleteLater();
return;
2020-12-17 17:32:29 +00:00
}
2023-02-25 16:27:03 +00:00
connect(g_emu_thread, &EmuThread::systemPaused, this, &GDBConnection::onEmulationPaused);
connect(g_emu_thread, &EmuThread::systemResumed, this, &GDBConnection::onEmulationResumed);
connect(this, &QTcpSocket::readyRead, this, &GDBConnection::receivedData);
connect(this, &QTcpSocket::disconnected, this, &GDBConnection::gotDisconnected);
Log_InfoPrintf("(%" PRIdPTR ") Client connected", m_descriptor);
m_seen_resume = System::IsPaused();
g_emu_thread->setSystemPaused(true);
2020-12-17 17:32:29 +00:00
}
void GDBConnection::gotDisconnected()
{
2023-02-25 16:27:03 +00:00
Log_InfoPrintf("(%" PRIdPTR ") Client disconnected", m_descriptor);
deleteLater();
2020-12-17 17:32:29 +00:00
}
void GDBConnection::receivedData()
{
qint64 bytesRead;
char buffer[256];
2023-02-25 16:27:03 +00:00
while ((bytesRead = read(buffer, sizeof(buffer))) > 0)
{
for (char c : std::string_view(buffer, bytesRead))
{
2020-12-17 17:32:29 +00:00
m_readBuffer.push_back(c);
if (GDBProtocol::IsPacketInterrupt(m_readBuffer))
{
2023-02-25 16:27:03 +00:00
Log_DebugPrintf("(%" PRIdPTR ") > Interrupt request", m_descriptor);
g_emu_thread->setSystemPaused(true);
2020-12-17 17:32:29 +00:00
m_readBuffer.erase();
}
else if (GDBProtocol::IsPacketContinue(m_readBuffer))
{
2023-02-25 16:27:03 +00:00
Log_DebugPrintf("(%" PRIdPTR ") > Continue request", m_descriptor);
g_emu_thread->setSystemPaused(false);
2020-12-17 17:32:29 +00:00
m_readBuffer.erase();
}
else if (GDBProtocol::IsPacketComplete(m_readBuffer))
{
2023-02-25 16:27:03 +00:00
Log_DebugPrintf("(%" PRIdPTR ") > %s", m_descriptor, m_readBuffer.c_str());
2020-12-17 17:32:29 +00:00
writePacket(GDBProtocol::ProcessPacket(m_readBuffer));
m_readBuffer.erase();
}
}
}
if (bytesRead == -1)
{
2023-02-25 16:27:03 +00:00
Log_ErrorPrintf("(%" PRIdPTR ") Failed to read from socket: %s", m_descriptor,
errorString().toUtf8().constData());
2020-12-17 17:32:29 +00:00
}
}
void GDBConnection::onEmulationPaused()
2020-12-17 17:32:29 +00:00
{
if (m_seen_resume)
{
m_seen_resume = false;
// Generate a stop reply packet, insert '?' command to generate it.
writePacket(GDBProtocol::ProcessPacket("$?#3f"));
2020-12-17 17:32:29 +00:00
}
}
void GDBConnection::onEmulationResumed()
{
m_seen_resume = true;
// Send ack, in case GDB sent a continue request.
writePacket("+");
}
2020-12-17 17:32:29 +00:00
void GDBConnection::writePacket(std::string_view packet)
{
2023-09-30 05:44:14 +00:00
Log_DebugPrintf("(%" PRIdPTR ") < %.*s", m_descriptor, static_cast<int>(packet.length()), packet.data());
2023-02-25 16:27:03 +00:00
if (write(packet.data(), packet.length()) == -1)
{
2023-02-25 16:27:03 +00:00
Log_ErrorPrintf("(%" PRIdPTR ") Failed to write to socket: %s", m_descriptor,
errorString().toUtf8().constData());
2020-12-17 17:32:29 +00:00
}
}