Switching debugger to not retain XThreads.
This commit is contained in:
parent
1b487b67c9
commit
07ba1be7f5
|
@ -183,10 +183,9 @@ std::vector<ThreadExecutionInfo*> Debugger::QueryThreadExecutionInfos() {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
ThreadExecutionInfo* Debugger::QueryThreadExecutionInfo(
|
ThreadExecutionInfo* Debugger::QueryThreadExecutionInfo(uint32_t thread_id) {
|
||||||
uint32_t thread_handle) {
|
|
||||||
auto global_lock = global_critical_region_.Acquire();
|
auto global_lock = global_critical_region_.Acquire();
|
||||||
const auto& it = thread_execution_infos_.find(thread_handle);
|
const auto& it = thread_execution_infos_.find(thread_id);
|
||||||
if (it == thread_execution_infos_.end()) {
|
if (it == thread_execution_infos_.end()) {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
@ -232,8 +231,7 @@ bool Debugger::SuspendAllThreads() {
|
||||||
// Thread is a host thread, and we aren't suspending those (for now).
|
// Thread is a host thread, and we aren't suspending those (for now).
|
||||||
continue;
|
continue;
|
||||||
} else if (XThread::IsInThread() &&
|
} else if (XThread::IsInThread() &&
|
||||||
thread_info->thread_handle ==
|
thread_info->thread_id == XThread::GetCurrentThreadId()) {
|
||||||
XThread::GetCurrentThreadHandle()) {
|
|
||||||
// Can't suspend ourselves.
|
// Can't suspend ourselves.
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
@ -244,9 +242,9 @@ bool Debugger::SuspendAllThreads() {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Debugger::ResumeThread(uint32_t thread_handle) {
|
bool Debugger::ResumeThread(uint32_t thread_id) {
|
||||||
auto global_lock = global_critical_region_.Acquire();
|
auto global_lock = global_critical_region_.Acquire();
|
||||||
auto it = thread_execution_infos_.find(thread_handle);
|
auto it = thread_execution_infos_.find(thread_id);
|
||||||
if (it == thread_execution_infos_.end()) {
|
if (it == thread_execution_infos_.end()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -273,8 +271,7 @@ bool Debugger::ResumeAllThreads() {
|
||||||
// Thread is a host thread, and we aren't suspending those (for now).
|
// Thread is a host thread, and we aren't suspending those (for now).
|
||||||
continue;
|
continue;
|
||||||
} else if (XThread::IsInThread() &&
|
} else if (XThread::IsInThread() &&
|
||||||
thread_info->thread_handle ==
|
thread_info->thread_id == XThread::GetCurrentThreadId()) {
|
||||||
XThread::GetCurrentThreadHandle()) {
|
|
||||||
// Can't resume ourselves.
|
// Can't resume ourselves.
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
@ -285,7 +282,7 @@ bool Debugger::ResumeAllThreads() {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Debugger::UpdateThreadExecutionStates(uint32_t override_handle,
|
void Debugger::UpdateThreadExecutionStates(uint32_t override_thread_id,
|
||||||
X64Context* override_context) {
|
X64Context* override_context) {
|
||||||
auto global_lock = global_critical_region_.Acquire();
|
auto global_lock = global_critical_region_.Acquire();
|
||||||
auto stack_walker = emulator_->processor()->stack_walker();
|
auto stack_walker = emulator_->processor()->stack_walker();
|
||||||
|
@ -310,7 +307,7 @@ void Debugger::UpdateThreadExecutionStates(uint32_t override_handle,
|
||||||
// Grab stack trace and X64 context then resolve all symbols.
|
// Grab stack trace and X64 context then resolve all symbols.
|
||||||
uint64_t hash;
|
uint64_t hash;
|
||||||
X64Context* in_host_context = nullptr;
|
X64Context* in_host_context = nullptr;
|
||||||
if (override_handle == thread_info->thread_handle) {
|
if (override_thread_id == thread_info->thread_id) {
|
||||||
// If we were passed an override context we use that. Otherwise, ask the
|
// If we were passed an override context we use that. Otherwise, ask the
|
||||||
// stack walker for a new context.
|
// stack walker for a new context.
|
||||||
in_host_context = override_context;
|
in_host_context = override_context;
|
||||||
|
@ -392,12 +389,12 @@ void Debugger::Continue() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Debugger::StepGuestInstruction(uint32_t thread_handle) {
|
void Debugger::StepGuestInstruction(uint32_t thread_id) {
|
||||||
auto global_lock = global_critical_region_.Acquire();
|
auto global_lock = global_critical_region_.Acquire();
|
||||||
assert_true(execution_state_ == ExecutionState::kPaused);
|
assert_true(execution_state_ == ExecutionState::kPaused);
|
||||||
execution_state_ = ExecutionState::kStepping;
|
execution_state_ = ExecutionState::kStepping;
|
||||||
|
|
||||||
auto thread_info = thread_execution_infos_[thread_handle].get();
|
auto thread_info = thread_execution_infos_[thread_id].get();
|
||||||
|
|
||||||
uint32_t next_pc = CalculateNextGuestInstruction(
|
uint32_t next_pc = CalculateNextGuestInstruction(
|
||||||
thread_info, thread_info->frames[0].guest_pc);
|
thread_info, thread_info->frames[0].guest_pc);
|
||||||
|
@ -409,15 +406,15 @@ void Debugger::StepGuestInstruction(uint32_t thread_handle) {
|
||||||
thread_info->step_breakpoint->Resume();
|
thread_info->step_breakpoint->Resume();
|
||||||
|
|
||||||
// ResumeAllBreakpoints();
|
// ResumeAllBreakpoints();
|
||||||
ResumeThread(thread_handle);
|
ResumeThread(thread_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Debugger::StepHostInstruction(uint32_t thread_handle) {
|
void Debugger::StepHostInstruction(uint32_t thread_id) {
|
||||||
auto global_lock = global_critical_region_.Acquire();
|
auto global_lock = global_critical_region_.Acquire();
|
||||||
assert_true(execution_state_ == ExecutionState::kPaused);
|
assert_true(execution_state_ == ExecutionState::kPaused);
|
||||||
execution_state_ = ExecutionState::kStepping;
|
execution_state_ = ExecutionState::kStepping;
|
||||||
|
|
||||||
auto thread_info = thread_execution_infos_[thread_handle].get();
|
auto thread_info = thread_execution_infos_[thread_id].get();
|
||||||
uint64_t new_host_pc =
|
uint64_t new_host_pc =
|
||||||
CalculateNextHostInstruction(thread_info, thread_info->frames[0].host_pc);
|
CalculateNextHostInstruction(thread_info, thread_info->frames[0].host_pc);
|
||||||
|
|
||||||
|
@ -428,23 +425,24 @@ void Debugger::StepHostInstruction(uint32_t thread_handle) {
|
||||||
thread_info->step_breakpoint->Resume();
|
thread_info->step_breakpoint->Resume();
|
||||||
|
|
||||||
// ResumeAllBreakpoints();
|
// ResumeAllBreakpoints();
|
||||||
ResumeThread(thread_handle);
|
ResumeThread(thread_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Debugger::OnThreadCreated(XThread* thread) {
|
void Debugger::OnThreadCreated(XThread* thread) {
|
||||||
auto global_lock = global_critical_region_.Acquire();
|
auto global_lock = global_critical_region_.Acquire();
|
||||||
auto thread_info = std::make_unique<ThreadExecutionInfo>();
|
auto thread_info = std::make_unique<ThreadExecutionInfo>();
|
||||||
thread_info->thread_handle = thread->handle();
|
thread_info->thread_handle = thread->handle();
|
||||||
|
thread_info->thread_id = thread->thread_id();
|
||||||
thread_info->thread = thread;
|
thread_info->thread = thread;
|
||||||
thread_info->state = ThreadExecutionInfo::State::kAlive;
|
thread_info->state = ThreadExecutionInfo::State::kAlive;
|
||||||
thread_info->suspended = false;
|
thread_info->suspended = false;
|
||||||
thread_execution_infos_.emplace(thread_info->thread_handle,
|
thread_execution_infos_.emplace(thread_info->thread_id,
|
||||||
std::move(thread_info));
|
std::move(thread_info));
|
||||||
}
|
}
|
||||||
|
|
||||||
void Debugger::OnThreadExit(XThread* thread) {
|
void Debugger::OnThreadExit(XThread* thread) {
|
||||||
auto global_lock = global_critical_region_.Acquire();
|
auto global_lock = global_critical_region_.Acquire();
|
||||||
auto it = thread_execution_infos_.find(thread->handle());
|
auto it = thread_execution_infos_.find(thread->thread_id());
|
||||||
assert_true(it != thread_execution_infos_.end());
|
assert_true(it != thread_execution_infos_.end());
|
||||||
auto thread_info = it->second.get();
|
auto thread_info = it->second.get();
|
||||||
thread_info->state = ThreadExecutionInfo::State::kExited;
|
thread_info->state = ThreadExecutionInfo::State::kExited;
|
||||||
|
@ -452,7 +450,7 @@ void Debugger::OnThreadExit(XThread* thread) {
|
||||||
|
|
||||||
void Debugger::OnThreadDestroyed(XThread* thread) {
|
void Debugger::OnThreadDestroyed(XThread* thread) {
|
||||||
auto global_lock = global_critical_region_.Acquire();
|
auto global_lock = global_critical_region_.Acquire();
|
||||||
auto it = thread_execution_infos_.find(thread->handle());
|
auto it = thread_execution_infos_.find(thread->thread_id());
|
||||||
assert_true(it != thread_execution_infos_.end());
|
assert_true(it != thread_execution_infos_.end());
|
||||||
auto thread_info = it->second.get();
|
auto thread_info = it->second.get();
|
||||||
// TODO(benvanik): retain the thread?
|
// TODO(benvanik): retain the thread?
|
||||||
|
@ -462,7 +460,7 @@ void Debugger::OnThreadDestroyed(XThread* thread) {
|
||||||
|
|
||||||
void Debugger::OnThreadEnteringWait(XThread* thread) {
|
void Debugger::OnThreadEnteringWait(XThread* thread) {
|
||||||
auto global_lock = global_critical_region_.Acquire();
|
auto global_lock = global_critical_region_.Acquire();
|
||||||
auto it = thread_execution_infos_.find(thread->handle());
|
auto it = thread_execution_infos_.find(thread->thread_id());
|
||||||
assert_true(it != thread_execution_infos_.end());
|
assert_true(it != thread_execution_infos_.end());
|
||||||
auto thread_info = it->second.get();
|
auto thread_info = it->second.get();
|
||||||
thread_info->state = ThreadExecutionInfo::State::kWaiting;
|
thread_info->state = ThreadExecutionInfo::State::kWaiting;
|
||||||
|
@ -470,7 +468,7 @@ void Debugger::OnThreadEnteringWait(XThread* thread) {
|
||||||
|
|
||||||
void Debugger::OnThreadLeavingWait(XThread* thread) {
|
void Debugger::OnThreadLeavingWait(XThread* thread) {
|
||||||
auto global_lock = global_critical_region_.Acquire();
|
auto global_lock = global_critical_region_.Acquire();
|
||||||
auto it = thread_execution_infos_.find(thread->handle());
|
auto it = thread_execution_infos_.find(thread->thread_id());
|
||||||
assert_true(it != thread_execution_infos_.end());
|
assert_true(it != thread_execution_infos_.end());
|
||||||
auto thread_info = it->second.get();
|
auto thread_info = it->second.get();
|
||||||
if (thread_info->state == ThreadExecutionInfo::State::kWaiting) {
|
if (thread_info->state == ThreadExecutionInfo::State::kWaiting) {
|
||||||
|
@ -531,7 +529,7 @@ bool Debugger::ExceptionCallback(Exception* ex) {
|
||||||
SuspendAllThreads();
|
SuspendAllThreads();
|
||||||
|
|
||||||
// Lookup thread info block.
|
// Lookup thread info block.
|
||||||
auto it = thread_execution_infos_.find(XThread::GetCurrentThreadHandle());
|
auto it = thread_execution_infos_.find(XThread::GetCurrentThreadId());
|
||||||
if (it == thread_execution_infos_.end()) {
|
if (it == thread_execution_infos_.end()) {
|
||||||
// Not found - exception on a thread we don't know about?
|
// Not found - exception on a thread we don't know about?
|
||||||
assert_always("UD2 on a thread we don't track");
|
assert_always("UD2 on a thread we don't track");
|
||||||
|
@ -548,7 +546,7 @@ bool Debugger::ExceptionCallback(Exception* ex) {
|
||||||
// Update all thread states with their latest values, using the context we got
|
// Update all thread states with their latest values, using the context we got
|
||||||
// from the exception instead of a sampled value (as it would just show the
|
// from the exception instead of a sampled value (as it would just show the
|
||||||
// exception handler).
|
// exception handler).
|
||||||
UpdateThreadExecutionStates(thread_info->thread_handle, ex->thread_context());
|
UpdateThreadExecutionStates(thread_info->thread_id, ex->thread_context());
|
||||||
|
|
||||||
// Whether we should block waiting for a continue event.
|
// Whether we should block waiting for a continue event.
|
||||||
bool wait_for_continue = false;
|
bool wait_for_continue = false;
|
||||||
|
@ -673,7 +671,7 @@ bool Debugger::OnUnhandledException(Exception* ex) {
|
||||||
// Suspend all guest threads (but this one).
|
// Suspend all guest threads (but this one).
|
||||||
SuspendAllThreads();
|
SuspendAllThreads();
|
||||||
|
|
||||||
UpdateThreadExecutionStates(XThread::GetCurrentThreadHandle(),
|
UpdateThreadExecutionStates(XThread::GetCurrentThreadId(),
|
||||||
ex->thread_context());
|
ex->thread_context());
|
||||||
|
|
||||||
// Stop and notify the listener.
|
// Stop and notify the listener.
|
||||||
|
|
|
@ -74,7 +74,10 @@ struct ThreadExecutionInfo {
|
||||||
kZombie,
|
kZombie,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// XThread::thread_id(), unique to the thread for the run of the emulator.
|
||||||
|
uint32_t thread_id = 0;
|
||||||
// XThread::handle() of the thread.
|
// XThread::handle() of the thread.
|
||||||
|
// This will be invalidated when the thread dies.
|
||||||
uint32_t thread_handle = 0;
|
uint32_t thread_handle = 0;
|
||||||
// Target XThread, if it has not been destroyed.
|
// Target XThread, if it has not been destroyed.
|
||||||
// TODO(benvanik): hold a ref here to keep zombie threads around?
|
// TODO(benvanik): hold a ref here to keep zombie threads around?
|
||||||
|
@ -203,7 +206,7 @@ class Debugger {
|
||||||
// the kernel.
|
// the kernel.
|
||||||
std::vector<ThreadExecutionInfo*> QueryThreadExecutionInfos();
|
std::vector<ThreadExecutionInfo*> QueryThreadExecutionInfos();
|
||||||
// Returns the debugger info for the given thread.
|
// Returns the debugger info for the given thread.
|
||||||
ThreadExecutionInfo* QueryThreadExecutionInfo(uint32_t thread_handle);
|
ThreadExecutionInfo* QueryThreadExecutionInfo(uint32_t thread_id);
|
||||||
|
|
||||||
// Adds a breakpoint to the debugger and activates it (if enabled).
|
// Adds a breakpoint to the debugger and activates it (if enabled).
|
||||||
// The given breakpoint will not be owned by the debugger and must remain
|
// The given breakpoint will not be owned by the debugger and must remain
|
||||||
|
@ -230,10 +233,10 @@ class Debugger {
|
||||||
|
|
||||||
// Steps the given thread a single PPC guest instruction.
|
// Steps the given thread a single PPC guest instruction.
|
||||||
// If the step is over a branch the branch will be followed.
|
// If the step is over a branch the branch will be followed.
|
||||||
void StepGuestInstruction(uint32_t thread_handle);
|
void StepGuestInstruction(uint32_t thread_id);
|
||||||
// Steps the given thread a single x64 host instruction.
|
// Steps the given thread a single x64 host instruction.
|
||||||
// If the step is over a branch the branch will be followed.
|
// If the step is over a branch the branch will be followed.
|
||||||
void StepHostInstruction(uint32_t thread_handle);
|
void StepHostInstruction(uint32_t thread_id);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
// TODO(benvanik): hide.
|
// TODO(benvanik): hide.
|
||||||
|
@ -254,7 +257,7 @@ class Debugger {
|
||||||
// Suspends all known threads (except the caller).
|
// Suspends all known threads (except the caller).
|
||||||
bool SuspendAllThreads();
|
bool SuspendAllThreads();
|
||||||
// Resumes the given thread.
|
// Resumes the given thread.
|
||||||
bool ResumeThread(uint32_t thread_handle);
|
bool ResumeThread(uint32_t thread_id);
|
||||||
// Resumes all known threads (except the caller).
|
// Resumes all known threads (except the caller).
|
||||||
bool ResumeAllThreads();
|
bool ResumeAllThreads();
|
||||||
// Updates all cached thread execution info (state, call stacks, etc).
|
// Updates all cached thread execution info (state, call stacks, etc).
|
||||||
|
|
|
@ -293,8 +293,8 @@ void DebugWindow::DrawToolbar() {
|
||||||
if (ImGui::Combo("##thread_combo", ¤t_thread_index,
|
if (ImGui::Combo("##thread_combo", ¤t_thread_index,
|
||||||
thread_combo.GetString(), 10)) {
|
thread_combo.GetString(), 10)) {
|
||||||
// Thread changed.
|
// Thread changed.
|
||||||
SelectThreadStackFrame(
|
SelectThreadStackFrame(cache_.thread_execution_infos[current_thread_index],
|
||||||
cache_.thread_execution_infos[current_thread_index]->thread, 0, true);
|
0, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -347,12 +347,12 @@ void DebugWindow::DrawSourcePane() {
|
||||||
|
|
||||||
ImGui::BeginGroup();
|
ImGui::BeginGroup();
|
||||||
ImGui::PushButtonRepeat(true);
|
ImGui::PushButtonRepeat(true);
|
||||||
bool can_step = !cache_.is_running && state_.thread;
|
bool can_step = !cache_.is_running && state_.thread_info;
|
||||||
if (ImGui::ButtonEx("Step PPC", ImVec2(0, 0),
|
if (ImGui::ButtonEx("Step PPC", ImVec2(0, 0),
|
||||||
can_step ? 0 : ImGuiButtonFlags_Disabled)) {
|
can_step ? 0 : ImGuiButtonFlags_Disabled)) {
|
||||||
// By enabling the button when stepping we allow repeat behavior.
|
// By enabling the button when stepping we allow repeat behavior.
|
||||||
if (debugger_->execution_state() != ExecutionState::kStepping) {
|
if (debugger_->execution_state() != ExecutionState::kStepping) {
|
||||||
debugger_->StepGuestInstruction(state_.thread->handle());
|
debugger_->StepGuestInstruction(state_.thread_info->thread_id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ImGui::PopButtonRepeat();
|
ImGui::PopButtonRepeat();
|
||||||
|
@ -370,7 +370,7 @@ void DebugWindow::DrawSourcePane() {
|
||||||
can_step ? 0 : ImGuiButtonFlags_Disabled)) {
|
can_step ? 0 : ImGuiButtonFlags_Disabled)) {
|
||||||
// By enabling the button when stepping we allow repeat behavior.
|
// By enabling the button when stepping we allow repeat behavior.
|
||||||
if (debugger_->execution_state() != ExecutionState::kStepping) {
|
if (debugger_->execution_state() != ExecutionState::kStepping) {
|
||||||
debugger_->StepHostInstruction(state_.thread->handle());
|
debugger_->StepHostInstruction(state_.thread_info->thread_id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ImGui::PopButtonRepeat();
|
ImGui::PopButtonRepeat();
|
||||||
|
@ -993,8 +993,8 @@ void DebugWindow::DrawThreadsPane() {
|
||||||
ImGui::BeginChild("##threads_listing");
|
ImGui::BeginChild("##threads_listing");
|
||||||
for (size_t i = 0; i < cache_.thread_execution_infos.size(); ++i) {
|
for (size_t i = 0; i < cache_.thread_execution_infos.size(); ++i) {
|
||||||
auto thread_info = cache_.thread_execution_infos[i];
|
auto thread_info = cache_.thread_execution_infos[i];
|
||||||
|
bool is_current_thread = thread_info == state_.thread_info;
|
||||||
auto thread = thread_info->thread;
|
auto thread = thread_info->thread;
|
||||||
bool is_current_thread = thread == state_.thread;
|
|
||||||
if (!thread) {
|
if (!thread) {
|
||||||
// TODO(benvanik): better display of zombie thread states.
|
// TODO(benvanik): better display of zombie thread states.
|
||||||
continue;
|
continue;
|
||||||
|
@ -1009,7 +1009,7 @@ void DebugWindow::DrawThreadsPane() {
|
||||||
ImGui::PushStyleColor(ImGuiCol_Header,
|
ImGui::PushStyleColor(ImGuiCol_Header,
|
||||||
ImGui::GetStyle().Colors[ImGuiCol_HeaderActive]);
|
ImGui::GetStyle().Colors[ImGuiCol_HeaderActive]);
|
||||||
}
|
}
|
||||||
ImGui::PushID(thread);
|
ImGui::PushID(thread_info);
|
||||||
if (is_current_thread) {
|
if (is_current_thread) {
|
||||||
ImGui::SetNextTreeNodeOpened(true, ImGuiSetCond_Always);
|
ImGui::SetNextTreeNodeOpened(true, ImGuiSetCond_Always);
|
||||||
}
|
}
|
||||||
|
@ -1049,7 +1049,7 @@ void DebugWindow::DrawThreadsPane() {
|
||||||
frame.host_pc, &frame);
|
frame.host_pc, &frame);
|
||||||
if (ImGui::Selectable(host_label, is_current_frame,
|
if (ImGui::Selectable(host_label, is_current_frame,
|
||||||
ImGuiSelectableFlags_SpanAllColumns)) {
|
ImGuiSelectableFlags_SpanAllColumns)) {
|
||||||
SelectThreadStackFrame(thread, j, true);
|
SelectThreadStackFrame(thread_info, j, true);
|
||||||
}
|
}
|
||||||
ImGui::SameLine();
|
ImGui::SameLine();
|
||||||
ImGui::Dummy(ImVec2(8, 0));
|
ImGui::Dummy(ImVec2(8, 0));
|
||||||
|
@ -1386,21 +1386,13 @@ void DebugWindow::DrawLogPane() {
|
||||||
// if big, click to open dialog with contents
|
// if big, click to open dialog with contents
|
||||||
}
|
}
|
||||||
|
|
||||||
void DebugWindow::SelectThreadStackFrame(XThread* thread,
|
void DebugWindow::SelectThreadStackFrame(ThreadExecutionInfo* thread_info,
|
||||||
size_t stack_frame_index,
|
size_t stack_frame_index,
|
||||||
bool always_navigate) {
|
bool always_navigate) {
|
||||||
state_.has_changed_thread = false;
|
state_.has_changed_thread = false;
|
||||||
if (thread != state_.thread) {
|
if (thread_info != state_.thread_info) {
|
||||||
state_.has_changed_thread = true;
|
state_.has_changed_thread = true;
|
||||||
state_.thread = thread;
|
|
||||||
|
|
||||||
state_.thread_info = nullptr;
|
|
||||||
for (auto thread_info : cache_.thread_execution_infos) {
|
|
||||||
if (thread_info->thread == thread) {
|
|
||||||
state_.thread_info = thread_info;
|
state_.thread_info = thread_info;
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if (state_.thread_info) {
|
if (state_.thread_info) {
|
||||||
stack_frame_index =
|
stack_frame_index =
|
||||||
|
@ -1469,7 +1461,8 @@ void DebugWindow::UpdateCache() {
|
||||||
|
|
||||||
cache_.thread_execution_infos = debugger_->QueryThreadExecutionInfos();
|
cache_.thread_execution_infos = debugger_->QueryThreadExecutionInfos();
|
||||||
|
|
||||||
SelectThreadStackFrame(state_.thread, state_.thread_stack_frame_index, false);
|
SelectThreadStackFrame(state_.thread_info, state_.thread_stack_frame_index,
|
||||||
|
false);
|
||||||
}
|
}
|
||||||
|
|
||||||
void DebugWindow::CreateCodeBreakpoint(CodeBreakpoint::AddressType address_type,
|
void DebugWindow::CreateCodeBreakpoint(CodeBreakpoint::AddressType address_type,
|
||||||
|
@ -1582,14 +1575,16 @@ void DebugWindow::OnExecutionEnded() {
|
||||||
|
|
||||||
void DebugWindow::OnStepCompleted(xe::kernel::XThread* thread) {
|
void DebugWindow::OnStepCompleted(xe::kernel::XThread* thread) {
|
||||||
UpdateCache();
|
UpdateCache();
|
||||||
SelectThreadStackFrame(thread, 0, true);
|
auto thread_info = debugger_->QueryThreadExecutionInfo(thread->thread_id());
|
||||||
|
SelectThreadStackFrame(thread_info, 0, true);
|
||||||
loop_->Post([this]() { window_->set_focus(true); });
|
loop_->Post([this]() { window_->set_focus(true); });
|
||||||
}
|
}
|
||||||
|
|
||||||
void DebugWindow::OnBreakpointHit(Breakpoint* breakpoint,
|
void DebugWindow::OnBreakpointHit(Breakpoint* breakpoint,
|
||||||
xe::kernel::XThread* thread) {
|
xe::kernel::XThread* thread) {
|
||||||
UpdateCache();
|
UpdateCache();
|
||||||
SelectThreadStackFrame(thread, 0, true);
|
auto thread_info = debugger_->QueryThreadExecutionInfo(thread->thread_id());
|
||||||
|
SelectThreadStackFrame(thread_info, 0, true);
|
||||||
loop_->Post([this]() { window_->set_focus(true); });
|
loop_->Post([this]() { window_->set_focus(true); });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -69,8 +69,8 @@ class DebugWindow : public DebugListener {
|
||||||
void DrawBreakpointsPane();
|
void DrawBreakpointsPane();
|
||||||
void DrawLogPane();
|
void DrawLogPane();
|
||||||
|
|
||||||
void SelectThreadStackFrame(kernel::XThread* thread, size_t stack_frame_index,
|
void SelectThreadStackFrame(ThreadExecutionInfo* thread_info,
|
||||||
bool always_navigate);
|
size_t stack_frame_index, bool always_navigate);
|
||||||
void NavigateToFunction(cpu::Function* function, uint32_t guest_pc = 0,
|
void NavigateToFunction(cpu::Function* function, uint32_t guest_pc = 0,
|
||||||
uint64_t host_pc = 0);
|
uint64_t host_pc = 0);
|
||||||
// void NavigateToMemory(uint64_t address, uint64_t length = 0);
|
// void NavigateToMemory(uint64_t address, uint64_t length = 0);
|
||||||
|
@ -117,7 +117,6 @@ class DebugWindow : public DebugListener {
|
||||||
static const int kRightPaneMemory = 1;
|
static const int kRightPaneMemory = 1;
|
||||||
int right_pane_tab = kRightPaneThreads;
|
int right_pane_tab = kRightPaneThreads;
|
||||||
|
|
||||||
xe::kernel::XThread* thread = nullptr;
|
|
||||||
ThreadExecutionInfo* thread_info = nullptr;
|
ThreadExecutionInfo* thread_info = nullptr;
|
||||||
size_t thread_stack_frame_index = 0;
|
size_t thread_stack_frame_index = 0;
|
||||||
bool has_changed_thread = false;
|
bool has_changed_thread = false;
|
||||||
|
|
Loading…
Reference in New Issue