Merge pull request #7675 from TryTwo/Debugger_Code_Features

Debugger: Get target memory in load/store instructions
This commit is contained in:
Admiral H. Curtiss 2022-04-08 05:28:22 +02:00 committed by GitHub
commit 23508cafb2
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 149 additions and 56 deletions

View File

@ -72,6 +72,11 @@ public:
virtual void WriteExtraMemory(int /*memory*/, u32 /*value*/, u32 /*address*/) {} virtual void WriteExtraMemory(int /*memory*/, u32 /*value*/, u32 /*address*/) {}
virtual u32 ReadExtraMemory(int /*memory*/, u32 /*address*/) const { return 0; } virtual u32 ReadExtraMemory(int /*memory*/, u32 /*address*/) const { return 0; }
virtual u32 ReadInstruction(u32 /*address*/) const { return 0; } virtual u32 ReadInstruction(u32 /*address*/) const { return 0; }
virtual std::optional<u32>
GetMemoryAddressFromInstruction(const std::string& /*instruction*/) const
{
return std::nullopt;
}
virtual u32 GetPC() const { return 0; } virtual u32 GetPC() const { return 0; }
virtual void SetPC(u32 /*address*/) {} virtual void SetPC(u32 /*address*/) {}
virtual void Step() {} virtual void Step() {}

View File

