1 //===-- ThreadMinidump.cpp --------------------------------------*- C++ -*-===// 2 // 3 // The LLVM Compiler Infrastructure 4 // 5 // This file is distributed under the University of Illinois Open Source 6 // License. See LICENSE.TXT for details. 7 // 8 //===----------------------------------------------------------------------===// 9 10 // Project includes 11 #include "ThreadMinidump.h" 12 #include "ProcessMinidump.h" 13 14 #include "RegisterContextMinidump_x86_32.h" 15 #include "RegisterContextMinidump_x86_64.h" 16 17 // Other libraries and framework includes 18 #include "Plugins/Process/Utility/RegisterContextLinux_i386.h" 19 #include "Plugins/Process/Utility/RegisterContextLinux_x86_64.h" 20 21 #include "Plugins/Process/elf-core/RegisterContextPOSIXCore_x86_64.h" 22 23 #include "lldb/Target/RegisterContext.h" 24 #include "lldb/Target/StopInfo.h" 25 #include "lldb/Target/Target.h" 26 #include "lldb/Target/Unwind.h" 27 #include "lldb/Utility/DataExtractor.h" 28 #include "lldb/Utility/Log.h" 29 30 // C Includes 31 // C++ Includes 32 33 using namespace lldb; 34 using namespace lldb_private; 35 using namespace minidump; 36 37 ThreadMinidump::ThreadMinidump(Process &process, const MinidumpThread &td, 38 llvm::ArrayRef<uint8_t> gpregset_data) 39 : Thread(process, td.thread_id), m_thread_reg_ctx_sp(), 40 m_gpregset_data(gpregset_data) {} 41 42 ThreadMinidump::~ThreadMinidump() {} 43 44 void ThreadMinidump::RefreshStateAfterStop() {} 45 46 RegisterContextSP ThreadMinidump::GetRegisterContext() { 47 if (!m_reg_context_sp) { 48 m_reg_context_sp = CreateRegisterContextForFrame(nullptr); 49 } 50 return m_reg_context_sp; 51 } 52 53 RegisterContextSP 54 ThreadMinidump::CreateRegisterContextForFrame(StackFrame *frame) { 55 RegisterContextSP reg_ctx_sp; 56 uint32_t concrete_frame_idx = 0; 57 Log *log(GetLogIfAllCategoriesSet(LIBLLDB_LOG_THREAD)); 58 59 if (frame) 60 concrete_frame_idx = frame->GetConcreteFrameIndex(); 61 62 if (concrete_frame_idx == 0) { 63 if (m_thread_reg_ctx_sp) 64 return m_thread_reg_ctx_sp; 65 66 ProcessMinidump *process = 67 static_cast<ProcessMinidump *>(GetProcess().get()); 68 ArchSpec arch = process->GetArchitecture(); 69 RegisterInfoInterface *reg_interface = nullptr; 70 71 // TODO write other register contexts and add them here 72 switch (arch.GetMachine()) { 73 case llvm::Triple::x86: { 74 reg_interface = new RegisterContextLinux_i386(arch); 75 lldb::DataBufferSP buf = 76 ConvertMinidumpContext_x86_32(m_gpregset_data, reg_interface); 77 DataExtractor gpregs(buf, lldb::eByteOrderLittle, 4); 78 DataExtractor fpregs; 79 m_thread_reg_ctx_sp.reset(new RegisterContextCorePOSIX_x86_64( 80 *this, reg_interface, gpregs, fpregs)); 81 break; 82 } 83 case llvm::Triple::x86_64: { 84 reg_interface = new RegisterContextLinux_x86_64(arch); 85 lldb::DataBufferSP buf = 86 ConvertMinidumpContext_x86_64(m_gpregset_data, reg_interface); 87 DataExtractor gpregs(buf, lldb::eByteOrderLittle, 8); 88 DataExtractor fpregs; 89 m_thread_reg_ctx_sp.reset(new RegisterContextCorePOSIX_x86_64( 90 *this, reg_interface, gpregs, fpregs)); 91 break; 92 } 93 default: 94 break; 95 } 96 97 if (!reg_interface) { 98 if (log) 99 log->Printf("elf-core::%s:: Architecture(%d) not supported", 100 __FUNCTION__, arch.GetMachine()); 101 assert(false && "Architecture not supported"); 102 } 103 104 reg_ctx_sp = m_thread_reg_ctx_sp; 105 } else if (m_unwinder_ap) { 106 reg_ctx_sp = m_unwinder_ap->CreateRegisterContextForFrame(frame); 107 } 108 109 return reg_ctx_sp; 110 } 111 112 bool ThreadMinidump::CalculateStopInfo() { return false; } 113