mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2026-04-25 07:58:40 -05:00
Debugger/NetworkWidget: Add Triforce support
This commit is contained in:
parent
f657b32521
commit
2d9b3c4118
|
|
@ -55,6 +55,21 @@ bool IPv4PortRange::IsMatch(IPv4Port subject) const
|
|||
port_u16 >= first.GetPortValue() && port_u16 <= last.GetPortValue();
|
||||
}
|
||||
|
||||
std::string IPv4PortRange::ToString() const
|
||||
{
|
||||
const std::string ip_range = first.ip_address == last.ip_address ?
|
||||
Common::IPAddressToString(first.ip_address) :
|
||||
fmt::format("{}-{}", Common::IPAddressToString(first.ip_address),
|
||||
Common::IPAddressToString(last.ip_address));
|
||||
|
||||
if (first.port == 0)
|
||||
return ip_range;
|
||||
else if (first.port == last.port)
|
||||
return fmt::format("{}:{}", ip_range, first.GetPortValue());
|
||||
else
|
||||
return fmt::format("{}:{}-{}", ip_range, first.GetPortValue(), last.GetPortValue());
|
||||
}
|
||||
|
||||
std::string IPAddressToString(IPAddress ip_address)
|
||||
{
|
||||
return fmt::format("{}", fmt::join(ip_address, "."));
|
||||
|
|
|
|||
|
|
@ -284,6 +284,7 @@ struct IPv4PortRange
|
|||
IPv4Port last;
|
||||
|
||||
bool IsMatch(IPv4Port subject) const;
|
||||
std::string ToString() const;
|
||||
};
|
||||
|
||||
std::string IPAddressToString(IPAddress ip_address);
|
||||
|
|
|
|||
|
|
@ -16,7 +16,6 @@
|
|||
#include "Common/FileUtil.h"
|
||||
#include "Common/IOFile.h"
|
||||
#include "Common/Logging/Log.h"
|
||||
#include "Common/Network.h"
|
||||
#include "Common/ScopeGuard.h"
|
||||
|
||||
#include "Core/Boot/Boot.h"
|
||||
|
|
@ -216,7 +215,8 @@ static const std::unordered_map<u16, GameType> s_game_map = {
|
|||
|
||||
// Sockets FDs are required to go from 0 to 63.
|
||||
// Games use the FD as indexes so we have to workaround it.
|
||||
static SOCKET s_sockets[64];
|
||||
|
||||
static SOCKET s_sockets[SOCKET_FD_MAX];
|
||||
|
||||
// TODO: Verify this.
|
||||
static constexpr u32 FIRST_VALID_FD = 1;
|
||||
|
|
@ -533,51 +533,48 @@ std::optional<ParsedIPOverride> ParseIPOverride(std::string_view str)
|
|||
};
|
||||
}
|
||||
|
||||
struct IPAddressOverride
|
||||
// Caller should check if it matches first!
|
||||
Common::IPv4Port IPAddressOverride::ApplyOverride(Common::IPv4Port subject) const
|
||||
{
|
||||
Common::IPv4PortRange original;
|
||||
Common::IPv4PortRange replacement;
|
||||
// This logic could probably be better.
|
||||
// Ranges of different sizes will be weird in general.
|
||||
|
||||
// Caller should check if it matches first!
|
||||
Common::IPv4Port ApplyOverride(Common::IPv4Port subject) const
|
||||
const auto replacement_first_ip_u32 = replacement.first.GetIPAddressValue();
|
||||
const auto ip_count = 1u + u64(replacement.last.GetIPAddressValue()) - replacement_first_ip_u32;
|
||||
const auto result_ip =
|
||||
u32(replacement_first_ip_u32 +
|
||||
((subject.GetIPAddressValue() - original.first.GetIPAddressValue()) % ip_count));
|
||||
|
||||
subject.ip_address = std::bit_cast<Common::IPAddress>(Common::BigEndianValue{result_ip});
|
||||
|
||||
const auto replacement_first_port_u16 = replacement.first.GetPortValue();
|
||||
const auto port_count = 1u + u32(replacement.last.GetPortValue()) - replacement_first_port_u16;
|
||||
|
||||
// If the replacement includes all ports then we don't alter the port.
|
||||
// This allows "10.0.0.1:80-88=10.0.0.2" to have the expected behavior.
|
||||
if (port_count != 65536u)
|
||||
{
|
||||
// This logic could probably be better.
|
||||
// Ranges of different sizes will be weird in general.
|
||||
|
||||
const auto replacement_first_ip_u32 = replacement.first.GetIPAddressValue();
|
||||
const auto ip_count = 1u + u64(replacement.last.GetIPAddressValue()) - replacement_first_ip_u32;
|
||||
const auto result_ip =
|
||||
u32(replacement_first_ip_u32 +
|
||||
((subject.GetIPAddressValue() - original.first.GetIPAddressValue()) % ip_count));
|
||||
|
||||
subject.ip_address = std::bit_cast<Common::IPAddress>(Common::BigEndianValue{result_ip});
|
||||
|
||||
const auto replacement_first_port_u16 = replacement.first.GetPortValue();
|
||||
const auto port_count = 1u + u32(replacement.last.GetPortValue()) - replacement_first_port_u16;
|
||||
|
||||
// If the replacement includes all ports then we don't alter the port.
|
||||
// This allows "10.0.0.1:80-88=10.0.0.2" to have the expected behavior.
|
||||
if (port_count != 65536u)
|
||||
{
|
||||
const auto result_port_u16 =
|
||||
u16(replacement_first_port_u16 +
|
||||
((subject.GetPortValue() - original.first.GetPortValue()) % port_count));
|
||||
subject.port = std::bit_cast<u16>(Common::BigEndianValue{result_port_u16});
|
||||
}
|
||||
|
||||
return subject;
|
||||
const auto result_port_u16 =
|
||||
u16(replacement_first_port_u16 +
|
||||
((subject.GetPortValue() - original.first.GetPortValue()) % port_count));
|
||||
subject.port = std::bit_cast<u16>(Common::BigEndianValue{result_port_u16});
|
||||
}
|
||||
|
||||
Common::IPv4Port ReverseOverride(Common::IPv4Port subject) const
|
||||
{
|
||||
// Low effort implementation..
|
||||
return IPAddressOverride{.original = replacement, .replacement = original}.ApplyOverride(
|
||||
subject);
|
||||
}
|
||||
};
|
||||
return subject;
|
||||
}
|
||||
|
||||
using IPOverrides = std::vector<IPAddressOverride>;
|
||||
static IPOverrides GetIPOverrides()
|
||||
Common::IPv4Port IPAddressOverride::ReverseOverride(Common::IPv4Port subject) const
|
||||
{
|
||||
// Low effort implementation..
|
||||
return IPAddressOverride{.original = replacement, .replacement = original}.ApplyOverride(subject);
|
||||
}
|
||||
|
||||
std::string IPAddressOverride::ToString() const
|
||||
{
|
||||
return fmt::format("{}={}", original.ToString(), replacement.ToString());
|
||||
}
|
||||
|
||||
IPOverrides GetIPOverrides()
|
||||
{
|
||||
IPOverrides result;
|
||||
|
||||
|
|
@ -2180,4 +2177,12 @@ void DoState(PointerWrap& p)
|
|||
}
|
||||
}
|
||||
|
||||
s32 DebuggerGetSocket(u32 triforce_fd)
|
||||
{
|
||||
if (triforce_fd < std::size(s_sockets))
|
||||
return s32(s_sockets[triforce_fd]);
|
||||
|
||||
WARN_LOG_FMT(AMMEDIABOARD, "GC-AM: Bad socket fd used by the debugger: {}", triforce_fd);
|
||||
return -1;
|
||||
}
|
||||
} // namespace AMMediaboard
|
||||
|
|
|
|||
|
|
@ -10,6 +10,7 @@
|
|||
|
||||
#include "Common/ChunkFile.h"
|
||||
#include "Common/CommonTypes.h"
|
||||
#include "Common/Network.h"
|
||||
|
||||
#include "DiscIO/Volume.h"
|
||||
|
||||
|
|
@ -231,6 +232,8 @@ enum SocketStatusCodes
|
|||
SSC_SUCCESS = 70,
|
||||
};
|
||||
|
||||
static constexpr std::size_t SOCKET_FD_MAX = 64;
|
||||
|
||||
void Init();
|
||||
void FirmwareMap(bool on);
|
||||
void InitDIMM(const DiscIO::Volume& volume);
|
||||
|
|
@ -248,6 +251,24 @@ struct ParsedIPOverride
|
|||
std::string_view replacement;
|
||||
std::string_view description;
|
||||
};
|
||||
|
||||
std::optional<ParsedIPOverride> ParseIPOverride(std::string_view str);
|
||||
|
||||
struct IPAddressOverride
|
||||
{
|
||||
Common::IPv4PortRange original;
|
||||
Common::IPv4PortRange replacement;
|
||||
|
||||
// Caller should check if it matches first!
|
||||
Common::IPv4Port ApplyOverride(Common::IPv4Port subject) const;
|
||||
Common::IPv4Port ReverseOverride(Common::IPv4Port subject) const;
|
||||
std::string ToString() const;
|
||||
};
|
||||
|
||||
using IPOverrides = std::vector<IPAddressOverride>;
|
||||
|
||||
IPOverrides GetIPOverrides();
|
||||
|
||||
s32 DebuggerGetSocket(u32 triforce_fd);
|
||||
|
||||
}; // namespace AMMediaboard
|
||||
|
|
|
|||
|
|
@ -21,9 +21,13 @@
|
|||
#include <sys/socket.h>
|
||||
#endif
|
||||
|
||||
#include <bit>
|
||||
|
||||
#include "Common/FileUtil.h"
|
||||
#include "Common/Network.h"
|
||||
#include "Core/Config/MainSettings.h"
|
||||
#include "Core/Core.h"
|
||||
#include "Core/HW/DVD/AMMediaboard.h"
|
||||
#include "Core/IOS/Network/SSL.h"
|
||||
#include "Core/IOS/Network/Socket.h"
|
||||
#include "Core/System.h"
|
||||
|
|
@ -139,6 +143,49 @@ QTableWidgetItem* GetSocketName(s32 host_fd)
|
|||
|
||||
return new QTableWidgetItem(QStringLiteral("%1->%2").arg(sock_name).arg(peer_name));
|
||||
}
|
||||
|
||||
QTableWidgetItem* GetSocketRedirections(s32 host_fd, const AMMediaboard::IPOverrides& ip_overrides)
|
||||
{
|
||||
if (host_fd < 0 || ip_overrides.empty())
|
||||
return new QTableWidgetItem();
|
||||
|
||||
sockaddr_in sock_addr;
|
||||
socklen_t sock_addr_len = sizeof(sockaddr_in);
|
||||
if (getsockname(host_fd, reinterpret_cast<sockaddr*>(&sock_addr), &sock_addr_len) != 0)
|
||||
return new QTableWidgetItem();
|
||||
const Common::IPv4Port sock_ip_port{std::bit_cast<Common::IPAddress>(sock_addr.sin_addr),
|
||||
sock_addr.sin_port};
|
||||
|
||||
sockaddr_in peer_addr;
|
||||
socklen_t peer_addr_len = sizeof(sockaddr_in);
|
||||
bool has_peer =
|
||||
getpeername(host_fd, reinterpret_cast<sockaddr*>(&peer_addr), &peer_addr_len) == 0;
|
||||
const Common::IPv4Port peer_ip_port =
|
||||
has_peer ? Common::IPv4Port{std::bit_cast<Common::IPAddress>(peer_addr.sin_addr),
|
||||
peer_addr.sin_port} :
|
||||
Common::IPv4Port{};
|
||||
|
||||
QStringList sock_rules;
|
||||
QStringList peer_rules;
|
||||
for (const auto& rule : ip_overrides)
|
||||
{
|
||||
if (rule.replacement.IsMatch(sock_ip_port))
|
||||
sock_rules << QString::fromStdString(rule.ToString());
|
||||
if (!has_peer)
|
||||
continue;
|
||||
if (rule.replacement.IsMatch(peer_ip_port))
|
||||
peer_rules << QString::fromStdString(rule.ToString());
|
||||
}
|
||||
|
||||
if (sock_rules.isEmpty() && peer_rules.isEmpty())
|
||||
return new QTableWidgetItem();
|
||||
if (peer_rules.isEmpty())
|
||||
return new QTableWidgetItem(sock_rules.join(QStringLiteral("\n")));
|
||||
return new QTableWidgetItem(QStringLiteral("Sock rules:\n%1\n"
|
||||
"\nPeer rules:%2")
|
||||
.arg(sock_rules.join(QStringLiteral("\n")))
|
||||
.arg(peer_rules.join(QStringLiteral("\n"))));
|
||||
}
|
||||
} // namespace
|
||||
|
||||
NetworkWidget::NetworkWidget(QWidget* parent) : QDockWidget(parent)
|
||||
|
|
@ -254,24 +301,12 @@ void NetworkWidget::ConnectWidgets()
|
|||
});
|
||||
}
|
||||
|
||||
void NetworkWidget::Update()
|
||||
void NetworkWidget::UpdateWiiSocketTable(Core::System& system)
|
||||
{
|
||||
if (!isVisible())
|
||||
return;
|
||||
|
||||
auto& system = Core::System::GetInstance();
|
||||
if (Core::GetState(system) != Core::State::Paused)
|
||||
{
|
||||
m_socket_table->setDisabled(true);
|
||||
m_ssl_table->setDisabled(true);
|
||||
return;
|
||||
}
|
||||
|
||||
m_socket_table->setDisabled(false);
|
||||
m_ssl_table->setDisabled(false);
|
||||
|
||||
// needed because there's a race condition on the IOS instance otherwise
|
||||
const Core::CPUThreadGuard guard(system);
|
||||
// Show Wii socket blocking state
|
||||
m_socket_table->showColumn(4);
|
||||
// Hide Triforce IP overrides
|
||||
m_socket_table->hideColumn(6);
|
||||
|
||||
auto* ios = system.GetIOS();
|
||||
if (!ios)
|
||||
|
|
@ -281,7 +316,6 @@ void NetworkWidget::Update()
|
|||
if (!socket_manager)
|
||||
return;
|
||||
|
||||
m_socket_table->setRowCount(0);
|
||||
for (s32 wii_fd = 0; wii_fd < IOS::HLE::WII_SOCKET_FD_MAX; wii_fd++)
|
||||
{
|
||||
m_socket_table->insertRow(wii_fd);
|
||||
|
|
@ -293,9 +327,7 @@ void NetworkWidget::Update()
|
|||
m_socket_table->setItem(wii_fd, 4, GetSocketBlocking(*socket_manager, wii_fd));
|
||||
m_socket_table->setItem(wii_fd, 5, GetSocketName(host_fd));
|
||||
}
|
||||
m_socket_table->resizeColumnsToContents();
|
||||
|
||||
m_ssl_table->setRowCount(0);
|
||||
for (s32 ssl_id = 0; ssl_id < IOS::HLE::NET_SSL_MAXINSTANCES; ssl_id++)
|
||||
{
|
||||
m_ssl_table->insertRow(ssl_id);
|
||||
|
|
@ -312,6 +344,58 @@ void NetworkWidget::Update()
|
|||
m_ssl_table->setItem(ssl_id, 3, GetSocketState(host_fd));
|
||||
m_ssl_table->setItem(ssl_id, 4, GetSocketName(host_fd));
|
||||
}
|
||||
}
|
||||
|
||||
void NetworkWidget::UpdateTriforceSocketTable()
|
||||
{
|
||||
// No easy way to get socket blocking state on Windows
|
||||
m_socket_table->hideColumn(4);
|
||||
// Show active IP overrides
|
||||
m_socket_table->showColumn(6);
|
||||
const auto ip_overrides = AMMediaboard::GetIPOverrides();
|
||||
for (s32 triforce_fd = 0; triforce_fd < AMMediaboard::SOCKET_FD_MAX; triforce_fd++)
|
||||
{
|
||||
m_socket_table->insertRow(triforce_fd);
|
||||
const s32 host_fd = AMMediaboard::DebuggerGetSocket(triforce_fd);
|
||||
m_socket_table->setItem(triforce_fd, 0, new QTableWidgetItem(QString::number(triforce_fd)));
|
||||
m_socket_table->setItem(triforce_fd, 1, GetSocketDomain(host_fd));
|
||||
m_socket_table->setItem(triforce_fd, 2, GetSocketType(host_fd));
|
||||
m_socket_table->setItem(triforce_fd, 3, GetSocketState(host_fd));
|
||||
m_socket_table->setItem(triforce_fd, 4, new QTableWidgetItem(QTableWidget::tr("Unknown")));
|
||||
m_socket_table->setItem(triforce_fd, 5, GetSocketName(host_fd));
|
||||
m_socket_table->setItem(triforce_fd, 6, GetSocketRedirections(host_fd, ip_overrides));
|
||||
}
|
||||
}
|
||||
|
||||
void NetworkWidget::Update()
|
||||
{
|
||||
if (!isVisible())
|
||||
return;
|
||||
|
||||
auto& system = Core::System::GetInstance();
|
||||
if (Core::GetState(system) != Core::State::Paused)
|
||||
{
|
||||
m_socket_table->setDisabled(true);
|
||||
m_ssl_table->setDisabled(true);
|
||||
return;
|
||||
}
|
||||
|
||||
// needed because there's a race condition on the IOS instance otherwise
|
||||
const Core::CPUThreadGuard guard(system);
|
||||
m_socket_table->setDisabled(false);
|
||||
m_socket_table->setRowCount(0);
|
||||
m_ssl_table->setRowCount(0);
|
||||
if (system.IsTriforce())
|
||||
{
|
||||
UpdateTriforceSocketTable();
|
||||
}
|
||||
else if (system.IsWii())
|
||||
{
|
||||
m_ssl_table->setDisabled(false);
|
||||
UpdateWiiSocketTable(system);
|
||||
}
|
||||
m_socket_table->resizeColumnsToContents();
|
||||
m_socket_table->resizeRowsToContents();
|
||||
m_ssl_table->resizeColumnsToContents();
|
||||
|
||||
const bool is_pcap = Config::Get(Config::MAIN_NETWORK_DUMP_AS_PCAP);
|
||||
|
|
@ -348,7 +432,8 @@ QGroupBox* NetworkWidget::CreateSocketTableGroup()
|
|||
|
||||
m_socket_table = new QTableWidget();
|
||||
// i18n: FD stands for file descriptor (and in this case refers to sockets, not regular files)
|
||||
QStringList header{tr("FD"), tr("Domain"), tr("Type"), tr("State"), tr("Blocking"), tr("Name")};
|
||||
QStringList header{tr("FD"), tr("Domain"), tr("Type"), tr("State"),
|
||||
tr("Blocking"), tr("Name"), tr("Redirections")};
|
||||
m_socket_table->setColumnCount(static_cast<int>(header.size()));
|
||||
|
||||
m_socket_table->setHorizontalHeaderLabels(header);
|
||||
|
|
|
|||
|
|
@ -14,6 +14,11 @@ class QShowEvent;
|
|||
class QTableWidget;
|
||||
class QTableWidgetItem;
|
||||
|
||||
namespace Core
|
||||
{
|
||||
class System;
|
||||
}
|
||||
|
||||
class NetworkWidget : public QDockWidget
|
||||
{
|
||||
Q_OBJECT
|
||||
|
|
@ -29,6 +34,8 @@ private:
|
|||
void CreateWidgets();
|
||||
void ConnectWidgets();
|
||||
|
||||
void UpdateWiiSocketTable(Core::System& system);
|
||||
void UpdateTriforceSocketTable();
|
||||
void Update();
|
||||
|
||||
QGroupBox* CreateSocketTableGroup();
|
||||
|
|
|
|||
Loading…
Reference in New Issue
Block a user