@ -177,24 +177,23 @@ static u32 HelperRotateMask(int r, int mb, int me)
static std::string ldst_offs(u32 val) static std::string ldst_offs(u32 val)
{ {
if (val == 0) if (val == 0)
{
return "0"; return "0";
}
if (val & 0x8000) if (val & 0x8000)
{
return fmt::format("-0x{:04X}", ((~val) & 0xffff) + 1); return fmt::format("-0x{:04X}", ((~val) & 0xffff) + 1);
}
return fmt::format("0x{:04X}", val); return fmt::format("0x{:04X}", val);
} }
static int SEX12(u32 x) static std::string psq_offs(u32 val)
{ {
if ((x & 0x800) != 0) if (val == 0)
return static_cast<int>(x | 0xFFFFF000); return "0";
return static_cast<int>(x); if ((val & 0x800) != 0)
return fmt::format("-0x{:04X}", ((~val) & 0xfff) + 1);
return fmt::format("0x{:04X}", val);
} }
static std::string spr_name(int i) static std::string spr_name(int i)
@ -452,24 +451,9 @@ std::string GekkoDisassembler::rd_ra_rb(u32 in, int mask)
return result; return result;
} }
std::string GekkoDisassembler::fd_ra_rb(u32 in, int mask) std::string GekkoDisassembler::fd_ra_rb(u32 in)
{ {
std::string result; return fmt::format("f{}, {}, {}", PPCGETD(in), regnames[PPCGETA(in)], regnames[PPCGETB(in)]);
if (mask)
{
if (mask & 4)
result += fmt::format("f{},", PPCGETD(in));
if (mask & 2)
result += fmt::format("{},", regnames[PPCGETA(in)]);
if (mask & 1)
result += fmt::format("{},", regnames[PPCGETB(in)]);
// Drop the trailing comma
result.pop_back();
}
return result;
} }
void GekkoDisassembler::trapi(u32 in, unsigned char dmode) void GekkoDisassembler::trapi(u32 in, unsigned char dmode)
@ -972,26 +956,23 @@ void GekkoDisassembler::fdabc(u32 in, std::string_view name, int mask, unsigned
m_flags |= dmode; m_flags |= dmode;
m_opcode = fmt::format("f{}{}", name, rcsel[in & 1]); m_opcode = fmt::format("f{}{}", name, rcsel[in & 1]);
m_operands += fmt::format("f{},", PPCGETD(in)); m_operands += fmt::format("f{}", PPCGETD(in));
if (mask & 4) if (mask & 4)
m_operands += fmt::format("f{},", PPCGETA(in)); m_operands += fmt::format(", f{}", PPCGETA(in));
else if ((mask & 8) == 0) else if ((mask & 8) == 0)
err |= (int)PPCGETA(in); err |= (int)PPCGETA(in);
if (mask & 2) if (mask & 2)
m_operands += fmt::format("f{},", PPCGETC(in)); m_operands += fmt::format(", f{}", PPCGETC(in));
else if (PPCGETC(in) && (mask & 8) == 0) else if (PPCGETC(in) && (mask & 8) == 0)
err |= (int)PPCGETC(in); err |= (int)PPCGETC(in);
if (mask & 1) if (mask & 1)
m_operands += fmt::format("f{},", PPCGETB(in)); m_operands += fmt::format(", f{}", PPCGETB(in));
else if (!(mask & 8)) else if (!(mask & 8))
err |= (int)PPCGETB(in); err |= (int)PPCGETB(in);
// Drop the trailing comma
m_operands.pop_back();
if (err) if (err)
ill(in); ill(in);
} }
@ -1003,10 +984,10 @@ void GekkoDisassembler::fmr(u32 in)
} }
// Indexed float instruction: xxxx fD,rA,rB // Indexed float instruction: xxxx fD,rA,rB
void GekkoDisassembler::fdab(u32 in, std::string_view name, int mask) void GekkoDisassembler::fdab(u32 in, std::string_view name)
{ {
m_opcode = name; m_opcode = name;
m_operands = fd_ra_rb(in, mask); m_operands = fd_ra_rb(in);
} }
void GekkoDisassembler::fcmp(u32 in, char c) void GekkoDisassembler::fcmp(u32 in, char c)
@ -1018,7 +999,7 @@ void GekkoDisassembler::fcmp(u32 in, char c)
else else
{ {
m_opcode = fmt::format("fcmp{}", c); m_opcode = fmt::format("fcmp{}", c);
m_operands = fmt::format("cr{},f{},f{}", PPCGETCRD(in), PPCGETA(in), PPCGETB(in)); m_operands = fmt::format("cr{}, f{}, f{}", PPCGETCRD(in), PPCGETA(in), PPCGETB(in));
} }
} }
@ -1092,7 +1073,7 @@ void GekkoDisassembler::ps(u32 inst)
{ {
case 6: case 6:
m_opcode = inst & 0x40 ? "psq_lux" : "psq_lx"; m_opcode = inst & 0x40 ? "psq_lux" : "psq_lx";
m_operands = fmt::format("p{}, (r{} + r{}), {}, qr{}", FD, RA, RB, WX, IX); m_operands = fmt::format("p{}, r{}, r{}, {}, qr{}", FD, RA, RB, WX, IX);
return; return;
case 7: case 7:
@ -1224,22 +1205,22 @@ void GekkoDisassembler::ps(u32 inst)
} }
case 528: case 528:
m_opcode = "ps_merge00"; m_opcode = "ps_merge00";
m_operands = fmt::format("p{}, p{}[0],p{}[0]", FD, FA, FB); m_operands = fmt::format("p{}, p{}[0], p{}[0]", FD, FA, FB);
return; return;
case 560: case 560:
m_opcode = "ps_merge01"; m_opcode = "ps_merge01";
m_operands = fmt::format("p{}, p{}[0],p{}[1]", FD, FA, FB); m_operands = fmt::format("p{}, p{}[0], p{}[1]", FD, FA, FB);
return; return;
case 592: case 592:
m_opcode = "ps_merge10"; m_opcode = "ps_merge10";
m_operands = fmt::format("p{}, p{}[1],p{}[0]", FD, FA, FB); m_operands = fmt::format("p{}, p{}[1], p{}[0]", FD, FA, FB);
return; return;
case 624: case 624:
m_opcode = "ps_merge11"; m_opcode = "ps_merge11";
m_operands = fmt::format("p{}, p{}[1],p{}[1]", FD, FA, FB); m_operands = fmt::format("p{}, p{}[1], p{}[1]", FD, FA, FB);
return; return;
case 1014: case 1014:
@ -1261,23 +1242,23 @@ void GekkoDisassembler::ps_mem(u32 inst)
{ {
case 56: case 56:
m_opcode = "psq_l"; m_opcode = "psq_l";
m_operands = fmt::format("p{}, {}(r{}), {}, qr{}", RS, SEX12(inst & 0xFFF), RA, W, I); m_operands = fmt::format("p{}, {}(r{}), {}, qr{}", RS, psq_offs(inst & 0xFFF), RA, W, I);
break; break;
case 57: case 57:
m_opcode = "psq_lu"; m_opcode = "psq_lu";
m_operands = fmt::format("p{}, {}(r{}), {}, qr{}", RS, SEX12(inst & 0xFFF), RA, W, I); m_operands = fmt::format("p{}, {}(r{}), {}, qr{}", RS, psq_offs(inst & 0xFFF), RA, W, I);
; ;
break; break;
case 60: case 60:
m_opcode = "psq_st"; m_opcode = "psq_st";
m_operands = fmt::format("p{}, {}(r{}), {}, qr{}", RS, SEX12(inst & 0xFFF), RA, W, I); m_operands = fmt::format("p{}, {}(r{}), {}, qr{}", RS, psq_offs(inst & 0xFFF), RA, W, I);
break; break;
case 61: case 61:
m_opcode = "psq_stu"; m_opcode = "psq_stu";
m_operands = fmt::format("p{}, {}(r{}), {}, qr{}", RS, SEX12(inst & 0xFFF), RA, W, I); m_operands = fmt::format("p{}, {}(r{}), {}, qr{}", RS, psq_offs(inst & 0xFFF), RA, W, I);
break; break;
} }
} }
@ -1924,7 +1905,7 @@ u32* GekkoDisassembler::DoDisassembly(bool big_endian)
break; break;
case 535: case 535:
fdab(in, "lfsx", 7); fdab(in, "lfsx");
break; break;
case 536: case 536:
@ -1940,7 +1921,7 @@ u32* GekkoDisassembler::DoDisassembly(bool big_endian)
break; break;
case 567: case 567:
fdab(in, "lfsux", 7); fdab(in, "lfsux");
break; break;
case 595: case 595:
@ -1956,11 +1937,11 @@ u32* GekkoDisassembler::DoDisassembly(bool big_endian)
break; break;
case 599: case 599:
fdab(in, "lfdx", 7); fdab(in, "lfdx");
break; break;
case 631: case 631:
fdab(in, "lfdux", 7); fdab(in, "lfdux");
break; break;
case 659: case 659:
@ -1979,11 +1960,11 @@ u32* GekkoDisassembler::DoDisassembly(bool big_endian)
break; break;
case 663: case 663:
fdab(in, "stfsx", 7); fdab(in, "stfsx");
break; break;
case 695: case 695:
fdab(in, "stfsux", 7); fdab(in, "stfsux");
break; break;
case 725: case 725:
@ -1991,11 +1972,11 @@ u32* GekkoDisassembler::DoDisassembly(bool big_endian)
break; break;
case 727: case 727:
fdab(in, "stfdx", 7); fdab(in, "stfdx");
break; break;
case 759: case 759:
fdab(in, "stfdux", 7); fdab(in, "stfdux");
break; break;
case 790: case 790:
@ -2044,7 +2025,7 @@ u32* GekkoDisassembler::DoDisassembly(bool big_endian)
break; break;
case 983: case 983:
fdab(in, "stfiwx", 7); fdab(in, "stfiwx");
break; break;
case 986: case 986:

