Merge pull request #1291 from skidau/debugger-step-out
Dolphin debugger enhancements
This commit is contained in:
commit
1630b0c684
|
@ -113,6 +113,19 @@ void BreakPoints::Clear()
|
||||||
m_BreakPoints.clear();
|
m_BreakPoints.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void BreakPoints::ClearAllTemporary()
|
||||||
|
{
|
||||||
|
for (const TBreakPoint& bp : m_BreakPoints)
|
||||||
|
{
|
||||||
|
if (bp.bTemporary)
|
||||||
|
{
|
||||||
|
if (jit)
|
||||||
|
jit->GetBlockCache()->InvalidateICache(bp.iAddress, 4, true);
|
||||||
|
Remove(bp.iAddress);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
MemChecks::TMemChecksStr MemChecks::GetStrings() const
|
MemChecks::TMemChecksStr MemChecks::GetStrings() const
|
||||||
{
|
{
|
||||||
TMemChecksStr mcs;
|
TMemChecksStr mcs;
|
||||||
|
@ -204,3 +217,87 @@ void TMemCheck::Action(DebugInterface *debug_interface, u32 iValue, u32 addr, bo
|
||||||
debug_interface->BreakNow();
|
debug_interface->BreakNow();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const bool Watches::IsAddressWatch(u32 _iAddress)
|
||||||
|
{
|
||||||
|
for (const TWatch& bp : m_Watches)
|
||||||
|
if (bp.iAddress == _iAddress)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
Watches::TWatchesStr Watches::GetStrings() const
|
||||||
|
{
|
||||||
|
TWatchesStr bps;
|
||||||
|
for (const TWatch& bp : m_Watches)
|
||||||
|
{
|
||||||
|
std::stringstream ss;
|
||||||
|
ss << std::hex << bp.iAddress << " " << bp.name;
|
||||||
|
bps.push_back(ss.str());
|
||||||
|
}
|
||||||
|
|
||||||
|
return bps;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Watches::AddFromStrings(const TWatchesStr& bpstrs)
|
||||||
|
{
|
||||||
|
for (const std::string& bpstr : bpstrs)
|
||||||
|
{
|
||||||
|
TWatch bp;
|
||||||
|
std::stringstream ss;
|
||||||
|
ss << std::hex << bpstr;
|
||||||
|
ss >> bp.iAddress;
|
||||||
|
ss >> std::ws;
|
||||||
|
getline(ss, bp.name);
|
||||||
|
Add(bp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Watches::Add(const TWatch& bp)
|
||||||
|
{
|
||||||
|
if (!IsAddressWatch(bp.iAddress))
|
||||||
|
{
|
||||||
|
m_Watches.push_back(bp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Watches::Add(u32 em_address)
|
||||||
|
{
|
||||||
|
if (!IsAddressWatch(em_address)) // only add new addresses
|
||||||
|
{
|
||||||
|
TWatch pt; // breakpoint settings
|
||||||
|
pt.bOn = true;
|
||||||
|
pt.iAddress = em_address;
|
||||||
|
|
||||||
|
m_Watches.push_back(pt);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Watches::Update(int count, u32 em_address)
|
||||||
|
{
|
||||||
|
m_Watches.at(count).iAddress = em_address;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Watches::UpdateName(int count, const std::string name)
|
||||||
|
{
|
||||||
|
m_Watches.at(count).name = name;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Watches::Remove(u32 em_address)
|
||||||
|
{
|
||||||
|
for (auto i = m_Watches.begin(); i != m_Watches.end(); ++i)
|
||||||
|
{
|
||||||
|
if (i->iAddress == em_address)
|
||||||
|
{
|
||||||
|
m_Watches.erase(i);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Watches::Clear()
|
||||||
|
{
|
||||||
|
m_Watches.clear();
|
||||||
|
}
|
||||||
|
|
|
@ -44,6 +44,13 @@ struct TMemCheck
|
||||||
bool write, int size, u32 pc);
|
bool write, int size, u32 pc);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct TWatch
|
||||||
|
{
|
||||||
|
std::string name = "";
|
||||||
|
u32 iAddress;
|
||||||
|
bool bOn;
|
||||||
|
};
|
||||||
|
|
||||||
// Code breakpoints.
|
// Code breakpoints.
|
||||||
class BreakPoints
|
class BreakPoints
|
||||||
{
|
{
|
||||||
|
@ -67,6 +74,7 @@ public:
|
||||||
// Remove Breakpoint
|
// Remove Breakpoint
|
||||||
void Remove(u32 _iAddress);
|
void Remove(u32 _iAddress);
|
||||||
void Clear();
|
void Clear();
|
||||||
|
void ClearAllTemporary();
|
||||||
|
|
||||||
void DeleteByAddress(u32 _Address);
|
void DeleteByAddress(u32 _Address);
|
||||||
|
|
||||||
|
@ -98,3 +106,33 @@ public:
|
||||||
|
|
||||||
void Clear() { m_MemChecks.clear(); }
|
void Clear() { m_MemChecks.clear(); }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class Watches
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
typedef std::vector<TWatch> TWatches;
|
||||||
|
typedef std::vector<std::string> TWatchesStr;
|
||||||
|
|
||||||
|
const TWatches& GetWatches() { return m_Watches; }
|
||||||
|
|
||||||
|
TWatchesStr GetStrings() const;
|
||||||
|
void AddFromStrings(const TWatchesStr& bps);
|
||||||
|
|
||||||
|
const bool IsAddressWatch(u32 _iAddress);
|
||||||
|
|
||||||
|
// Add BreakPoint
|
||||||
|
void Add(u32 em_address);
|
||||||
|
void Add(const TWatch& bp);
|
||||||
|
|
||||||
|
void Update(int count, u32 em_address);
|
||||||
|
void UpdateName(int count, const std::string name);
|
||||||
|
|
||||||
|
// Remove Breakpoint
|
||||||
|
void Remove(u32 _iAddress);
|
||||||
|
void Clear();
|
||||||
|
|
||||||
|
void DeleteByAddress(u32 _Address);
|
||||||
|
|
||||||
|
private:
|
||||||
|
TWatches m_Watches;
|
||||||
|
};
|
|
@ -18,6 +18,7 @@ public:
|
||||||
virtual void ClearBreakpoint(unsigned int /*address*/){}
|
virtual void ClearBreakpoint(unsigned int /*address*/){}
|
||||||
virtual void ClearAllBreakpoints() {}
|
virtual void ClearAllBreakpoints() {}
|
||||||
virtual void ToggleBreakpoint(unsigned int /*address*/){}
|
virtual void ToggleBreakpoint(unsigned int /*address*/){}
|
||||||
|
virtual void AddWatch(unsigned int /*address*/){}
|
||||||
virtual void ClearAllMemChecks() {}
|
virtual void ClearAllMemChecks() {}
|
||||||
virtual bool IsMemCheck(unsigned int /*address*/) {return false;}
|
virtual bool IsMemCheck(unsigned int /*address*/) {return false;}
|
||||||
virtual void ToggleMemCheck(unsigned int /*address*/){}
|
virtual void ToggleMemCheck(unsigned int /*address*/){}
|
||||||
|
|
|
@ -131,6 +131,11 @@ void PPCDebugInterface::ToggleBreakpoint(unsigned int address)
|
||||||
PowerPC::breakpoints.Add(address);
|
PowerPC::breakpoints.Add(address);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void PPCDebugInterface::AddWatch(unsigned int address)
|
||||||
|
{
|
||||||
|
PowerPC::watches.Add(address);
|
||||||
|
}
|
||||||
|
|
||||||
void PPCDebugInterface::ClearAllMemChecks()
|
void PPCDebugInterface::ClearAllMemChecks()
|
||||||
{
|
{
|
||||||
PowerPC::memchecks.Clear();
|
PowerPC::memchecks.Clear();
|
||||||
|
|
|
@ -22,6 +22,7 @@ public:
|
||||||
virtual void SetBreakpoint(unsigned int address) override;
|
virtual void SetBreakpoint(unsigned int address) override;
|
||||||
virtual void ClearBreakpoint(unsigned int address) override;
|
virtual void ClearBreakpoint(unsigned int address) override;
|
||||||
virtual void ClearAllBreakpoints() override;
|
virtual void ClearAllBreakpoints() override;
|
||||||
|
virtual void AddWatch(unsigned int address) override;
|
||||||
virtual void ToggleBreakpoint(unsigned int address) override;
|
virtual void ToggleBreakpoint(unsigned int address) override;
|
||||||
virtual void ClearAllMemChecks() override;
|
virtual void ClearAllMemChecks() override;
|
||||||
virtual bool IsMemCheck(unsigned int address) override;
|
virtual bool IsMemCheck(unsigned int address) override;
|
||||||
|
|
|
@ -117,6 +117,15 @@ void CCPU::EnableStepping(const bool _bStepping)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
// SingleStep so that the "continue", "step over" and "step out" debugger functions
|
||||||
|
// work when the PC is at a breakpoint at the beginning of the block
|
||||||
|
if (PowerPC::breakpoints.IsAddressBreakPoint(PC) && PowerPC::GetMode() != PowerPC::MODE_INTERPRETER)
|
||||||
|
{
|
||||||
|
PowerPC::CoreMode oldMode = PowerPC::GetMode();
|
||||||
|
PowerPC::SetMode(PowerPC::MODE_INTERPRETER);
|
||||||
|
PowerPC::SingleStep();
|
||||||
|
PowerPC::SetMode(oldMode);
|
||||||
|
}
|
||||||
PowerPC::Start();
|
PowerPC::Start();
|
||||||
m_StepEvent.Set();
|
m_StepEvent.Set();
|
||||||
g_video_backend->EmuStateChange(EMUSTATE_CHANGE_PLAY);
|
g_video_backend->EmuStateChange(EMUSTATE_CHANGE_PLAY);
|
||||||
|
|
|
@ -174,13 +174,8 @@ bool Jit64::HandleFault(uintptr_t access_address, SContext* ctx)
|
||||||
void Jit64::Init()
|
void Jit64::Init()
|
||||||
{
|
{
|
||||||
jo.optimizeStack = true;
|
jo.optimizeStack = true;
|
||||||
jo.enableBlocklink = true;
|
EnableBlockLink();
|
||||||
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bJITNoBlockLinking ||
|
|
||||||
SConfig::GetInstance().m_LocalCoreStartupParameter.bMMU)
|
|
||||||
{
|
|
||||||
// TODO: support block linking with MMU
|
|
||||||
jo.enableBlocklink = false;
|
|
||||||
}
|
|
||||||
jo.fpAccurateFcmp = SConfig::GetInstance().m_LocalCoreStartupParameter.bFPRF;
|
jo.fpAccurateFcmp = SConfig::GetInstance().m_LocalCoreStartupParameter.bFPRF;
|
||||||
jo.optimizeGatherPipe = true;
|
jo.optimizeGatherPipe = true;
|
||||||
jo.fastInterrupts = false;
|
jo.fastInterrupts = false;
|
||||||
|
@ -195,7 +190,7 @@ void Jit64::Init()
|
||||||
|
|
||||||
// BLR optimization has the same consequences as block linking, as well as
|
// BLR optimization has the same consequences as block linking, as well as
|
||||||
// depending on the fault handler to be safe in the event of excessive BL.
|
// depending on the fault handler to be safe in the event of excessive BL.
|
||||||
m_enable_blr_optimization = jo.enableBlocklink && SConfig::GetInstance().m_LocalCoreStartupParameter.bFastmem;
|
m_enable_blr_optimization = jo.enableBlocklink && SConfig::GetInstance().m_LocalCoreStartupParameter.bFastmem && !SConfig::GetInstance().m_LocalCoreStartupParameter.bEnableDebugging;
|
||||||
m_clear_cache_asap = false;
|
m_clear_cache_asap = false;
|
||||||
|
|
||||||
m_stack = nullptr;
|
m_stack = nullptr;
|
||||||
|
@ -212,9 +207,7 @@ void Jit64::Init()
|
||||||
code_block.m_stats = &js.st;
|
code_block.m_stats = &js.st;
|
||||||
code_block.m_gpa = &js.gpa;
|
code_block.m_gpa = &js.gpa;
|
||||||
code_block.m_fpa = &js.fpa;
|
code_block.m_fpa = &js.fpa;
|
||||||
analyzer.SetOption(PPCAnalyst::PPCAnalyzer::OPTION_CONDITIONAL_CONTINUE);
|
EnableOptimization();
|
||||||
analyzer.SetOption(PPCAnalyst::PPCAnalyzer::OPTION_BRANCH_MERGE);
|
|
||||||
analyzer.SetOption(PPCAnalyst::PPCAnalyzer::OPTION_CARRY_MERGE);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Jit64::ClearCache()
|
void Jit64::ClearCache()
|
||||||
|
@ -521,11 +514,23 @@ const u8* Jit64::DoJit(u32 em_address, PPCAnalyst::CodeBuffer *code_buf, JitBloc
|
||||||
|
|
||||||
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bEnableDebugging)
|
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bEnableDebugging)
|
||||||
{
|
{
|
||||||
|
// We can link blocks as long as we are not single stepping and there are no breakpoints here
|
||||||
|
EnableBlockLink();
|
||||||
|
EnableOptimization();
|
||||||
|
|
||||||
// Comment out the following to disable breakpoints (speed-up)
|
// Comment out the following to disable breakpoints (speed-up)
|
||||||
if (!Profiler::g_ProfileBlocks)
|
if (!Profiler::g_ProfileBlocks)
|
||||||
{
|
{
|
||||||
if (GetState() == CPU_STEPPING)
|
if (GetState() == CPU_STEPPING)
|
||||||
|
{
|
||||||
blockSize = 1;
|
blockSize = 1;
|
||||||
|
|
||||||
|
// Do not link this block to other blocks While single stepping
|
||||||
|
jo.enableBlocklink = false;
|
||||||
|
analyzer.ClearOption(PPCAnalyst::PPCAnalyzer::OPTION_CONDITIONAL_CONTINUE);
|
||||||
|
analyzer.ClearOption(PPCAnalyst::PPCAnalyzer::OPTION_BRANCH_MERGE);
|
||||||
|
analyzer.ClearOption(PPCAnalyst::PPCAnalyzer::OPTION_CARRY_MERGE);
|
||||||
|
}
|
||||||
Trace();
|
Trace();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -715,6 +720,9 @@ const u8* Jit64::DoJit(u32 em_address, PPCAnalyst::CodeBuffer *code_buf, JitBloc
|
||||||
|
|
||||||
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bEnableDebugging && breakpoints.IsAddressBreakPoint(ops[i].address) && GetState() != CPU_STEPPING)
|
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bEnableDebugging && breakpoints.IsAddressBreakPoint(ops[i].address) && GetState() != CPU_STEPPING)
|
||||||
{
|
{
|
||||||
|
// Turn off block linking if there are breakpoints so that the Step Over command does not link this block.
|
||||||
|
jo.enableBlocklink = false;
|
||||||
|
|
||||||
gpr.Flush();
|
gpr.Flush();
|
||||||
fpr.Flush();
|
fpr.Flush();
|
||||||
|
|
||||||
|
@ -856,3 +864,21 @@ u32 Jit64::CallerSavedRegistersInUse()
|
||||||
}
|
}
|
||||||
return result & ABI_ALL_CALLER_SAVED;
|
return result & ABI_ALL_CALLER_SAVED;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Jit64::EnableBlockLink()
|
||||||
|
{
|
||||||
|
jo.enableBlocklink = true;
|
||||||
|
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bJITNoBlockLinking ||
|
||||||
|
SConfig::GetInstance().m_LocalCoreStartupParameter.bMMU)
|
||||||
|
{
|
||||||
|
// TODO: support block linking with MMU
|
||||||
|
jo.enableBlocklink = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Jit64::EnableOptimization()
|
||||||
|
{
|
||||||
|
analyzer.SetOption(PPCAnalyst::PPCAnalyzer::OPTION_CONDITIONAL_CONTINUE);
|
||||||
|
analyzer.SetOption(PPCAnalyst::PPCAnalyzer::OPTION_BRANCH_MERGE);
|
||||||
|
analyzer.SetOption(PPCAnalyst::PPCAnalyzer::OPTION_CARRY_MERGE);
|
||||||
|
}
|
||||||
|
|
|
@ -64,6 +64,11 @@ public:
|
||||||
~Jit64() {}
|
~Jit64() {}
|
||||||
|
|
||||||
void Init() override;
|
void Init() override;
|
||||||
|
|
||||||
|
void EnableOptimization();
|
||||||
|
|
||||||
|
void EnableBlockLink();
|
||||||
|
|
||||||
void Shutdown() override;
|
void Shutdown() override;
|
||||||
|
|
||||||
bool HandleFault(uintptr_t access_address, SContext* ctx) override;
|
bool HandleFault(uintptr_t access_address, SContext* ctx) override;
|
||||||
|
|
|
@ -98,9 +98,12 @@ void Jit64::lXXx(UGeckoInstruction inst)
|
||||||
if (accessSize == 8 && js.next_inst.OPCD == 31 && js.next_inst.SUBOP10 == 954 &&
|
if (accessSize == 8 && js.next_inst.OPCD == 31 && js.next_inst.SUBOP10 == 954 &&
|
||||||
js.next_inst.RS == inst.RD && js.next_inst.RA == inst.RD && !js.next_inst.Rc)
|
js.next_inst.RS == inst.RD && js.next_inst.RA == inst.RD && !js.next_inst.Rc)
|
||||||
{
|
{
|
||||||
js.downcountAmount++;
|
if (PowerPC::GetState() != PowerPC::CPU_STEPPING)
|
||||||
js.skipnext = true;
|
{
|
||||||
signExtend = true;
|
js.downcountAmount++;
|
||||||
|
js.skipnext = true;
|
||||||
|
signExtend = true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO(ector): Make it dynamically enable/disable idle skipping where appropriate
|
// TODO(ector): Make it dynamically enable/disable idle skipping where appropriate
|
||||||
|
@ -109,6 +112,7 @@ void Jit64::lXXx(UGeckoInstruction inst)
|
||||||
// IMHO those Idles should always be skipped and replaced by a more controllable "native" Idle methode
|
// IMHO those Idles should always be skipped and replaced by a more controllable "native" Idle methode
|
||||||
// ... maybe the throttle one already do that :p
|
// ... maybe the throttle one already do that :p
|
||||||
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bSkipIdle &&
|
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bSkipIdle &&
|
||||||
|
PowerPC::GetState() != PowerPC::CPU_STEPPING &&
|
||||||
inst.OPCD == 32 &&
|
inst.OPCD == 32 &&
|
||||||
(inst.hex & 0xFFFF0000) == 0x800D0000 &&
|
(inst.hex & 0xFFFF0000) == 0x800D0000 &&
|
||||||
(Memory::ReadUnchecked_U32(js.compilerPC + 4) == 0x28000000 ||
|
(Memory::ReadUnchecked_U32(js.compilerPC + 4) == 0x28000000 ||
|
||||||
|
|
|
@ -228,21 +228,24 @@ void Jit64::mfspr(UGeckoInstruction inst)
|
||||||
// Be careful; the actual opcode is for mftb (371), not mfspr (339)
|
// Be careful; the actual opcode is for mftb (371), not mfspr (339)
|
||||||
if (js.next_inst.OPCD == 31 && js.next_inst.SUBOP10 == 371 && (nextIndex == SPR_TU || nextIndex == SPR_TL))
|
if (js.next_inst.OPCD == 31 && js.next_inst.SUBOP10 == 371 && (nextIndex == SPR_TU || nextIndex == SPR_TL))
|
||||||
{
|
{
|
||||||
int n = js.next_inst.RD;
|
if (PowerPC::GetState() != PowerPC::CPU_STEPPING)
|
||||||
js.downcountAmount++;
|
{
|
||||||
js.skipnext = true;
|
int n = js.next_inst.RD;
|
||||||
gpr.Lock(d, n);
|
js.downcountAmount++;
|
||||||
gpr.BindToRegister(d, false);
|
js.skipnext = true;
|
||||||
gpr.BindToRegister(n, false);
|
gpr.Lock(d, n);
|
||||||
if (iIndex == SPR_TL)
|
gpr.BindToRegister(d, false);
|
||||||
MOV(32, gpr.R(d), R(RAX));
|
gpr.BindToRegister(n, false);
|
||||||
if (nextIndex == SPR_TL)
|
if (iIndex == SPR_TL)
|
||||||
MOV(32, gpr.R(n), R(RAX));
|
MOV(32, gpr.R(d), R(RAX));
|
||||||
SHR(64, R(RAX), Imm8(32));
|
if (nextIndex == SPR_TL)
|
||||||
if (iIndex == SPR_TU)
|
MOV(32, gpr.R(n), R(RAX));
|
||||||
MOV(32, gpr.R(d), R(RAX));
|
SHR(64, R(RAX), Imm8(32));
|
||||||
if (nextIndex == SPR_TU)
|
if (iIndex == SPR_TU)
|
||||||
MOV(32, gpr.R(n), R(RAX));
|
MOV(32, gpr.R(d), R(RAX));
|
||||||
|
if (nextIndex == SPR_TU)
|
||||||
|
MOV(32, gpr.R(n), R(RAX));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
|
|
@ -244,13 +244,7 @@ namespace JitILProfiler
|
||||||
void JitIL::Init()
|
void JitIL::Init()
|
||||||
{
|
{
|
||||||
jo.optimizeStack = true;
|
jo.optimizeStack = true;
|
||||||
jo.enableBlocklink = true;
|
EnableBlockLink();
|
||||||
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bJITNoBlockLinking ||
|
|
||||||
SConfig::GetInstance().m_LocalCoreStartupParameter.bMMU)
|
|
||||||
{
|
|
||||||
// TODO: support block linking with MMU
|
|
||||||
jo.enableBlocklink = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
jo.fpAccurateFcmp = false;
|
jo.fpAccurateFcmp = false;
|
||||||
jo.optimizeGatherPipe = true;
|
jo.optimizeGatherPipe = true;
|
||||||
|
@ -509,11 +503,19 @@ const u8* JitIL::DoJit(u32 em_address, PPCAnalyst::CodeBuffer *code_buf, JitBloc
|
||||||
|
|
||||||
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bEnableDebugging)
|
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bEnableDebugging)
|
||||||
{
|
{
|
||||||
|
// We can link blocks as long as we are not single stepping and there are no breakpoints here
|
||||||
|
EnableBlockLink();
|
||||||
|
|
||||||
// Comment out the following to disable breakpoints (speed-up)
|
// Comment out the following to disable breakpoints (speed-up)
|
||||||
if (!Profiler::g_ProfileBlocks)
|
if (!Profiler::g_ProfileBlocks)
|
||||||
{
|
{
|
||||||
if (GetState() == CPU_STEPPING)
|
if (GetState() == CPU_STEPPING)
|
||||||
|
{
|
||||||
blockSize = 1;
|
blockSize = 1;
|
||||||
|
|
||||||
|
// Do not link this block to other blocks While single stepping
|
||||||
|
jo.enableBlocklink = false;
|
||||||
|
}
|
||||||
Trace();
|
Trace();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -648,6 +650,9 @@ const u8* JitIL::DoJit(u32 em_address, PPCAnalyst::CodeBuffer *code_buf, JitBloc
|
||||||
|
|
||||||
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bEnableDebugging && breakpoints.IsAddressBreakPoint(ops[i].address) && GetState() != CPU_STEPPING)
|
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bEnableDebugging && breakpoints.IsAddressBreakPoint(ops[i].address) && GetState() != CPU_STEPPING)
|
||||||
{
|
{
|
||||||
|
// Turn off block linking if there are breakpoints so that the Step Over command does not link this block.
|
||||||
|
jo.enableBlocklink = false;
|
||||||
|
|
||||||
ibuild.EmitBreakPointCheck(ibuild.EmitIntConst(ops[i].address));
|
ibuild.EmitBreakPointCheck(ibuild.EmitIntConst(ops[i].address));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -702,3 +707,14 @@ const u8* JitIL::DoJit(u32 em_address, PPCAnalyst::CodeBuffer *code_buf, JitBloc
|
||||||
|
|
||||||
return normalEntry;
|
return normalEntry;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void JitIL::EnableBlockLink()
|
||||||
|
{
|
||||||
|
jo.enableBlocklink = true;
|
||||||
|
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bJITNoBlockLinking ||
|
||||||
|
SConfig::GetInstance().m_LocalCoreStartupParameter.bMMU)
|
||||||
|
{
|
||||||
|
// TODO: support block linking with MMU
|
||||||
|
jo.enableBlocklink = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -47,6 +47,9 @@ public:
|
||||||
// Initialization, etc
|
// Initialization, etc
|
||||||
|
|
||||||
void Init() override;
|
void Init() override;
|
||||||
|
|
||||||
|
void EnableBlockLink();
|
||||||
|
|
||||||
void Shutdown() override;
|
void Shutdown() override;
|
||||||
|
|
||||||
// Jit!
|
// Jit!
|
||||||
|
|
|
@ -316,13 +316,6 @@ void EmuCodeBlock::SafeLoadToReg(X64Reg reg_value, const Gen::OpArg & opAddress,
|
||||||
// The following masks the region used by the GC/Wii virtual memory lib
|
// The following masks the region used by the GC/Wii virtual memory lib
|
||||||
mem_mask |= Memory::ADDR_MASK_MEM1;
|
mem_mask |= Memory::ADDR_MASK_MEM1;
|
||||||
|
|
||||||
#ifdef ENABLE_MEM_CHECK
|
|
||||||
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bEnableDebugging)
|
|
||||||
{
|
|
||||||
mem_mask |= Memory::EXRAM_MASK;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
if (opAddress.IsImm())
|
if (opAddress.IsImm())
|
||||||
{
|
{
|
||||||
u32 address = (u32)opAddress.offset + offset;
|
u32 address = (u32)opAddress.offset + offset;
|
||||||
|
@ -519,13 +512,6 @@ void EmuCodeBlock::SafeWriteRegToReg(OpArg reg_value, X64Reg reg_addr, int acces
|
||||||
// The following masks the region used by the GC/Wii virtual memory lib
|
// The following masks the region used by the GC/Wii virtual memory lib
|
||||||
mem_mask |= Memory::ADDR_MASK_MEM1;
|
mem_mask |= Memory::ADDR_MASK_MEM1;
|
||||||
|
|
||||||
#ifdef ENABLE_MEM_CHECK
|
|
||||||
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bEnableDebugging)
|
|
||||||
{
|
|
||||||
mem_mask |= Memory::EXRAM_MASK;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
bool swap = !(flags & SAFE_LOADSTORE_NO_SWAP);
|
bool swap = !(flags & SAFE_LOADSTORE_NO_SWAP);
|
||||||
|
|
||||||
FixupBranch slow, exit;
|
FixupBranch slow, exit;
|
||||||
|
|
|
@ -37,6 +37,7 @@ void JitILBase::lXz(UGeckoInstruction inst)
|
||||||
// Idle Skipping. This really should be done somewhere else.
|
// Idle Skipping. This really should be done somewhere else.
|
||||||
// Either lower in the IR or higher in PPCAnalyist
|
// Either lower in the IR or higher in PPCAnalyist
|
||||||
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bSkipIdle &&
|
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bSkipIdle &&
|
||||||
|
PowerPC::GetState() != PowerPC::CPU_STEPPING &&
|
||||||
inst.OPCD == 32 && // Lwx
|
inst.OPCD == 32 && // Lwx
|
||||||
(inst.hex & 0xFFFF0000) == 0x800D0000 &&
|
(inst.hex & 0xFFFF0000) == 0x800D0000 &&
|
||||||
(Memory::ReadUnchecked_U32(js.compilerPC + 4) == 0x28000000 ||
|
(Memory::ReadUnchecked_U32(js.compilerPC + 4) == 0x28000000 ||
|
||||||
|
|
|
@ -35,6 +35,7 @@ static volatile CPUState state = CPU_POWERDOWN;
|
||||||
Interpreter * const interpreter = Interpreter::getInstance();
|
Interpreter * const interpreter = Interpreter::getInstance();
|
||||||
static CoreMode mode;
|
static CoreMode mode;
|
||||||
|
|
||||||
|
Watches watches;
|
||||||
BreakPoints breakpoints;
|
BreakPoints breakpoints;
|
||||||
MemChecks memchecks;
|
MemChecks memchecks;
|
||||||
PPCDebugInterface debug_interface;
|
PPCDebugInterface debug_interface;
|
||||||
|
@ -161,6 +162,9 @@ void Init(int cpu_core)
|
||||||
state = CPU_STEPPING;
|
state = CPU_STEPPING;
|
||||||
|
|
||||||
ppcState.iCache.Init();
|
ppcState.iCache.Init();
|
||||||
|
|
||||||
|
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bEnableDebugging)
|
||||||
|
breakpoints.ClearAllTemporary();
|
||||||
}
|
}
|
||||||
|
|
||||||
void Shutdown()
|
void Shutdown()
|
||||||
|
|
|
@ -114,6 +114,7 @@ enum CPUState
|
||||||
|
|
||||||
extern PowerPCState ppcState;
|
extern PowerPCState ppcState;
|
||||||
|
|
||||||
|
extern Watches watches;
|
||||||
extern BreakPoints breakpoints;
|
extern BreakPoints breakpoints;
|
||||||
extern MemChecks memchecks;
|
extern MemChecks memchecks;
|
||||||
extern PPCDebugInterface debug_interface;
|
extern PPCDebugInterface debug_interface;
|
||||||
|
|
|
@ -45,6 +45,8 @@ set(GUI_SRCS
|
||||||
Debugger/MemoryWindow.cpp
|
Debugger/MemoryWindow.cpp
|
||||||
Debugger/RegisterView.cpp
|
Debugger/RegisterView.cpp
|
||||||
Debugger/RegisterWindow.cpp
|
Debugger/RegisterWindow.cpp
|
||||||
|
Debugger/WatchView.cpp
|
||||||
|
Debugger/WatchWindow.cpp
|
||||||
FifoPlayerDlg.cpp
|
FifoPlayerDlg.cpp
|
||||||
Frame.cpp
|
Frame.cpp
|
||||||
FrameAui.cpp
|
FrameAui.cpp
|
||||||
|
|
|
@ -19,6 +19,7 @@
|
||||||
#include "Common/CommonTypes.h"
|
#include "Common/CommonTypes.h"
|
||||||
#include "Common/FileUtil.h"
|
#include "Common/FileUtil.h"
|
||||||
#include "Common/IniFile.h"
|
#include "Common/IniFile.h"
|
||||||
|
#include "Core/ConfigManager.h"
|
||||||
#include "Core/HW/Memmap.h"
|
#include "Core/HW/Memmap.h"
|
||||||
#include "Core/PowerPC/PowerPC.h"
|
#include "Core/PowerPC/PowerPC.h"
|
||||||
#include "DolphinWX/WxUtils.h"
|
#include "DolphinWX/WxUtils.h"
|
||||||
|
@ -45,9 +46,9 @@ public:
|
||||||
{
|
{
|
||||||
SetToolBitmapSize(wxSize(24, 24));
|
SetToolBitmapSize(wxSize(24, 24));
|
||||||
|
|
||||||
m_Bitmaps[Toolbar_Delete] = wxBitmap(wxGetBitmapFromMemory(toolbar_delete_png).ConvertToImage().Rescale(24, 24));
|
m_Bitmaps[Toolbar_Delete] = wxBitmap(wxGetBitmapFromMemory(toolbar_delete_png).ConvertToImage().Rescale(16, 16));
|
||||||
m_Bitmaps[Toolbar_Add_BP] = wxBitmap(wxGetBitmapFromMemory(toolbar_add_breakpoint_png).ConvertToImage().Rescale(24, 24));
|
m_Bitmaps[Toolbar_Add_BP] = wxBitmap(wxGetBitmapFromMemory(toolbar_add_breakpoint_png).ConvertToImage().Rescale(16, 16));
|
||||||
m_Bitmaps[Toolbar_Add_MC] = wxBitmap(wxGetBitmapFromMemory(toolbar_add_memcheck_png).ConvertToImage().Rescale(24, 24));
|
m_Bitmaps[Toolbar_Add_MC] = wxBitmap(wxGetBitmapFromMemory(toolbar_add_memcheck_png).ConvertToImage().Rescale(16, 16));
|
||||||
|
|
||||||
AddTool(ID_DELETE, _("Delete"), m_Bitmaps[Toolbar_Delete]);
|
AddTool(ID_DELETE, _("Delete"), m_Bitmaps[Toolbar_Delete]);
|
||||||
Bind(wxEVT_TOOL, &CBreakPointWindow::OnDelete, parent, ID_DELETE);
|
Bind(wxEVT_TOOL, &CBreakPointWindow::OnDelete, parent, ID_DELETE);
|
||||||
|
@ -66,7 +67,7 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
AddTool(ID_LOAD, _("Load"), m_Bitmaps[Toolbar_Delete]);
|
AddTool(ID_LOAD, _("Load"), m_Bitmaps[Toolbar_Delete]);
|
||||||
Bind(wxEVT_TOOL, &CBreakPointWindow::LoadAll, parent, ID_LOAD);
|
Bind(wxEVT_TOOL, &CBreakPointWindow::Event_LoadAll, parent, ID_LOAD);
|
||||||
|
|
||||||
AddTool(ID_SAVE, _("Save"), m_Bitmaps[Toolbar_Delete]);
|
AddTool(ID_SAVE, _("Save"), m_Bitmaps[Toolbar_Delete]);
|
||||||
Bind(wxEVT_TOOL, &CBreakPointWindow::Event_SaveAll, parent, ID_SAVE);
|
Bind(wxEVT_TOOL, &CBreakPointWindow::Event_SaveAll, parent, ID_SAVE);
|
||||||
|
@ -112,7 +113,7 @@ CBreakPointWindow::CBreakPointWindow(CCodeWindow* _pCodeWindow, wxWindow* parent
|
||||||
m_BreakPointListView = new CBreakPointView(this, wxID_ANY);
|
m_BreakPointListView = new CBreakPointView(this, wxID_ANY);
|
||||||
|
|
||||||
m_mgr.AddPane(new CBreakPointBar(this, wxID_ANY), wxAuiPaneInfo().ToolbarPane().Top().
|
m_mgr.AddPane(new CBreakPointBar(this, wxID_ANY), wxAuiPaneInfo().ToolbarPane().Top().
|
||||||
LeftDockable(false).RightDockable(false).BottomDockable(false).Floatable(false));
|
LeftDockable(true).RightDockable(true).BottomDockable(false).Floatable(false));
|
||||||
m_mgr.AddPane(m_BreakPointListView, wxAuiPaneInfo().CenterPane());
|
m_mgr.AddPane(m_BreakPointListView, wxAuiPaneInfo().CenterPane());
|
||||||
m_mgr.Update();
|
m_mgr.Update();
|
||||||
}
|
}
|
||||||
|
@ -180,32 +181,38 @@ void CBreakPointWindow::SaveAll()
|
||||||
{
|
{
|
||||||
// simply dump all to bp/mc files in a way we can read again
|
// simply dump all to bp/mc files in a way we can read again
|
||||||
IniFile ini;
|
IniFile ini;
|
||||||
if (ini.Load(File::GetUserPath(F_DEBUGGERCONFIG_IDX)))
|
ini.Load(File::GetUserPath(D_GAMESETTINGS_IDX) + SConfig::GetInstance().m_LocalCoreStartupParameter.GetUniqueID() + ".ini", false);
|
||||||
{
|
ini.SetLines("BreakPoints", PowerPC::breakpoints.GetStrings());
|
||||||
ini.SetLines("BreakPoints", PowerPC::breakpoints.GetStrings());
|
ini.SetLines("MemoryChecks", PowerPC::memchecks.GetStrings());
|
||||||
ini.SetLines("MemoryChecks", PowerPC::memchecks.GetStrings());
|
ini.Save(File::GetUserPath(D_GAMESETTINGS_IDX) + SConfig::GetInstance().m_LocalCoreStartupParameter.GetUniqueID() + ".ini");
|
||||||
ini.Save(File::GetUserPath(F_DEBUGGERCONFIG_IDX));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void CBreakPointWindow::LoadAll(wxCommandEvent& WXUNUSED(event))
|
void CBreakPointWindow::Event_LoadAll(wxCommandEvent& WXUNUSED(event))
|
||||||
|
{
|
||||||
|
LoadAll();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
void CBreakPointWindow::LoadAll()
|
||||||
{
|
{
|
||||||
IniFile ini;
|
IniFile ini;
|
||||||
BreakPoints::TBreakPointsStr newbps;
|
BreakPoints::TBreakPointsStr newbps;
|
||||||
MemChecks::TMemChecksStr newmcs;
|
MemChecks::TMemChecksStr newmcs;
|
||||||
|
|
||||||
if (!ini.Load(File::GetUserPath(F_DEBUGGERCONFIG_IDX)))
|
if (!ini.Load(File::GetUserPath(D_GAMESETTINGS_IDX) + SConfig::GetInstance().m_LocalCoreStartupParameter.GetUniqueID() + ".ini", false))
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ini.GetLines("BreakPoints", &newbps, false))
|
if (ini.GetLines("BreakPoints", &newbps, false))
|
||||||
{
|
{
|
||||||
|
PowerPC::breakpoints.Clear();
|
||||||
PowerPC::breakpoints.AddFromStrings(newbps);
|
PowerPC::breakpoints.AddFromStrings(newbps);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ini.GetLines("MemoryChecks", &newmcs, false))
|
if (ini.GetLines("MemoryChecks", &newmcs, false))
|
||||||
{
|
{
|
||||||
|
PowerPC::memchecks.Clear();
|
||||||
PowerPC::memchecks.AddFromStrings(newmcs);
|
PowerPC::memchecks.AddFromStrings(newmcs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -39,7 +39,8 @@ public:
|
||||||
void OnAddMemoryCheck(wxCommandEvent& WXUNUSED(event));
|
void OnAddMemoryCheck(wxCommandEvent& WXUNUSED(event));
|
||||||
void Event_SaveAll(wxCommandEvent& WXUNUSED(event));
|
void Event_SaveAll(wxCommandEvent& WXUNUSED(event));
|
||||||
void SaveAll();
|
void SaveAll();
|
||||||
void LoadAll(wxCommandEvent& WXUNUSED(event));
|
void Event_LoadAll(wxCommandEvent& WXUNUSED(event));
|
||||||
|
void LoadAll();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
DECLARE_EVENT_TABLE();
|
DECLARE_EVENT_TABLE();
|
||||||
|
|
|
@ -37,6 +37,7 @@
|
||||||
#include "Core/Debugger/PPCDebugInterface.h"
|
#include "Core/Debugger/PPCDebugInterface.h"
|
||||||
#include "Core/HW/CPU.h"
|
#include "Core/HW/CPU.h"
|
||||||
#include "Core/HW/Memmap.h"
|
#include "Core/HW/Memmap.h"
|
||||||
|
#include "Core/HW/SystemTimers.h"
|
||||||
#include "Core/PowerPC/Gekko.h"
|
#include "Core/PowerPC/Gekko.h"
|
||||||
#include "Core/PowerPC/JitInterface.h"
|
#include "Core/PowerPC/JitInterface.h"
|
||||||
#include "Core/PowerPC/PowerPC.h"
|
#include "Core/PowerPC/PowerPC.h"
|
||||||
|
@ -51,6 +52,7 @@
|
||||||
#include "DolphinWX/Debugger/DebuggerUIUtil.h"
|
#include "DolphinWX/Debugger/DebuggerUIUtil.h"
|
||||||
#include "DolphinWX/Debugger/JitWindow.h"
|
#include "DolphinWX/Debugger/JitWindow.h"
|
||||||
#include "DolphinWX/Debugger/RegisterWindow.h"
|
#include "DolphinWX/Debugger/RegisterWindow.h"
|
||||||
|
#include "DolphinWX/Debugger/WatchWindow.h"
|
||||||
|
|
||||||
extern "C" // Bitmaps
|
extern "C" // Bitmaps
|
||||||
{
|
{
|
||||||
|
@ -92,6 +94,7 @@ CCodeWindow::CCodeWindow(const SCoreStartupParameter& _LocalCoreStartupParameter
|
||||||
: wxPanel(parent, id, position, size, style, name)
|
: wxPanel(parent, id, position, size, style, name)
|
||||||
, Parent(parent)
|
, Parent(parent)
|
||||||
, m_RegisterWindow(nullptr)
|
, m_RegisterWindow(nullptr)
|
||||||
|
, m_WatchWindow(nullptr)
|
||||||
, m_BreakpointWindow(nullptr)
|
, m_BreakpointWindow(nullptr)
|
||||||
, m_MemoryWindow(nullptr)
|
, m_MemoryWindow(nullptr)
|
||||||
, m_JitWindow(nullptr)
|
, m_JitWindow(nullptr)
|
||||||
|
@ -151,6 +154,7 @@ void CCodeWindow::OnHostMessage(wxCommandEvent& event)
|
||||||
Update();
|
Update();
|
||||||
if (codeview) codeview->Center(PC);
|
if (codeview) codeview->Center(PC);
|
||||||
if (m_RegisterWindow) m_RegisterWindow->NotifyUpdate();
|
if (m_RegisterWindow) m_RegisterWindow->NotifyUpdate();
|
||||||
|
if (m_WatchWindow) m_WatchWindow->NotifyUpdate();
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case IDM_UPDATEBREAKPOINTS:
|
case IDM_UPDATEBREAKPOINTS:
|
||||||
|
@ -180,6 +184,10 @@ void CCodeWindow::OnCodeStep(wxCommandEvent& event)
|
||||||
StepOver();
|
StepOver();
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case IDM_STEPOUT:
|
||||||
|
StepOut();
|
||||||
|
break;
|
||||||
|
|
||||||
case IDM_TOGGLE_BREAKPOINT:
|
case IDM_TOGGLE_BREAKPOINT:
|
||||||
ToggleBreakpoint();
|
ToggleBreakpoint();
|
||||||
break;
|
break;
|
||||||
|
@ -288,6 +296,7 @@ void CCodeWindow::SingleStep()
|
||||||
{
|
{
|
||||||
if (CCPU::IsStepping())
|
if (CCPU::IsStepping())
|
||||||
{
|
{
|
||||||
|
PowerPC::breakpoints.ClearAllTemporary();
|
||||||
JitInterface::InvalidateICache(PC, 4, true);
|
JitInterface::InvalidateICache(PC, 4, true);
|
||||||
CCPU::StepOpcode(&sync_event);
|
CCPU::StepOpcode(&sync_event);
|
||||||
wxThread::Sleep(20);
|
wxThread::Sleep(20);
|
||||||
|
@ -304,6 +313,7 @@ void CCodeWindow::StepOver()
|
||||||
UGeckoInstruction inst = Memory::Read_Instruction(PC);
|
UGeckoInstruction inst = Memory::Read_Instruction(PC);
|
||||||
if (inst.LK)
|
if (inst.LK)
|
||||||
{
|
{
|
||||||
|
PowerPC::breakpoints.ClearAllTemporary();
|
||||||
PowerPC::breakpoints.Add(PC + 4, true);
|
PowerPC::breakpoints.Add(PC + 4, true);
|
||||||
CCPU::EnableStepping(false);
|
CCPU::EnableStepping(false);
|
||||||
JumpToAddress(PC);
|
JumpToAddress(PC);
|
||||||
|
@ -320,6 +330,52 @@ void CCodeWindow::StepOver()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void CCodeWindow::StepOut()
|
||||||
|
{
|
||||||
|
if (CCPU::IsStepping())
|
||||||
|
{
|
||||||
|
PowerPC::breakpoints.ClearAllTemporary();
|
||||||
|
|
||||||
|
// Keep stepping until the next blr or timeout after one second
|
||||||
|
u64 timeout = SystemTimers::GetTicksPerSecond();
|
||||||
|
u64 steps = 0;
|
||||||
|
PowerPC::CoreMode oldMode = PowerPC::GetMode();
|
||||||
|
PowerPC::SetMode(PowerPC::MODE_INTERPRETER);
|
||||||
|
UGeckoInstruction inst = Memory::Read_Instruction(PC);
|
||||||
|
GekkoOPInfo* opinfo = GetOpInfo(inst);
|
||||||
|
while (inst.hex != 0x4e800020 && steps < timeout) // check for blr
|
||||||
|
{
|
||||||
|
if (inst.LK)
|
||||||
|
{
|
||||||
|
// Step over branches
|
||||||
|
u32 next_pc = PC + 4;
|
||||||
|
while (PC != next_pc && steps < timeout)
|
||||||
|
{
|
||||||
|
PowerPC::SingleStep();
|
||||||
|
++steps;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
PowerPC::SingleStep();
|
||||||
|
++steps;
|
||||||
|
}
|
||||||
|
inst = Memory::Read_Instruction(PC);
|
||||||
|
opinfo = GetOpInfo(inst);
|
||||||
|
}
|
||||||
|
|
||||||
|
PowerPC::SingleStep();
|
||||||
|
PowerPC::SetMode(oldMode);
|
||||||
|
|
||||||
|
JumpToAddress(PC);
|
||||||
|
Update();
|
||||||
|
|
||||||
|
UpdateButtonStates();
|
||||||
|
// Update all toolbars in the aui manager
|
||||||
|
Parent->UpdateGUI();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void CCodeWindow::ToggleBreakpoint()
|
void CCodeWindow::ToggleBreakpoint()
|
||||||
{
|
{
|
||||||
if (CCPU::IsStepping())
|
if (CCPU::IsStepping())
|
||||||
|
@ -443,6 +499,7 @@ void CCodeWindow::CreateMenu(const SCoreStartupParameter& core_startup_parameter
|
||||||
|
|
||||||
pDebugMenu->Append(IDM_STEP, _("Step &Into\tF11"));
|
pDebugMenu->Append(IDM_STEP, _("Step &Into\tF11"));
|
||||||
pDebugMenu->Append(IDM_STEPOVER, _("Step &Over\tF10"));
|
pDebugMenu->Append(IDM_STEPOVER, _("Step &Over\tF10"));
|
||||||
|
pDebugMenu->Append(IDM_STEPOUT, _("Step O&ut\tSHIFT+F11"));
|
||||||
pDebugMenu->Append(IDM_TOGGLE_BREAKPOINT, _("Toggle &Breakpoint\tF9"));
|
pDebugMenu->Append(IDM_TOGGLE_BREAKPOINT, _("Toggle &Breakpoint\tF9"));
|
||||||
pDebugMenu->AppendSeparator();
|
pDebugMenu->AppendSeparator();
|
||||||
|
|
||||||
|
@ -607,6 +664,7 @@ void CCodeWindow::InitBitmaps()
|
||||||
// load original size 48x48
|
// load original size 48x48
|
||||||
m_Bitmaps[Toolbar_Step] = wxGetBitmapFromMemory(toolbar_add_breakpoint_png);
|
m_Bitmaps[Toolbar_Step] = wxGetBitmapFromMemory(toolbar_add_breakpoint_png);
|
||||||
m_Bitmaps[Toolbar_StepOver] = wxGetBitmapFromMemory(toolbar_add_memcheck_png);
|
m_Bitmaps[Toolbar_StepOver] = wxGetBitmapFromMemory(toolbar_add_memcheck_png);
|
||||||
|
m_Bitmaps[Toolbar_StepOut] = wxGetBitmapFromMemory(toolbar_add_memcheck_png);
|
||||||
m_Bitmaps[Toolbar_Skip] = wxGetBitmapFromMemory(toolbar_add_memcheck_png);
|
m_Bitmaps[Toolbar_Skip] = wxGetBitmapFromMemory(toolbar_add_memcheck_png);
|
||||||
m_Bitmaps[Toolbar_GotoPC] = wxGetBitmapFromMemory(toolbar_add_memcheck_png);
|
m_Bitmaps[Toolbar_GotoPC] = wxGetBitmapFromMemory(toolbar_add_memcheck_png);
|
||||||
m_Bitmaps[Toolbar_SetPC] = wxGetBitmapFromMemory(toolbar_add_memcheck_png);
|
m_Bitmaps[Toolbar_SetPC] = wxGetBitmapFromMemory(toolbar_add_memcheck_png);
|
||||||
|
@ -624,6 +682,7 @@ void CCodeWindow::PopulateToolbar(wxToolBar* toolBar)
|
||||||
toolBar->SetToolBitmapSize(wxSize(w, h));
|
toolBar->SetToolBitmapSize(wxSize(w, h));
|
||||||
WxUtils::AddToolbarButton(toolBar, IDM_STEP, _("Step"), m_Bitmaps[Toolbar_Step], _("Step into the next instruction"));
|
WxUtils::AddToolbarButton(toolBar, IDM_STEP, _("Step"), m_Bitmaps[Toolbar_Step], _("Step into the next instruction"));
|
||||||
WxUtils::AddToolbarButton(toolBar, IDM_STEPOVER, _("Step Over"), m_Bitmaps[Toolbar_StepOver], _("Step over the next instruction"));
|
WxUtils::AddToolbarButton(toolBar, IDM_STEPOVER, _("Step Over"), m_Bitmaps[Toolbar_StepOver], _("Step over the next instruction"));
|
||||||
|
WxUtils::AddToolbarButton(toolBar, IDM_STEPOUT, _("Step Out"), m_Bitmaps[Toolbar_StepOut], _("Step out of the current function"));
|
||||||
WxUtils::AddToolbarButton(toolBar, IDM_SKIP, _("Skip"), m_Bitmaps[Toolbar_Skip], _("Skips the next instruction completely"));
|
WxUtils::AddToolbarButton(toolBar, IDM_SKIP, _("Skip"), m_Bitmaps[Toolbar_Skip], _("Skips the next instruction completely"));
|
||||||
toolBar->AddSeparator();
|
toolBar->AddSeparator();
|
||||||
WxUtils::AddToolbarButton(toolBar, IDM_GOTOPC, _("Show PC"), m_Bitmaps[Toolbar_GotoPC], _("Go to the current instruction"));
|
WxUtils::AddToolbarButton(toolBar, IDM_GOTOPC, _("Show PC"), m_Bitmaps[Toolbar_GotoPC], _("Go to the current instruction"));
|
||||||
|
@ -660,6 +719,7 @@ void CCodeWindow::UpdateButtonStates()
|
||||||
if (!Initialized)
|
if (!Initialized)
|
||||||
{
|
{
|
||||||
ToolBar->EnableTool(IDM_STEPOVER, false);
|
ToolBar->EnableTool(IDM_STEPOVER, false);
|
||||||
|
ToolBar->EnableTool(IDM_STEPOUT, false);
|
||||||
ToolBar->EnableTool(IDM_SKIP, false);
|
ToolBar->EnableTool(IDM_SKIP, false);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
@ -667,11 +727,13 @@ void CCodeWindow::UpdateButtonStates()
|
||||||
if (!Stepping)
|
if (!Stepping)
|
||||||
{
|
{
|
||||||
ToolBar->EnableTool(IDM_STEPOVER, false);
|
ToolBar->EnableTool(IDM_STEPOVER, false);
|
||||||
|
ToolBar->EnableTool(IDM_STEPOUT, false);
|
||||||
ToolBar->EnableTool(IDM_SKIP, false);
|
ToolBar->EnableTool(IDM_SKIP, false);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
ToolBar->EnableTool(IDM_STEPOVER, true);
|
ToolBar->EnableTool(IDM_STEPOVER, true);
|
||||||
|
ToolBar->EnableTool(IDM_STEPOUT, true);
|
||||||
ToolBar->EnableTool(IDM_SKIP, true);
|
ToolBar->EnableTool(IDM_SKIP, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,6 +19,7 @@
|
||||||
|
|
||||||
class CFrame;
|
class CFrame;
|
||||||
class CRegisterWindow;
|
class CRegisterWindow;
|
||||||
|
class CWatchWindow;
|
||||||
class CBreakPointWindow;
|
class CBreakPointWindow;
|
||||||
class CMemoryWindow;
|
class CMemoryWindow;
|
||||||
class CJitWindow;
|
class CJitWindow;
|
||||||
|
@ -77,6 +78,7 @@ public:
|
||||||
|
|
||||||
void ToggleCodeWindow(bool bShow);
|
void ToggleCodeWindow(bool bShow);
|
||||||
void ToggleRegisterWindow(bool bShow);
|
void ToggleRegisterWindow(bool bShow);
|
||||||
|
void ToggleWatchWindow(bool bShow);
|
||||||
void ToggleBreakPointWindow(bool bShow);
|
void ToggleBreakPointWindow(bool bShow);
|
||||||
void ToggleMemoryWindow(bool bShow);
|
void ToggleMemoryWindow(bool bShow);
|
||||||
void ToggleJitWindow(bool bShow);
|
void ToggleJitWindow(bool bShow);
|
||||||
|
@ -93,6 +95,7 @@ public:
|
||||||
|
|
||||||
// Sub dialogs
|
// Sub dialogs
|
||||||
CRegisterWindow* m_RegisterWindow;
|
CRegisterWindow* m_RegisterWindow;
|
||||||
|
CWatchWindow* m_WatchWindow;
|
||||||
CBreakPointWindow* m_BreakpointWindow;
|
CBreakPointWindow* m_BreakpointWindow;
|
||||||
CMemoryWindow* m_MemoryWindow;
|
CMemoryWindow* m_MemoryWindow;
|
||||||
CJitWindow* m_JitWindow;
|
CJitWindow* m_JitWindow;
|
||||||
|
@ -126,6 +129,7 @@ private:
|
||||||
// Debugger functions
|
// Debugger functions
|
||||||
void SingleStep();
|
void SingleStep();
|
||||||
void StepOver();
|
void StepOver();
|
||||||
|
void StepOut();
|
||||||
void ToggleBreakpoint();
|
void ToggleBreakpoint();
|
||||||
|
|
||||||
void UpdateLists();
|
void UpdateLists();
|
||||||
|
|
|
@ -51,6 +51,7 @@
|
||||||
#include "DolphinWX/Debugger/JitWindow.h"
|
#include "DolphinWX/Debugger/JitWindow.h"
|
||||||
#include "DolphinWX/Debugger/MemoryWindow.h"
|
#include "DolphinWX/Debugger/MemoryWindow.h"
|
||||||
#include "DolphinWX/Debugger/RegisterWindow.h"
|
#include "DolphinWX/Debugger/RegisterWindow.h"
|
||||||
|
#include "DolphinWX/Debugger/WatchWindow.h"
|
||||||
|
|
||||||
|
|
||||||
// Save and load settings
|
// Save and load settings
|
||||||
|
@ -421,6 +422,8 @@ void CCodeWindow::OpenPages()
|
||||||
Parent->ToggleLogConfigWindow(true);
|
Parent->ToggleLogConfigWindow(true);
|
||||||
if (bShowOnStart[IDM_REGISTERWINDOW - IDM_LOGWINDOW])
|
if (bShowOnStart[IDM_REGISTERWINDOW - IDM_LOGWINDOW])
|
||||||
ToggleRegisterWindow(true);
|
ToggleRegisterWindow(true);
|
||||||
|
if (bShowOnStart[IDM_WATCHWINDOW - IDM_LOGWINDOW])
|
||||||
|
ToggleWatchWindow(true);
|
||||||
if (bShowOnStart[IDM_BREAKPOINTWINDOW - IDM_LOGWINDOW])
|
if (bShowOnStart[IDM_BREAKPOINTWINDOW - IDM_LOGWINDOW])
|
||||||
ToggleBreakPointWindow(true);
|
ToggleBreakPointWindow(true);
|
||||||
if (bShowOnStart[IDM_MEMORYWINDOW - IDM_LOGWINDOW])
|
if (bShowOnStart[IDM_MEMORYWINDOW - IDM_LOGWINDOW])
|
||||||
|
@ -461,6 +464,24 @@ void CCodeWindow::ToggleRegisterWindow(bool bShow)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void CCodeWindow::ToggleWatchWindow(bool bShow)
|
||||||
|
{
|
||||||
|
GetMenuBar()->FindItem(IDM_WATCHWINDOW)->Check(bShow);
|
||||||
|
if (bShow)
|
||||||
|
{
|
||||||
|
if (!m_WatchWindow)
|
||||||
|
m_WatchWindow = new CWatchWindow(Parent, IDM_WATCHWINDOW);
|
||||||
|
Parent->DoAddPage(m_WatchWindow,
|
||||||
|
iNbAffiliation[IDM_WATCHWINDOW - IDM_LOGWINDOW],
|
||||||
|
Parent->bFloatWindow[IDM_WATCHWINDOW - IDM_LOGWINDOW]);
|
||||||
|
}
|
||||||
|
else // Close
|
||||||
|
{
|
||||||
|
Parent->DoRemovePage(m_WatchWindow, false);
|
||||||
|
m_WatchWindow = nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void CCodeWindow::ToggleBreakPointWindow(bool bShow)
|
void CCodeWindow::ToggleBreakPointWindow(bool bShow)
|
||||||
{
|
{
|
||||||
GetMenuBar()->FindItem(IDM_BREAKPOINTWINDOW)->Check(bShow);
|
GetMenuBar()->FindItem(IDM_BREAKPOINTWINDOW)->Check(bShow);
|
||||||
|
|
|
@ -25,10 +25,13 @@
|
||||||
#include "Common/DebugInterface.h"
|
#include "Common/DebugInterface.h"
|
||||||
#include "Common/StringUtil.h"
|
#include "Common/StringUtil.h"
|
||||||
#include "Core/HW/Memmap.h"
|
#include "Core/HW/Memmap.h"
|
||||||
|
#include "DolphinWX/Frame.h"
|
||||||
#include "DolphinWX/Globals.h"
|
#include "DolphinWX/Globals.h"
|
||||||
#include "DolphinWX/WxUtils.h"
|
#include "DolphinWX/WxUtils.h"
|
||||||
|
#include "DolphinWX/Debugger/CodeWindow.h"
|
||||||
#include "DolphinWX/Debugger/DebuggerUIUtil.h"
|
#include "DolphinWX/Debugger/DebuggerUIUtil.h"
|
||||||
#include "DolphinWX/Debugger/MemoryView.h"
|
#include "DolphinWX/Debugger/MemoryView.h"
|
||||||
|
#include "DolphinWX/Debugger/WatchWindow.h"
|
||||||
|
|
||||||
enum
|
enum
|
||||||
{
|
{
|
||||||
|
@ -38,6 +41,7 @@ enum
|
||||||
IDM_COPYCODE,
|
IDM_COPYCODE,
|
||||||
IDM_RUNTOHERE,
|
IDM_RUNTOHERE,
|
||||||
IDM_DYNARECRESULTS,
|
IDM_DYNARECRESULTS,
|
||||||
|
IDM_WATCHADDRESS,
|
||||||
IDM_TOGGLEMEMORY,
|
IDM_TOGGLEMEMORY,
|
||||||
IDM_VIEWASFP,
|
IDM_VIEWASFP,
|
||||||
IDM_VIEWASASCII,
|
IDM_VIEWASASCII,
|
||||||
|
@ -161,6 +165,10 @@ void CMemoryView::OnScrollWheel(wxMouseEvent& event)
|
||||||
|
|
||||||
void CMemoryView::OnPopupMenu(wxCommandEvent& event)
|
void CMemoryView::OnPopupMenu(wxCommandEvent& event)
|
||||||
{
|
{
|
||||||
|
CFrame* main_frame = (CFrame*)(GetParent()->GetParent()->GetParent());
|
||||||
|
CCodeWindow* code_window = main_frame->g_pCodeWindow;
|
||||||
|
CWatchWindow* watch_window = code_window->m_WatchWindow;
|
||||||
|
|
||||||
#if wxUSE_CLIPBOARD
|
#if wxUSE_CLIPBOARD
|
||||||
wxTheClipboard->Open();
|
wxTheClipboard->Open();
|
||||||
#endif
|
#endif
|
||||||
|
@ -180,6 +188,13 @@ void CMemoryView::OnPopupMenu(wxCommandEvent& event)
|
||||||
break;
|
break;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
case IDM_WATCHADDRESS:
|
||||||
|
debugger->AddWatch(selection);
|
||||||
|
if (watch_window)
|
||||||
|
watch_window->NotifyUpdate();
|
||||||
|
Refresh();
|
||||||
|
break;
|
||||||
|
|
||||||
case IDM_TOGGLEMEMORY:
|
case IDM_TOGGLEMEMORY:
|
||||||
memory ^= 1;
|
memory ^= 1;
|
||||||
Refresh();
|
Refresh();
|
||||||
|
@ -215,6 +230,7 @@ void CMemoryView::OnMouseDownR(wxMouseEvent& event)
|
||||||
menu->Append(IDM_COPYADDRESS, _("Copy &address"));
|
menu->Append(IDM_COPYADDRESS, _("Copy &address"));
|
||||||
menu->Append(IDM_COPYHEX, _("Copy &hex"));
|
menu->Append(IDM_COPYHEX, _("Copy &hex"));
|
||||||
#endif
|
#endif
|
||||||
|
menu->Append(IDM_WATCHADDRESS, _("Add to &watch"));
|
||||||
menu->Append(IDM_TOGGLEMEMORY, _("Toggle &memory"));
|
menu->Append(IDM_TOGGLEMEMORY, _("Toggle &memory"));
|
||||||
|
|
||||||
wxMenu* viewAsSubMenu = new wxMenu;
|
wxMenu* viewAsSubMenu = new wxMenu;
|
||||||
|
|
|
@ -6,6 +6,7 @@
|
||||||
#include <wx/colour.h>
|
#include <wx/colour.h>
|
||||||
#include <wx/defs.h>
|
#include <wx/defs.h>
|
||||||
#include <wx/grid.h>
|
#include <wx/grid.h>
|
||||||
|
#include <wx/menu.h>
|
||||||
#include <wx/string.h>
|
#include <wx/string.h>
|
||||||
#include <wx/windowid.h>
|
#include <wx/windowid.h>
|
||||||
|
|
||||||
|
@ -15,14 +16,25 @@
|
||||||
#include "Core/HW/ProcessorInterface.h"
|
#include "Core/HW/ProcessorInterface.h"
|
||||||
#include "Core/PowerPC/Gekko.h"
|
#include "Core/PowerPC/Gekko.h"
|
||||||
#include "Core/PowerPC/PowerPC.h"
|
#include "Core/PowerPC/PowerPC.h"
|
||||||
|
#include "DolphinWX/Frame.h"
|
||||||
|
#include "DolphinWX/Globals.h"
|
||||||
#include "DolphinWX/WxUtils.h"
|
#include "DolphinWX/WxUtils.h"
|
||||||
|
#include "DolphinWX/Debugger/CodeWindow.h"
|
||||||
#include "DolphinWX/Debugger/DebuggerUIUtil.h"
|
#include "DolphinWX/Debugger/DebuggerUIUtil.h"
|
||||||
|
#include "DolphinWX/Debugger/MemoryWindow.h"
|
||||||
#include "DolphinWX/Debugger/RegisterView.h"
|
#include "DolphinWX/Debugger/RegisterView.h"
|
||||||
|
#include "DolphinWX/Debugger/WatchWindow.h"
|
||||||
|
|
||||||
class wxWindow;
|
class wxWindow;
|
||||||
|
|
||||||
// F-zero 80005e60 wtf??
|
// F-zero 80005e60 wtf??
|
||||||
|
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
IDM_WATCHADDRESS,
|
||||||
|
IDM_VIEWMEMORY,
|
||||||
|
};
|
||||||
|
|
||||||
static const char *special_reg_names[] = {
|
static const char *special_reg_names[] = {
|
||||||
"PC", "LR", "CTR", "CR", "FPSCR", "MSR", "SRR0", "SRR1", "Exceptions", "Int Mask", "Int Cause", "DSISR", "DAR", "PT hashmask"
|
"PC", "LR", "CTR", "CR", "FPSCR", "MSR", "SRR0", "SRR1", "Exceptions", "Int Mask", "Int Cause", "DSISR", "DAR", "PT hashmask"
|
||||||
};
|
};
|
||||||
|
@ -49,7 +61,7 @@ static u32 GetSpecialRegValue(int reg)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
wxString CRegTable::GetValue(int row, int col)
|
static wxString GetValueByRowCol(int row, int col)
|
||||||
{
|
{
|
||||||
if (row < 32)
|
if (row < 32)
|
||||||
{
|
{
|
||||||
|
@ -130,6 +142,11 @@ wxString CRegTable::GetValue(int row, int col)
|
||||||
return wxEmptyString;
|
return wxEmptyString;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
wxString CRegTable::GetValue(int row, int col)
|
||||||
|
{
|
||||||
|
return GetValueByRowCol(row, col);
|
||||||
|
}
|
||||||
|
|
||||||
static void SetSpecialRegValue(int reg, u32 value)
|
static void SetSpecialRegValue(int reg, u32 value)
|
||||||
{
|
{
|
||||||
switch (reg)
|
switch (reg)
|
||||||
|
@ -246,3 +263,42 @@ void CRegisterView::Update()
|
||||||
ForceRefresh();
|
ForceRefresh();
|
||||||
((CRegTable *)GetTable())->UpdateCachedRegs();
|
((CRegTable *)GetTable())->UpdateCachedRegs();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void CRegisterView::OnMouseDownR(wxGridEvent& event)
|
||||||
|
{
|
||||||
|
// popup menu
|
||||||
|
int row = event.GetRow();
|
||||||
|
int col = event.GetCol();
|
||||||
|
|
||||||
|
wxString strNewVal = GetValueByRowCol(row, col);
|
||||||
|
TryParse("0x" + WxStrToStr(strNewVal), &m_selectedAddress);
|
||||||
|
|
||||||
|
wxMenu* menu = new wxMenu;
|
||||||
|
menu->Append(IDM_WATCHADDRESS, _("Add to &watch"));
|
||||||
|
menu->Append(IDM_VIEWMEMORY, _("View &memory"));
|
||||||
|
PopupMenu(menu);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CRegisterView::OnPopupMenu(wxCommandEvent& event)
|
||||||
|
{
|
||||||
|
CFrame* main_frame = (CFrame*)(GetParent()->GetParent());
|
||||||
|
CCodeWindow* code_window = main_frame->g_pCodeWindow;
|
||||||
|
CWatchWindow* watch_window = code_window->m_WatchWindow;
|
||||||
|
CMemoryWindow* memory_window = code_window->m_MemoryWindow;
|
||||||
|
|
||||||
|
switch (event.GetId())
|
||||||
|
{
|
||||||
|
case IDM_WATCHADDRESS:
|
||||||
|
PowerPC::watches.Add(m_selectedAddress);
|
||||||
|
if (watch_window)
|
||||||
|
watch_window->NotifyUpdate();
|
||||||
|
Refresh();
|
||||||
|
break;
|
||||||
|
case IDM_VIEWMEMORY:
|
||||||
|
if (memory_window)
|
||||||
|
memory_window->JumpToAddress(m_selectedAddress);
|
||||||
|
Refresh();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
event.Skip();
|
||||||
|
}
|
||||||
|
|
|
@ -30,14 +30,13 @@ class wxWindow;
|
||||||
// Interrupt Mask (PI)
|
// Interrupt Mask (PI)
|
||||||
// Interrupt Cause(PI)
|
// Interrupt Cause(PI)
|
||||||
|
|
||||||
|
#define NUM_SPECIALS 14
|
||||||
|
|
||||||
class CRegTable : public wxGridTableBase
|
class CRegTable : public wxGridTableBase
|
||||||
{
|
{
|
||||||
enum
|
|
||||||
{
|
|
||||||
NUM_SPECIALS = 14,
|
|
||||||
};
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
CRegTable()
|
CRegTable()
|
||||||
{
|
{
|
||||||
memset(m_CachedRegs, 0, sizeof(m_CachedRegs));
|
memset(m_CachedRegs, 0, sizeof(m_CachedRegs));
|
||||||
|
@ -72,4 +71,9 @@ class CRegisterView : public wxGrid
|
||||||
public:
|
public:
|
||||||
CRegisterView(wxWindow* parent, wxWindowID id);
|
CRegisterView(wxWindow* parent, wxWindowID id);
|
||||||
void Update() override;
|
void Update() override;
|
||||||
|
void OnMouseDownR(wxGridEvent& event);
|
||||||
|
void OnPopupMenu(wxCommandEvent& event);
|
||||||
|
|
||||||
|
private:
|
||||||
|
u32 m_selectedAddress = 0;
|
||||||
};
|
};
|
||||||
|
|
|
@ -11,12 +11,15 @@
|
||||||
#include <wx/string.h>
|
#include <wx/string.h>
|
||||||
#include <wx/windowid.h>
|
#include <wx/windowid.h>
|
||||||
|
|
||||||
|
#include "Core/PowerPC/PowerPC.h"
|
||||||
#include "DolphinWX/Debugger/RegisterView.h"
|
#include "DolphinWX/Debugger/RegisterView.h"
|
||||||
#include "DolphinWX/Debugger/RegisterWindow.h"
|
#include "DolphinWX/Debugger/RegisterWindow.h"
|
||||||
|
|
||||||
class wxWindow;
|
class wxWindow;
|
||||||
|
|
||||||
BEGIN_EVENT_TABLE(CRegisterWindow, wxPanel)
|
BEGIN_EVENT_TABLE(CRegisterWindow, wxPanel)
|
||||||
|
EVT_GRID_CELL_RIGHT_CLICK(CRegisterView::OnMouseDownR)
|
||||||
|
EVT_MENU(-1, CRegisterView::OnPopupMenu)
|
||||||
END_EVENT_TABLE()
|
END_EVENT_TABLE()
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,307 @@
|
||||||
|
// Copyright 2013 Dolphin Emulator Project
|
||||||
|
// Licensed under GPLv2
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#include <wx/chartype.h>
|
||||||
|
#include <wx/colour.h>
|
||||||
|
#include <wx/defs.h>
|
||||||
|
#include <wx/grid.h>
|
||||||
|
#include <wx/menu.h>
|
||||||
|
#include <wx/string.h>
|
||||||
|
#include <wx/windowid.h>
|
||||||
|
|
||||||
|
#include "Common/GekkoDisassembler.h"
|
||||||
|
#include "Core/HW/Memmap.h"
|
||||||
|
#include "Core/PowerPC/PowerPC.h"
|
||||||
|
#include "DolphinWX/Frame.h"
|
||||||
|
#include "DolphinWX/WxUtils.h"
|
||||||
|
#include "DolphinWX/Debugger/BreakpointWindow.h"
|
||||||
|
#include "DolphinWX/Debugger/CodeWindow.h"
|
||||||
|
#include "DolphinWX/Debugger/DebuggerUIUtil.h"
|
||||||
|
#include "DolphinWX/Debugger/MemoryWindow.h"
|
||||||
|
#include "DolphinWX/Debugger/RegisterView.h"
|
||||||
|
#include "DolphinWX/Debugger/WatchView.h"
|
||||||
|
#include "DolphinWX/Debugger/WatchWindow.h"
|
||||||
|
|
||||||
|
class wxWindow;
|
||||||
|
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
IDM_DELETEWATCH,
|
||||||
|
IDM_ADDMEMCHECK,
|
||||||
|
IDM_VIEWMEMORY,
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
static std::string GetWatchName(int count)
|
||||||
|
{
|
||||||
|
return PowerPC::watches.GetWatches().at(count - 1).name;
|
||||||
|
}
|
||||||
|
|
||||||
|
static u32 GetWatchAddr(int count)
|
||||||
|
{
|
||||||
|
return PowerPC::watches.GetWatches().at(count - 1).iAddress;
|
||||||
|
}
|
||||||
|
|
||||||
|
static u32 GetWatchValue(int count)
|
||||||
|
{
|
||||||
|
return Memory::ReadUnchecked_U32(GetWatchAddr(count));
|
||||||
|
}
|
||||||
|
|
||||||
|
static void AddWatchAddr(int count, u32 value)
|
||||||
|
{
|
||||||
|
PowerPC::watches.Add(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void UpdateWatchAddr(int count, u32 value)
|
||||||
|
{
|
||||||
|
PowerPC::watches.Update(count - 1, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void SetWatchName(int count, const std::string value)
|
||||||
|
{
|
||||||
|
if ((count - 1) < PowerPC::watches.GetWatches().size())
|
||||||
|
{
|
||||||
|
PowerPC::watches.UpdateName(count - 1, value);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
PowerPC::watches.Add(0);
|
||||||
|
PowerPC::watches.UpdateName(PowerPC::watches.GetWatches().size() - 1, value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void SetWatchValue(int count, u32 value)
|
||||||
|
{
|
||||||
|
Memory::WriteUnchecked_U32(value, GetWatchAddr(count));
|
||||||
|
}
|
||||||
|
|
||||||
|
static wxString GetValueByRowCol(int row, int col)
|
||||||
|
{
|
||||||
|
if (row == 0)
|
||||||
|
{
|
||||||
|
// Column Labels
|
||||||
|
switch (col)
|
||||||
|
{
|
||||||
|
case 0: return wxString::Format("Label");
|
||||||
|
case 1: return wxString::Format("Addr");
|
||||||
|
case 2: return wxString::Format("Hex");
|
||||||
|
case 3: return wxString::Format("Dec");
|
||||||
|
case 4: return wxString::Format("Str");
|
||||||
|
default: return wxEmptyString;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (row <= PowerPC::watches.GetWatches().size())
|
||||||
|
{
|
||||||
|
if (PowerPC::GetState() != PowerPC::CPU_POWERDOWN)
|
||||||
|
{
|
||||||
|
switch (col)
|
||||||
|
{
|
||||||
|
case 0: return wxString::Format("%s", GetWatchName(row));
|
||||||
|
case 1: return wxString::Format("%08x", GetWatchAddr(row));
|
||||||
|
case 2: return wxString::Format("%08x", GetWatchValue(row));
|
||||||
|
case 3: return wxString::Format("%lu", GetWatchValue(row));
|
||||||
|
case 4:
|
||||||
|
{
|
||||||
|
u32 addr = GetWatchAddr(row);
|
||||||
|
if (Memory::IsRAMAddress(addr))
|
||||||
|
return Memory::GetString(addr, 32).c_str();
|
||||||
|
else
|
||||||
|
return wxEmptyString;
|
||||||
|
}
|
||||||
|
default: return wxEmptyString;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return wxEmptyString;
|
||||||
|
}
|
||||||
|
|
||||||
|
wxString CWatchTable::GetValue(int row, int col)
|
||||||
|
{
|
||||||
|
return GetValueByRowCol(row, col);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CWatchTable::SetValue(int row, int col, const wxString& strNewVal)
|
||||||
|
{
|
||||||
|
u32 newVal = 0;
|
||||||
|
if (col == 0 || TryParse("0x" + WxStrToStr(strNewVal), &newVal))
|
||||||
|
{
|
||||||
|
if (row > 0)
|
||||||
|
{
|
||||||
|
switch (col)
|
||||||
|
{
|
||||||
|
case 0:
|
||||||
|
{
|
||||||
|
SetWatchName(row, std::string(WxStrToStr(strNewVal)));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 1:
|
||||||
|
{
|
||||||
|
if (row > (int)PowerPC::watches.GetWatches().size())
|
||||||
|
{
|
||||||
|
AddWatchAddr(row, newVal);
|
||||||
|
row = (int)PowerPC::watches.GetWatches().size();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
UpdateWatchAddr(row, newVal);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 2:
|
||||||
|
{
|
||||||
|
SetWatchValue(row, newVal);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void CWatchTable::UpdateWatch()
|
||||||
|
{
|
||||||
|
for (int i = 0; i < (int)PowerPC::watches.GetWatches().size(); ++i)
|
||||||
|
{
|
||||||
|
m_CachedWatchHasChanged[i] = (m_CachedWatch[i] != GetWatchValue(i + 1));
|
||||||
|
m_CachedWatch[i] = GetWatchValue(i + 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
wxGridCellAttr* CWatchTable::GetAttr(int row, int col, wxGridCellAttr::wxAttrKind)
|
||||||
|
{
|
||||||
|
wxGridCellAttr* attr = new wxGridCellAttr();
|
||||||
|
|
||||||
|
attr->SetBackgroundColour(*wxWHITE);
|
||||||
|
attr->SetFont(DebuggerFont);
|
||||||
|
|
||||||
|
switch (col)
|
||||||
|
{
|
||||||
|
case 1:
|
||||||
|
attr->SetAlignment(wxALIGN_LEFT, wxALIGN_CENTER);
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
case 4:
|
||||||
|
attr->SetAlignment(wxALIGN_LEFT, wxALIGN_CENTER);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
attr->SetAlignment(wxALIGN_LEFT, wxALIGN_CENTER);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (row == 0)
|
||||||
|
{
|
||||||
|
attr->SetReadOnly(true);
|
||||||
|
attr->SetBackgroundColour(*wxBLACK);
|
||||||
|
attr->SetTextColour(*wxWHITE);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
bool red = false;
|
||||||
|
if (col == 1)
|
||||||
|
red = m_CachedWatchHasChanged[row];
|
||||||
|
|
||||||
|
attr->SetTextColour(red ? *wxRED : *wxBLACK);
|
||||||
|
|
||||||
|
if (row > (int)(PowerPC::watches.GetWatches().size() + 1) || (PowerPC::GetState() == PowerPC::CPU_POWERDOWN))
|
||||||
|
{
|
||||||
|
attr->SetReadOnly(true);
|
||||||
|
attr->SetBackgroundColour(*wxLIGHT_GREY);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
attr->IncRef();
|
||||||
|
return attr;
|
||||||
|
}
|
||||||
|
|
||||||
|
CWatchView::CWatchView(wxWindow* parent, wxWindowID id)
|
||||||
|
: wxGrid(parent, id)
|
||||||
|
{
|
||||||
|
SetTable(new CWatchTable(), false);
|
||||||
|
SetRowLabelSize(0);
|
||||||
|
SetColLabelSize(0);
|
||||||
|
DisableDragRowSize();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CWatchView::Update()
|
||||||
|
{
|
||||||
|
if (PowerPC::GetState() != PowerPC::CPU_POWERDOWN)
|
||||||
|
{
|
||||||
|
ForceRefresh();
|
||||||
|
((CWatchTable *)GetTable())->UpdateWatch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void CWatchView::OnMouseDownR(wxGridEvent& event)
|
||||||
|
{
|
||||||
|
// popup menu
|
||||||
|
int row = event.GetRow();
|
||||||
|
int col = event.GetCol();
|
||||||
|
|
||||||
|
m_selectedRow = row;
|
||||||
|
|
||||||
|
if (col == 1 || col == 2)
|
||||||
|
{
|
||||||
|
wxString strNewVal = GetValueByRowCol(row, col);
|
||||||
|
TryParse("0x" + WxStrToStr(strNewVal), &m_selectedAddress);
|
||||||
|
}
|
||||||
|
|
||||||
|
wxMenu* menu = new wxMenu;
|
||||||
|
if (row != 0 && row != (int)(PowerPC::watches.GetWatches().size() + 1))
|
||||||
|
menu->Append(IDM_DELETEWATCH, _("&Delete watch"));
|
||||||
|
|
||||||
|
if (row != 0 && row != (int)(PowerPC::watches.GetWatches().size() + 1) && (col == 1 || col == 2))
|
||||||
|
{
|
||||||
|
#ifdef ENABLE_MEM_CHECK
|
||||||
|
menu->Append(IDM_ADDMEMCHECK, _("Add memory &breakpoint"));
|
||||||
|
#endif
|
||||||
|
menu->Append(IDM_VIEWMEMORY, _("View &memory"));
|
||||||
|
}
|
||||||
|
PopupMenu(menu);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CWatchView::OnPopupMenu(wxCommandEvent& event)
|
||||||
|
{
|
||||||
|
CFrame* main_frame = (CFrame*)(GetParent()->GetParent());
|
||||||
|
CCodeWindow* code_window = main_frame->g_pCodeWindow;
|
||||||
|
CWatchWindow* watch_window = code_window->m_WatchWindow;
|
||||||
|
CMemoryWindow* memory_window = code_window->m_MemoryWindow;
|
||||||
|
CBreakPointWindow* breakpoint_window = code_window->m_BreakpointWindow;
|
||||||
|
|
||||||
|
wxString strNewVal;
|
||||||
|
TMemCheck MemCheck;
|
||||||
|
|
||||||
|
switch (event.GetId())
|
||||||
|
{
|
||||||
|
case IDM_DELETEWATCH:
|
||||||
|
strNewVal = GetValueByRowCol(m_selectedRow, 1);
|
||||||
|
if (TryParse("0x" + WxStrToStr(strNewVal), &m_selectedAddress))
|
||||||
|
{
|
||||||
|
PowerPC::watches.Remove(m_selectedAddress);
|
||||||
|
if (watch_window)
|
||||||
|
watch_window->NotifyUpdate();
|
||||||
|
Refresh();
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case IDM_ADDMEMCHECK:
|
||||||
|
MemCheck.StartAddress = m_selectedAddress;
|
||||||
|
MemCheck.EndAddress = m_selectedAddress;
|
||||||
|
MemCheck.bRange = m_selectedAddress != m_selectedAddress;
|
||||||
|
MemCheck.OnRead = true;
|
||||||
|
MemCheck.OnWrite = true;
|
||||||
|
MemCheck.Log = true;
|
||||||
|
MemCheck.Break = true;
|
||||||
|
PowerPC::memchecks.Add(MemCheck);
|
||||||
|
|
||||||
|
if (breakpoint_window)
|
||||||
|
breakpoint_window->NotifyUpdate();
|
||||||
|
Refresh();
|
||||||
|
break;
|
||||||
|
case IDM_VIEWMEMORY:
|
||||||
|
if (memory_window)
|
||||||
|
memory_window->JumpToAddress(m_selectedAddress);
|
||||||
|
Refresh();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
event.Skip();
|
||||||
|
}
|
|
@ -0,0 +1,55 @@
|
||||||
|
// Copyright 2013 Dolphin Emulator Project
|
||||||
|
// Licensed under GPLv2
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <array>
|
||||||
|
#include <string>
|
||||||
|
#include <wx/defs.h>
|
||||||
|
#include <wx/grid.h>
|
||||||
|
#include <wx/string.h>
|
||||||
|
#include <wx/windowid.h>
|
||||||
|
|
||||||
|
#include "Common/CommonTypes.h"
|
||||||
|
|
||||||
|
class wxWindow;
|
||||||
|
|
||||||
|
class CWatchTable : public wxGridTableBase
|
||||||
|
{
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
MAX_SPECIALS = 256,
|
||||||
|
};
|
||||||
|
|
||||||
|
public:
|
||||||
|
CWatchTable()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
int GetNumberCols() override { return 5; }
|
||||||
|
int GetNumberRows() override { return MAX_SPECIALS; }
|
||||||
|
wxString GetValue(int row, int col) override;
|
||||||
|
void SetValue(int row, int col, const wxString&) override;
|
||||||
|
wxGridCellAttr* GetAttr(int, int, wxGridCellAttr::wxAttrKind) override;
|
||||||
|
void UpdateWatch();
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::array<u32, MAX_SPECIALS> m_CachedWatch;
|
||||||
|
std::array<bool, MAX_SPECIALS> m_CachedWatchHasChanged;
|
||||||
|
|
||||||
|
DECLARE_NO_COPY_CLASS(CWatchTable);
|
||||||
|
};
|
||||||
|
|
||||||
|
class CWatchView : public wxGrid
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
CWatchView(wxWindow* parent, wxWindowID id);
|
||||||
|
void Update() override;
|
||||||
|
void OnMouseDownR(wxGridEvent& event);
|
||||||
|
void OnPopupMenu(wxCommandEvent& event);
|
||||||
|
|
||||||
|
private:
|
||||||
|
u32 m_selectedAddress = 0;
|
||||||
|
u32 m_selectedRow = 0;
|
||||||
|
};
|
|
@ -0,0 +1,137 @@
|
||||||
|
// Copyright 2013 Dolphin Emulator Project
|
||||||
|
// Licensed under GPLv2
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#include <cstddef>
|
||||||
|
|
||||||
|
#include <wx/bitmap.h>
|
||||||
|
#include <wx/defs.h>
|
||||||
|
#include <wx/event.h>
|
||||||
|
#include <wx/gdicmn.h>
|
||||||
|
#include <wx/panel.h>
|
||||||
|
#include <wx/sizer.h>
|
||||||
|
#include <wx/string.h>
|
||||||
|
#include <wx/windowid.h>
|
||||||
|
#include <wx/aui/auibar.h>
|
||||||
|
|
||||||
|
#include "Common/FileUtil.h"
|
||||||
|
#include "Common/IniFile.h"
|
||||||
|
#include "Core/ConfigManager.h"
|
||||||
|
#include "Core/PowerPC/PowerPC.h"
|
||||||
|
#include "DolphinWX/WxUtils.h"
|
||||||
|
#include "DolphinWX/Debugger/WatchView.h"
|
||||||
|
#include "DolphinWX/Debugger/WatchWindow.h"
|
||||||
|
|
||||||
|
extern "C" {
|
||||||
|
#include "DolphinWX/resources/toolbar_debugger_delete.c"
|
||||||
|
}
|
||||||
|
|
||||||
|
class wxWindow;
|
||||||
|
|
||||||
|
BEGIN_EVENT_TABLE(CWatchWindow, wxPanel)
|
||||||
|
EVT_GRID_CELL_RIGHT_CLICK(CWatchView::OnMouseDownR)
|
||||||
|
EVT_MENU(-1, CWatchView::OnPopupMenu)
|
||||||
|
END_EVENT_TABLE()
|
||||||
|
|
||||||
|
class CWatchToolbar : public wxAuiToolBar
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
CWatchToolbar(CWatchWindow* parent, const wxWindowID id)
|
||||||
|
: wxAuiToolBar(parent, id, wxDefaultPosition, wxDefaultSize,
|
||||||
|
wxAUI_TB_DEFAULT_STYLE | wxAUI_TB_TEXT)
|
||||||
|
{
|
||||||
|
SetToolBitmapSize(wxSize(16, 16));
|
||||||
|
|
||||||
|
m_Bitmaps[Toolbar_File] = wxBitmap(wxGetBitmapFromMemory(toolbar_delete_png).ConvertToImage().Rescale(16, 16));
|
||||||
|
|
||||||
|
AddTool(ID_LOAD, _("Load"), m_Bitmaps[Toolbar_File]);
|
||||||
|
Bind(wxEVT_TOOL, &CWatchWindow::Event_LoadAll, parent, ID_LOAD);
|
||||||
|
|
||||||
|
AddTool(ID_SAVE, _("Save"), m_Bitmaps[Toolbar_File]);
|
||||||
|
Bind(wxEVT_TOOL, &CWatchWindow::Event_SaveAll, parent, ID_SAVE);
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
Toolbar_File,
|
||||||
|
Num_Bitmaps
|
||||||
|
};
|
||||||
|
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
ID_LOAD,
|
||||||
|
ID_SAVE
|
||||||
|
};
|
||||||
|
|
||||||
|
wxBitmap m_Bitmaps[Num_Bitmaps];
|
||||||
|
};
|
||||||
|
|
||||||
|
CWatchWindow::CWatchWindow(wxWindow* parent, wxWindowID id,
|
||||||
|
const wxPoint& position, const wxSize& size,
|
||||||
|
long style, const wxString& name)
|
||||||
|
: wxPanel(parent, id, position, size, style, name)
|
||||||
|
, m_GPRGridView(nullptr)
|
||||||
|
{
|
||||||
|
m_mgr.SetManagedWindow(this);
|
||||||
|
m_mgr.SetFlags(wxAUI_MGR_DEFAULT | wxAUI_MGR_LIVE_RESIZE);
|
||||||
|
|
||||||
|
wxBoxSizer *sGrid = new wxBoxSizer(wxVERTICAL);
|
||||||
|
m_GPRGridView = new CWatchView(this, ID_GPR);
|
||||||
|
sGrid->Add(m_GPRGridView, 1, wxGROW);
|
||||||
|
SetSizer(sGrid);
|
||||||
|
|
||||||
|
m_mgr.AddPane(new CWatchToolbar(this, wxID_ANY), wxAuiPaneInfo().ToolbarPane().Top().
|
||||||
|
LeftDockable(true).RightDockable(true).BottomDockable(false).Floatable(false));
|
||||||
|
m_mgr.AddPane(m_GPRGridView, wxAuiPaneInfo().CenterPane());
|
||||||
|
m_mgr.Update();
|
||||||
|
}
|
||||||
|
|
||||||
|
CWatchWindow::~CWatchWindow()
|
||||||
|
{
|
||||||
|
m_mgr.UnInit();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CWatchWindow::NotifyUpdate()
|
||||||
|
{
|
||||||
|
if (m_GPRGridView != nullptr)
|
||||||
|
m_GPRGridView->Update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CWatchWindow::Event_SaveAll(wxCommandEvent& WXUNUSED(event))
|
||||||
|
{
|
||||||
|
SaveAll();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CWatchWindow::SaveAll()
|
||||||
|
{
|
||||||
|
IniFile ini;
|
||||||
|
ini.Load(File::GetUserPath(D_GAMESETTINGS_IDX) + SConfig::GetInstance().m_LocalCoreStartupParameter.GetUniqueID() + ".ini", false);
|
||||||
|
ini.SetLines("Watches", PowerPC::watches.GetStrings());
|
||||||
|
ini.Save(File::GetUserPath(D_GAMESETTINGS_IDX) + SConfig::GetInstance().m_LocalCoreStartupParameter.GetUniqueID() + ".ini");
|
||||||
|
}
|
||||||
|
|
||||||
|
void CWatchWindow::Event_LoadAll(wxCommandEvent& WXUNUSED(event))
|
||||||
|
{
|
||||||
|
LoadAll();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CWatchWindow::LoadAll()
|
||||||
|
{
|
||||||
|
IniFile ini;
|
||||||
|
Watches::TWatchesStr watches;
|
||||||
|
|
||||||
|
if (!ini.Load(File::GetUserPath(D_GAMESETTINGS_IDX) + SConfig::GetInstance().m_LocalCoreStartupParameter.GetUniqueID() + ".ini", false))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ini.GetLines("Watches", &watches, false))
|
||||||
|
{
|
||||||
|
PowerPC::watches.Clear();
|
||||||
|
PowerPC::watches.AddFromStrings(watches);
|
||||||
|
}
|
||||||
|
|
||||||
|
NotifyUpdate();
|
||||||
|
}
|
|
@ -0,0 +1,49 @@
|
||||||
|
// Copyright 2013 Dolphin Emulator Project
|
||||||
|
// Licensed under GPLv2
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <wx/defs.h>
|
||||||
|
#include <wx/event.h>
|
||||||
|
#include <wx/gdicmn.h>
|
||||||
|
#include <wx/panel.h>
|
||||||
|
#include <wx/string.h>
|
||||||
|
#include <wx/translation.h>
|
||||||
|
#include <wx/windowid.h>
|
||||||
|
#include <wx/aui/framemanager.h>
|
||||||
|
|
||||||
|
class CWatchView;
|
||||||
|
class wxWindow;
|
||||||
|
|
||||||
|
class CWatchWindow
|
||||||
|
: public wxPanel
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
CWatchWindow(wxWindow* parent,
|
||||||
|
wxWindowID id = wxID_ANY,
|
||||||
|
const wxPoint& pos = wxDefaultPosition,
|
||||||
|
const wxSize& size = wxDefaultSize,
|
||||||
|
long style = wxTAB_TRAVERSAL | wxNO_BORDER,
|
||||||
|
const wxString& name = _("Watch"));
|
||||||
|
~CWatchWindow();
|
||||||
|
|
||||||
|
void NotifyUpdate();
|
||||||
|
void Event_SaveAll(wxCommandEvent& WXUNUSED(event));
|
||||||
|
void SaveAll();
|
||||||
|
void Event_LoadAll(wxCommandEvent& WXUNUSED(event));
|
||||||
|
void LoadAll();
|
||||||
|
|
||||||
|
private:
|
||||||
|
DECLARE_EVENT_TABLE();
|
||||||
|
|
||||||
|
wxAuiManager m_mgr;
|
||||||
|
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
ID_GPR = 1002
|
||||||
|
};
|
||||||
|
|
||||||
|
CWatchView* m_GPRGridView;
|
||||||
|
void CreateGUIControls();
|
||||||
|
};
|
|
@ -71,6 +71,8 @@
|
||||||
<ClCompile Include="Debugger\MemoryWindow.cpp" />
|
<ClCompile Include="Debugger\MemoryWindow.cpp" />
|
||||||
<ClCompile Include="Debugger\RegisterView.cpp" />
|
<ClCompile Include="Debugger\RegisterView.cpp" />
|
||||||
<ClCompile Include="Debugger\RegisterWindow.cpp" />
|
<ClCompile Include="Debugger\RegisterWindow.cpp" />
|
||||||
|
<ClCompile Include="Debugger\WatchView.cpp" />
|
||||||
|
<ClCompile Include="Debugger\WatchWindow.cpp" />
|
||||||
<ClCompile Include="FifoPlayerDlg.cpp" />
|
<ClCompile Include="FifoPlayerDlg.cpp" />
|
||||||
<ClCompile Include="Frame.cpp" />
|
<ClCompile Include="Frame.cpp" />
|
||||||
<ClCompile Include="FrameAui.cpp" />
|
<ClCompile Include="FrameAui.cpp" />
|
||||||
|
@ -124,6 +126,8 @@
|
||||||
<ClInclude Include="Debugger\MemoryWindow.h" />
|
<ClInclude Include="Debugger\MemoryWindow.h" />
|
||||||
<ClInclude Include="Debugger\RegisterView.h" />
|
<ClInclude Include="Debugger\RegisterView.h" />
|
||||||
<ClInclude Include="Debugger\RegisterWindow.h" />
|
<ClInclude Include="Debugger\RegisterWindow.h" />
|
||||||
|
<ClInclude Include="Debugger\WatchView.h" />
|
||||||
|
<ClInclude Include="Debugger\WatchWindow.h" />
|
||||||
<ClInclude Include="FifoPlayerDlg.h" />
|
<ClInclude Include="FifoPlayerDlg.h" />
|
||||||
<ClInclude Include="Frame.h" />
|
<ClInclude Include="Frame.h" />
|
||||||
<ClInclude Include="GameListCtrl.h" />
|
<ClInclude Include="GameListCtrl.h" />
|
||||||
|
|
|
@ -90,6 +90,12 @@
|
||||||
<ClCompile Include="Debugger\RegisterWindow.cpp">
|
<ClCompile Include="Debugger\RegisterWindow.cpp">
|
||||||
<Filter>GUI\Debugger</Filter>
|
<Filter>GUI\Debugger</Filter>
|
||||||
</ClCompile>
|
</ClCompile>
|
||||||
|
<ClCompile Include="Debugger\WatchView.cpp">
|
||||||
|
<Filter>GUI\Debugger</Filter>
|
||||||
|
</ClCompile>
|
||||||
|
<ClCompile Include="Debugger\WatchWindow.cpp">
|
||||||
|
<Filter>GUI\Debugger</Filter>
|
||||||
|
</ClCompile>
|
||||||
<ClCompile Include="InputConfigDiag.cpp">
|
<ClCompile Include="InputConfigDiag.cpp">
|
||||||
<Filter>GUI\InputConfig</Filter>
|
<Filter>GUI\InputConfig</Filter>
|
||||||
</ClCompile>
|
</ClCompile>
|
||||||
|
@ -222,6 +228,12 @@
|
||||||
<ClInclude Include="Debugger\RegisterWindow.h">
|
<ClInclude Include="Debugger\RegisterWindow.h">
|
||||||
<Filter>GUI\Debugger</Filter>
|
<Filter>GUI\Debugger</Filter>
|
||||||
</ClInclude>
|
</ClInclude>
|
||||||
|
<ClInclude Include="Debugger\WatchView.h">
|
||||||
|
<Filter>GUI\Debugger</Filter>
|
||||||
|
</ClInclude>
|
||||||
|
<ClInclude Include="Debugger\WatchWindow.h">
|
||||||
|
<Filter>GUI\Debugger</Filter>
|
||||||
|
</ClInclude>
|
||||||
<ClInclude Include="InputConfigDiag.h">
|
<ClInclude Include="InputConfigDiag.h">
|
||||||
<Filter>GUI\InputConfig</Filter>
|
<Filter>GUI\InputConfig</Filter>
|
||||||
</ClInclude>
|
</ClInclude>
|
||||||
|
|
|
@ -179,6 +179,9 @@ void CFrame::OnToggleWindow(wxCommandEvent& event)
|
||||||
case IDM_REGISTERWINDOW:
|
case IDM_REGISTERWINDOW:
|
||||||
g_pCodeWindow->ToggleRegisterWindow(bShow);
|
g_pCodeWindow->ToggleRegisterWindow(bShow);
|
||||||
break;
|
break;
|
||||||
|
case IDM_WATCHWINDOW:
|
||||||
|
g_pCodeWindow->ToggleWatchWindow(bShow);
|
||||||
|
break;
|
||||||
case IDM_BREAKPOINTWINDOW:
|
case IDM_BREAKPOINTWINDOW:
|
||||||
g_pCodeWindow->ToggleBreakPointWindow(bShow);
|
g_pCodeWindow->ToggleBreakPointWindow(bShow);
|
||||||
break;
|
break;
|
||||||
|
@ -208,6 +211,7 @@ void CFrame::ClosePages()
|
||||||
{
|
{
|
||||||
g_pCodeWindow->ToggleCodeWindow(false);
|
g_pCodeWindow->ToggleCodeWindow(false);
|
||||||
g_pCodeWindow->ToggleRegisterWindow(false);
|
g_pCodeWindow->ToggleRegisterWindow(false);
|
||||||
|
g_pCodeWindow->ToggleWatchWindow(false);
|
||||||
g_pCodeWindow->ToggleBreakPointWindow(false);
|
g_pCodeWindow->ToggleBreakPointWindow(false);
|
||||||
g_pCodeWindow->ToggleMemoryWindow(false);
|
g_pCodeWindow->ToggleMemoryWindow(false);
|
||||||
g_pCodeWindow->ToggleJitWindow(false);
|
g_pCodeWindow->ToggleJitWindow(false);
|
||||||
|
@ -247,6 +251,8 @@ void CFrame::OnNotebookPageClose(wxAuiNotebookEvent& event)
|
||||||
ToggleLogConfigWindow(false);
|
ToggleLogConfigWindow(false);
|
||||||
if (Ctrl->GetPage(event.GetSelection())->GetId() == IDM_REGISTERWINDOW)
|
if (Ctrl->GetPage(event.GetSelection())->GetId() == IDM_REGISTERWINDOW)
|
||||||
g_pCodeWindow->ToggleRegisterWindow(false);
|
g_pCodeWindow->ToggleRegisterWindow(false);
|
||||||
|
if (Ctrl->GetPage(event.GetSelection())->GetId() == IDM_WATCHWINDOW)
|
||||||
|
g_pCodeWindow->ToggleWatchWindow(false);
|
||||||
if (Ctrl->GetPage(event.GetSelection())->GetId() == IDM_BREAKPOINTWINDOW)
|
if (Ctrl->GetPage(event.GetSelection())->GetId() == IDM_BREAKPOINTWINDOW)
|
||||||
g_pCodeWindow->ToggleBreakPointWindow(false);
|
g_pCodeWindow->ToggleBreakPointWindow(false);
|
||||||
if (Ctrl->GetPage(event.GetSelection())->GetId() == IDM_JITWINDOW)
|
if (Ctrl->GetPage(event.GetSelection())->GetId() == IDM_JITWINDOW)
|
||||||
|
|
|
@ -58,6 +58,7 @@
|
||||||
#include "Core/IPC_HLE/WII_IPC_HLE_Device_usb.h"
|
#include "Core/IPC_HLE/WII_IPC_HLE_Device_usb.h"
|
||||||
#include "Core/IPC_HLE/WII_IPC_HLE_WiiMote.h"
|
#include "Core/IPC_HLE/WII_IPC_HLE_WiiMote.h"
|
||||||
#include "Core/PowerPC/PowerPC.h"
|
#include "Core/PowerPC/PowerPC.h"
|
||||||
|
#include "Core/PowerPC/PPCSymbolDB.h"
|
||||||
|
|
||||||
#include "DiscIO/NANDContentLoader.h"
|
#include "DiscIO/NANDContentLoader.h"
|
||||||
|
|
||||||
|
@ -78,7 +79,9 @@
|
||||||
#include "DolphinWX/WXInputBase.h"
|
#include "DolphinWX/WXInputBase.h"
|
||||||
#include "DolphinWX/WxUtils.h"
|
#include "DolphinWX/WxUtils.h"
|
||||||
#include "DolphinWX/Cheats/CheatsWindow.h"
|
#include "DolphinWX/Cheats/CheatsWindow.h"
|
||||||
|
#include "DolphinWX/Debugger/BreakpointWindow.h"
|
||||||
#include "DolphinWX/Debugger/CodeWindow.h"
|
#include "DolphinWX/Debugger/CodeWindow.h"
|
||||||
|
#include "DolphinWX/Debugger/WatchWindow.h"
|
||||||
|
|
||||||
#include "InputCommon/ControllerInterface/ControllerInterface.h"
|
#include "InputCommon/ControllerInterface/ControllerInterface.h"
|
||||||
|
|
||||||
|
@ -285,6 +288,7 @@ wxMenuBar* CFrame::CreateMenu()
|
||||||
|
|
||||||
const wxString MenuText[] = {
|
const wxString MenuText[] = {
|
||||||
wxTRANSLATE("&Registers"),
|
wxTRANSLATE("&Registers"),
|
||||||
|
wxTRANSLATE("&Watch"),
|
||||||
wxTRANSLATE("&Breakpoints"),
|
wxTRANSLATE("&Breakpoints"),
|
||||||
wxTRANSLATE("&Memory"),
|
wxTRANSLATE("&Memory"),
|
||||||
wxTRANSLATE("&JIT"),
|
wxTRANSLATE("&JIT"),
|
||||||
|
@ -655,7 +659,16 @@ void CFrame::BootGame(const std::string& filename)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!bootfile.empty())
|
if (!bootfile.empty())
|
||||||
|
{
|
||||||
StartGame(bootfile);
|
StartGame(bootfile);
|
||||||
|
if (UseDebugger && g_pCodeWindow)
|
||||||
|
{
|
||||||
|
if (g_pCodeWindow->m_WatchWindow)
|
||||||
|
g_pCodeWindow->m_WatchWindow->LoadAll();
|
||||||
|
if (g_pCodeWindow->m_BreakpointWindow)
|
||||||
|
g_pCodeWindow->m_BreakpointWindow->LoadAll();
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Open file to boot
|
// Open file to boot
|
||||||
|
@ -1162,6 +1175,24 @@ void CFrame::DoStop()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (UseDebugger && g_pCodeWindow)
|
||||||
|
{
|
||||||
|
if (g_pCodeWindow->m_WatchWindow)
|
||||||
|
{
|
||||||
|
g_pCodeWindow->m_WatchWindow->SaveAll();
|
||||||
|
PowerPC::watches.Clear();
|
||||||
|
}
|
||||||
|
if (g_pCodeWindow->m_BreakpointWindow)
|
||||||
|
{
|
||||||
|
g_pCodeWindow->m_BreakpointWindow->SaveAll();
|
||||||
|
PowerPC::breakpoints.Clear();
|
||||||
|
PowerPC::memchecks.Clear();
|
||||||
|
g_pCodeWindow->m_BreakpointWindow->NotifyUpdate();
|
||||||
|
}
|
||||||
|
g_symbolDB.Clear();
|
||||||
|
Host_NotifyMapLoaded();
|
||||||
|
}
|
||||||
|
|
||||||
// TODO: Show the author/description dialog here
|
// TODO: Show the author/description dialog here
|
||||||
if (Movie::IsRecordingInput())
|
if (Movie::IsRecordingInput())
|
||||||
DoRecordingSave();
|
DoRecordingSave();
|
||||||
|
|
|
@ -14,6 +14,7 @@ enum
|
||||||
{
|
{
|
||||||
Toolbar_Step,
|
Toolbar_Step,
|
||||||
Toolbar_StepOver,
|
Toolbar_StepOver,
|
||||||
|
Toolbar_StepOut,
|
||||||
Toolbar_Skip,
|
Toolbar_Skip,
|
||||||
Toolbar_GotoPC,
|
Toolbar_GotoPC,
|
||||||
Toolbar_SetPC,
|
Toolbar_SetPC,
|
||||||
|
@ -149,6 +150,7 @@ enum
|
||||||
IDM_LOGWINDOW,
|
IDM_LOGWINDOW,
|
||||||
IDM_LOGCONFIGWINDOW,
|
IDM_LOGCONFIGWINDOW,
|
||||||
IDM_REGISTERWINDOW,
|
IDM_REGISTERWINDOW,
|
||||||
|
IDM_WATCHWINDOW,
|
||||||
IDM_BREAKPOINTWINDOW,
|
IDM_BREAKPOINTWINDOW,
|
||||||
IDM_MEMORYWINDOW,
|
IDM_MEMORYWINDOW,
|
||||||
IDM_JITWINDOW,
|
IDM_JITWINDOW,
|
||||||
|
@ -230,6 +232,7 @@ enum
|
||||||
ID_TOOLBAR_DEBUG,
|
ID_TOOLBAR_DEBUG,
|
||||||
IDM_STEP,
|
IDM_STEP,
|
||||||
IDM_STEPOVER,
|
IDM_STEPOVER,
|
||||||
|
IDM_STEPOUT,
|
||||||
IDM_TOGGLE_BREAKPOINT,
|
IDM_TOGGLE_BREAKPOINT,
|
||||||
IDM_SKIP,
|
IDM_SKIP,
|
||||||
IDM_SETPC,
|
IDM_SETPC,
|
||||||
|
|
Loading…
Reference in New Issue