#include "ThreadKDP.h"
#include "lldb/Host/SafeMachO.h"
#include "lldb/Breakpoint/Watchpoint.h"
#include "lldb/Target/Process.h"
#include "lldb/Target/RegisterContext.h"
#include "lldb/Target/StopInfo.h"
#include "lldb/Target/Target.h"
#include "lldb/Target/Unwind.h"
#include "lldb/Utility/ArchSpec.h"
#include "lldb/Utility/DataExtractor.h"
#include "lldb/Utility/State.h"
#include "lldb/Utility/StreamString.h"
#include "Plugins/Process/Utility/StopInfoMachException.h"
#include "ProcessKDP.h"
#include "ProcessKDPLog.h"
#include "RegisterContextKDP_arm.h"
#include "RegisterContextKDP_arm64.h"
#include "RegisterContextKDP_i386.h"
#include "RegisterContextKDP_x86_64.h"
#include <memory>
using namespace lldb;
using namespace lldb_private;
ThreadKDP::ThreadKDP(Process &process, lldb::tid_t tid)
: Thread(process, tid), m_thread_name(), m_dispatch_queue_name(),
m_thread_dispatch_qaddr(LLDB_INVALID_ADDRESS) {
Log *log = GetLog(KDPLog::Thread);
LLDB_LOG(log, "this = {0}, tid = {1:x}", this, GetID());
}
ThreadKDP::~ThreadKDP() {
Log *log = GetLog(KDPLog::Thread);
LLDB_LOG(log, "this = {0}, tid = {1:x}", this, GetID());
DestroyThread();
}
const char *ThreadKDP::GetName() {
if (m_thread_name.empty())
return nullptr;
return m_thread_name.c_str();
}
const char *ThreadKDP::GetQueueName() { return nullptr; }
void ThreadKDP::RefreshStateAfterStop() {
const bool force = false;
lldb::RegisterContextSP reg_ctx_sp(GetRegisterContext());
if (reg_ctx_sp)
reg_ctx_sp->InvalidateIfNeeded(force);
}
bool ThreadKDP::ThreadIDIsValid(lldb::tid_t thread) { return thread != 0; }
void ThreadKDP::Dump(Log *log, uint32_t index) {}
bool ThreadKDP::ShouldStop(bool &step_more) { return true; }
lldb::RegisterContextSP ThreadKDP::GetRegisterContext() {
if (!m_reg_context_sp)
m_reg_context_sp = CreateRegisterContextForFrame(nullptr);
return m_reg_context_sp;
}
lldb::RegisterContextSP
ThreadKDP::CreateRegisterContextForFrame(StackFrame *frame) {
lldb::RegisterContextSP reg_ctx_sp;
uint32_t concrete_frame_idx = 0;
if (frame)
concrete_frame_idx = frame->GetConcreteFrameIndex();
if (concrete_frame_idx == 0) {
ProcessSP process_sp(CalculateProcess());
if (process_sp) {
switch (static_cast<ProcessKDP *>(process_sp.get())
->GetCommunication()
.GetCPUType()) {
case llvm::MachO::CPU_TYPE_ARM:
reg_ctx_sp =
std::make_shared<RegisterContextKDP_arm>(*this, concrete_frame_idx);
break;
case llvm::MachO::CPU_TYPE_ARM64:
reg_ctx_sp = std::make_shared<RegisterContextKDP_arm64>(
*this, concrete_frame_idx);
break;
case llvm::MachO::CPU_TYPE_I386:
reg_ctx_sp = std::make_shared<RegisterContextKDP_i386>(
*this, concrete_frame_idx);
break;
case llvm::MachO::CPU_TYPE_X86_64:
reg_ctx_sp = std::make_shared<RegisterContextKDP_x86_64>(
*this, concrete_frame_idx);
break;
default:
llvm_unreachable("Add CPU type support in KDP");
}
}
} else {
reg_ctx_sp = GetUnwinder().CreateRegisterContextForFrame(frame);
}
return reg_ctx_sp;
}
bool ThreadKDP::CalculateStopInfo() {
ProcessSP process_sp(GetProcess());
if (process_sp) {
if (m_cached_stop_info_sp) {
SetStopInfo(m_cached_stop_info_sp);
} else {
SetStopInfo(StopInfo::CreateStopReasonWithSignal(*this, SIGSTOP));
}
return true;
}
return false;
}
void ThreadKDP::SetStopInfoFrom_KDP_EXCEPTION(
const DataExtractor &exc_reply_packet) {
lldb::offset_t offset = 0;
uint8_t reply_command = exc_reply_packet.GetU8(&offset);
if (reply_command == CommunicationKDP::KDP_EXCEPTION) {
offset = 8;
const uint32_t count = exc_reply_packet.GetU32(&offset);
if (count >= 1) {
offset += 4;
const uint32_t exc_type = exc_reply_packet.GetU32(&offset);
const uint32_t exc_code = exc_reply_packet.GetU32(&offset);
const uint32_t exc_subcode = exc_reply_packet.GetU32(&offset);
const bool pc_already_adjusted = false;
const bool adjust_pc_if_needed = true;
m_cached_stop_info_sp =
StopInfoMachException::CreateStopReasonWithMachException(
*this, exc_type, 2, exc_code, exc_subcode, 0, pc_already_adjusted,
adjust_pc_if_needed);
}
}
}