View File

@ -56,7 +56,7 @@ private:
static std::string ra_rb(u32 in); static std::string ra_rb(u32 in);
static std::string rd_ra_rb(u32 in, int mask); static std::string rd_ra_rb(u32 in, int mask);
static std::string fd_ra_rb(u32 in, int mask); static std::string fd_ra_rb(u32 in);
static void trapi(u32 in, unsigned char dmode); static void trapi(u32 in, unsigned char dmode);
static void cmpi(u32 in, int uimm); static void cmpi(u32 in, int uimm);
@ -84,7 +84,7 @@ private:
static void ldst(u32 in, std::string_view name, char reg, unsigned char dmode); static void ldst(u32 in, std::string_view name, char reg, unsigned char dmode);
static void fdabc(u32 in, std::string_view name, int mask, unsigned char dmode); static void fdabc(u32 in, std::string_view name, int mask, unsigned char dmode);
static void fmr(u32 in); static void fmr(u32 in);
static void fdab(u32 in, std::string_view name, int mask); static void fdab(u32 in, std::string_view name);
static void fcmp(u32 in, char c); static void fcmp(u32 in, char c);
static void mtfsb(u32 in, int n); static void mtfsb(u32 in, int n);
static void ps(u32 inst); static void ps(u32 inst);

View File

@ -6,6 +6,7 @@
#include <algorithm> #include <algorithm>
#include <array> #include <array>
#include <cstddef> #include <cstddef>
#include <regex>
#include <string> #include <string>
#include <vector> #include <vector>
@ -375,6 +376,55 @@ std::string PPCDebugInterface::GetDescription(u32 address) const
return g_symbolDB.GetDescription(address); return g_symbolDB.GetDescription(address);
} }
std::optional<u32>
PPCDebugInterface::GetMemoryAddressFromInstruction(const std::string& instruction) const
{
std::regex re(",[^r0-]*(-?)(0[xX]?[0-9a-fA-F]*|r\\d+)[^r^s]*.(p|toc|\\d+)");
std::smatch match;
// Instructions should be identified as a load or store before using this function. This error
// check should never trigger.
if (!std::regex_search(instruction, match, re))
return std::nullopt;
// Output: match.str(1): negative sign for offset or no match. match.str(2): 0xNNNN, 0, or
// rNN. Check next for 'r' to see if a gpr needs to be loaded. match.str(3): will either be p,
// toc, or NN. Always a gpr.
const std::string offset_match = match.str(2);
const std::string register_match = match.str(3);
constexpr char is_reg = 'r';
u32 offset = 0;
if (is_reg == offset_match[0])
{
const int register_index = std::stoi(offset_match.substr(1), nullptr, 10);
offset = (register_index == 0 ? 0 : GPR(register_index));
}
else
{
offset = static_cast<u32>(std::stoi(offset_match, nullptr, 16));
}
// sp and rtoc need to be converted to 1 and 2.
constexpr char is_sp = 'p';
constexpr char is_rtoc = 't';
u32 i = 0;
if (is_sp == register_match[0])
i = 1;
else if (is_rtoc == register_match[0])
i = 2;
else
i = std::stoi(register_match, nullptr, 10);
const u32 base_address = GPR(i);
if (!match.str(1).empty())
return base_address - offset;
return base_address + offset;
}
u32 PPCDebugInterface::GetPC() const u32 PPCDebugInterface::GetPC() const
{ {
return PowerPC::ppcState.pc; return PowerPC::ppcState.pc;

View File

@ -76,6 +76,7 @@ public:
u32 ReadExtraMemory(int memory, u32 address) const override; u32 ReadExtraMemory(int memory, u32 address) const override;
u32 ReadInstruction(u32 address) const override; u32 ReadInstruction(u32 address) const override;
std::optional<u32> GetMemoryAddressFromInstruction(const std::string& instruction) const override;
u32 GetPC() const override; u32 GetPC() const override;
void SetPC(u32 address) override; void SetPC(u32 address) override;
void Step() override {} void Step() override {}

View File

@ -235,6 +235,15 @@ static bool IsBranchInstructionWithLink(std::string_view ins)
StringEndsWith(ins, "la+") || StringEndsWith(ins, "l-") || StringEndsWith(ins, "la-"); StringEndsWith(ins, "la+") || StringEndsWith(ins, "l-") || StringEndsWith(ins, "la-");
} }
static bool IsInstructionLoadStore(std::string_view ins)
{
// Could add check for context address being near PC, because we need gprs to be correct for the
// load/store.
return (StringBeginsWith(ins, "l") && !StringBeginsWith(ins, "li")) ||
StringBeginsWith(ins, "st") || StringBeginsWith(ins, "psq_l") ||
StringBeginsWith(ins, "psq_s");
}
void CodeViewWidget::Update() void CodeViewWidget::Update()
{ {
if (!isVisible()) if (!isVisible())
@ -530,7 +539,10 @@ void CodeViewWidget::OnContextMenu()
auto* copy_hex_action = menu->addAction(tr("Copy &hex"), this, &CodeViewWidget::OnCopyHex); auto* copy_hex_action = menu->addAction(tr("Copy &hex"), this, &CodeViewWidget::OnCopyHex);
menu->addAction(tr("Show in &memory"), this, &CodeViewWidget::OnShowInMemory); menu->addAction(tr("Show in &memory"), this, &CodeViewWidget::OnShowInMemory);
auto* show_target_memory =
menu->addAction(tr("Show target in memor&y"), this, &CodeViewWidget::OnShowTargetInMemory);
auto* copy_target_memory =
menu->addAction(tr("Copy tar&get address"), this, &CodeViewWidget::OnCopyTargetAddress);
menu->addSeparator(); menu->addSeparator();
auto* symbol_rename_action = auto* symbol_rename_action =
@ -561,6 +573,14 @@ void CodeViewWidget::OnContextMenu()
for (auto* action : {symbol_rename_action, symbol_size_action, symbol_end_action}) for (auto* action : {symbol_rename_action, symbol_size_action, symbol_end_action})
action->setEnabled(has_symbol); action->setEnabled(has_symbol);
const bool valid_load_store = Core::GetState() == Core::State::Paused &&
IsInstructionLoadStore(PowerPC::debug_interface.Disassemble(addr));
for (auto* action : {copy_target_memory, show_target_memory})
{
action->setEnabled(valid_load_store);
}
restore_action->setEnabled(running && PowerPC::debug_interface.HasEnabledPatch(addr)); restore_action->setEnabled(running && PowerPC::debug_interface.HasEnabledPatch(addr));
menu->exec(QCursor::pos()); menu->exec(QCursor::pos());
@ -574,11 +594,45 @@ void CodeViewWidget::OnCopyAddress()
QApplication::clipboard()->setText(QStringLiteral("%1").arg(addr, 8, 16, QLatin1Char('0'))); QApplication::clipboard()->setText(QStringLiteral("%1").arg(addr, 8, 16, QLatin1Char('0')));
} }
void CodeViewWidget::OnCopyTargetAddress()
{
if (Core::GetState() != Core::State::Paused)
return;
const std::string code_line = PowerPC::debug_interface.Disassemble(GetContextAddress());
if (!IsInstructionLoadStore(code_line))
return;
const std::optional<u32> addr =
PowerPC::debug_interface.GetMemoryAddressFromInstruction(code_line);
if (addr)
QApplication::clipboard()->setText(QStringLiteral("%1").arg(*addr, 8, 16, QLatin1Char('0')));
}
void CodeViewWidget::OnShowInMemory() void CodeViewWidget::OnShowInMemory()
{ {
emit ShowMemory(GetContextAddress()); emit ShowMemory(GetContextAddress());
} }
void CodeViewWidget::OnShowTargetInMemory()
{
if (Core::GetState() != Core::State::Paused)
return;
const std::string code_line = PowerPC::debug_interface.Disassemble(GetContextAddress());
if (!IsInstructionLoadStore(code_line))
return;
const std::optional<u32> addr =
PowerPC::debug_interface.GetMemoryAddressFromInstruction(code_line);
if (addr)
emit ShowMemory(*addr);
}
void CodeViewWidget::OnCopyCode() void CodeViewWidget::OnCopyCode()
{ {
const u32 addr = GetContextAddress(); const u32 addr = GetContextAddress();

View File

@ -70,7 +70,9 @@ private:
void OnFollowBranch(); void OnFollowBranch();
void OnCopyAddress(); void OnCopyAddress();
void OnCopyTargetAddress();
void OnShowInMemory(); void OnShowInMemory();
void OnShowTargetInMemory();
void OnCopyFunction(); void OnCopyFunction();
void OnCopyCode(); void OnCopyCode();
void OnCopyHex(); void OnCopyHex();