106e827ccSJim Ingham //===-- ThreadPlan.cpp ------------------------------------------*- C++ -*-===// 206e827ccSJim Ingham // 306e827ccSJim Ingham // The LLVM Compiler Infrastructure 406e827ccSJim Ingham // 506e827ccSJim Ingham // This file is distributed under the University of Illinois Open Source 606e827ccSJim Ingham // License. See LICENSE.TXT for details. 706e827ccSJim Ingham // 806e827ccSJim Ingham //===----------------------------------------------------------------------===// 906e827ccSJim Ingham 1093a64300SDaniel Malea #include "lldb/lldb-python.h" 1193a64300SDaniel Malea 1206e827ccSJim Ingham #include "lldb/Target/ThreadPlan.h" 1306e827ccSJim Ingham 1406e827ccSJim Ingham // C Includes 158c9e5383SSean Callanan #include <string.h> 1606e827ccSJim Ingham // C++ Includes 1706e827ccSJim Ingham // Other libraries and framework includes 1806e827ccSJim Ingham // Project includes 198c9e5383SSean Callanan #include "lldb/Core/ArchSpec.h" 208c9e5383SSean Callanan #include "lldb/Core/DataBufferHeap.h" 21*a78bd7ffSZachary Turner #include "lldb/Core/DataExtractor.h" 2206e827ccSJim Ingham #include "lldb/Core/Debugger.h" 238c9e5383SSean Callanan #include "lldb/Core/Disassembler.h" 2406e827ccSJim Ingham #include "lldb/Core/Log.h" 251f746071SGreg Clayton #include "lldb/Core/Module.h" 2606e827ccSJim Ingham #include "lldb/Core/State.h" 2744d93782SGreg Clayton #include "lldb/Core/StreamFile.h" 288c9e5383SSean Callanan #include "lldb/Core/Value.h" 29*a78bd7ffSZachary Turner #include "lldb/Symbol/ClangASTContext.h" 308c9e5383SSean Callanan #include "lldb/Symbol/TypeList.h" 3132abc6edSZachary Turner #include "lldb/Target/ABI.h" 3206e827ccSJim Ingham #include "lldb/Target/RegisterContext.h" 3306e827ccSJim Ingham #include "lldb/Target/Thread.h" 3406e827ccSJim Ingham #include "lldb/Target/Process.h" 35d5944cd1SGreg Clayton #include "lldb/Target/SectionLoadList.h" 3606e827ccSJim Ingham #include "lldb/Target/Target.h" 3706e827ccSJim Ingham 3806e827ccSJim Ingham using namespace lldb; 3906e827ccSJim Ingham using namespace lldb_private; 4006e827ccSJim Ingham 418c9e5383SSean Callanan #pragma mark ThreadPlanTracer 428c9e5383SSean Callanan 4306e827ccSJim Ingham ThreadPlanTracer::ThreadPlanTracer (Thread &thread, lldb::StreamSP &stream_sp) : 4471c21d18SStephen Wilson m_thread (thread), 4506e827ccSJim Ingham m_single_step(true), 4606e827ccSJim Ingham m_enabled (false), 4706e827ccSJim Ingham m_stream_sp (stream_sp) 4806e827ccSJim Ingham { 4906e827ccSJim Ingham } 5006e827ccSJim Ingham 5106e827ccSJim Ingham ThreadPlanTracer::ThreadPlanTracer (Thread &thread) : 5271c21d18SStephen Wilson m_thread (thread), 5306e827ccSJim Ingham m_single_step(true), 5406e827ccSJim Ingham m_enabled (false), 5506e827ccSJim Ingham m_stream_sp () 5606e827ccSJim Ingham { 5706e827ccSJim Ingham } 5806e827ccSJim Ingham 5906e827ccSJim Ingham Stream * 6006e827ccSJim Ingham ThreadPlanTracer::GetLogStream () 6106e827ccSJim Ingham { 6206e827ccSJim Ingham 6306e827ccSJim Ingham if (m_stream_sp.get()) 6406e827ccSJim Ingham return m_stream_sp.get(); 6506e827ccSJim Ingham else 661ac04c30SGreg Clayton { 671ac04c30SGreg Clayton TargetSP target_sp (m_thread.CalculateTarget()); 681ac04c30SGreg Clayton if (target_sp) 6944d93782SGreg Clayton return target_sp->GetDebugger().GetOutputFile().get(); 701ac04c30SGreg Clayton } 711ac04c30SGreg Clayton return NULL; 7206e827ccSJim Ingham } 7306e827ccSJim Ingham 7406e827ccSJim Ingham void 7506e827ccSJim Ingham ThreadPlanTracer::Log() 7606e827ccSJim Ingham { 7706e827ccSJim Ingham SymbolContext sc; 7806e827ccSJim Ingham bool show_frame_index = false; 7906e827ccSJim Ingham bool show_fullpaths = false; 8006e827ccSJim Ingham 8115356e7fSCaroline Tice Stream *stream = GetLogStream(); 8228eb5711SJim Ingham if (stream) 8328eb5711SJim Ingham { 8415356e7fSCaroline Tice m_thread.GetStackFrameAtIndex(0)->Dump (stream, show_frame_index, show_fullpaths); 8515356e7fSCaroline Tice stream->Printf("\n"); 8615356e7fSCaroline Tice stream->Flush(); 8728eb5711SJim Ingham } 8815356e7fSCaroline Tice 8906e827ccSJim Ingham } 9006e827ccSJim Ingham 9106e827ccSJim Ingham bool 9206e827ccSJim Ingham ThreadPlanTracer::TracerExplainsStop () 9306e827ccSJim Ingham { 9406e827ccSJim Ingham if (m_enabled && m_single_step) 9506e827ccSJim Ingham { 9606e827ccSJim Ingham lldb::StopInfoSP stop_info = m_thread.GetStopInfo(); 9706e827ccSJim Ingham if (stop_info->GetStopReason() == eStopReasonTrace) 9806e827ccSJim Ingham return true; 9906e827ccSJim Ingham else 10006e827ccSJim Ingham return false; 10106e827ccSJim Ingham } 10206e827ccSJim Ingham else 10306e827ccSJim Ingham return false; 10406e827ccSJim Ingham } 1058c9e5383SSean Callanan 1068c9e5383SSean Callanan #pragma mark ThreadPlanAssemblyTracer 1078c9e5383SSean Callanan 1088c9e5383SSean Callanan ThreadPlanAssemblyTracer::ThreadPlanAssemblyTracer (Thread &thread, lldb::StreamSP &stream_sp) : 1098c9e5383SSean Callanan ThreadPlanTracer (thread, stream_sp), 1107e6d4e5aSSean Callanan m_disassembler_sp (), 1117e9b1fd0SGreg Clayton m_intptr_type (), 1127e9b1fd0SGreg Clayton m_register_values () 1138c9e5383SSean Callanan { 114a80ef359SJim Ingham } 115a80ef359SJim Ingham 116a80ef359SJim Ingham ThreadPlanAssemblyTracer::ThreadPlanAssemblyTracer (Thread &thread) : 117a80ef359SJim Ingham ThreadPlanTracer (thread), 1187e6d4e5aSSean Callanan m_disassembler_sp (), 1197e9b1fd0SGreg Clayton m_intptr_type (), 1207e9b1fd0SGreg Clayton m_register_values () 121a80ef359SJim Ingham { 122a80ef359SJim Ingham } 123a80ef359SJim Ingham 1247e9b1fd0SGreg Clayton Disassembler * 1257e9b1fd0SGreg Clayton ThreadPlanAssemblyTracer::GetDisassembler () 126a80ef359SJim Ingham { 1277e6d4e5aSSean Callanan if (m_disassembler_sp.get() == NULL) 1280f063ba6SJim Ingham m_disassembler_sp = Disassembler::FindPlugin(m_thread.GetProcess()->GetTarget().GetArchitecture(), NULL, NULL); 1297e6d4e5aSSean Callanan return m_disassembler_sp.get(); 1307e9b1fd0SGreg Clayton } 1318c9e5383SSean Callanan 1327e9b1fd0SGreg Clayton TypeFromUser 1337e9b1fd0SGreg Clayton ThreadPlanAssemblyTracer::GetIntPointerType() 1347e9b1fd0SGreg Clayton { 1357e9b1fd0SGreg Clayton if (!m_intptr_type.IsValid ()) 1367e9b1fd0SGreg Clayton { 1371ac04c30SGreg Clayton TargetSP target_sp (m_thread.CalculateTarget()); 1381ac04c30SGreg Clayton if (target_sp) 1391ac04c30SGreg Clayton { 1401ac04c30SGreg Clayton Module *exe_module = target_sp->GetExecutableModulePointer(); 1418c9e5383SSean Callanan 142aa149cbdSGreg Clayton if (exe_module) 1438c9e5383SSean Callanan { 14457ee3067SGreg Clayton m_intptr_type = TypeFromUser(exe_module->GetClangASTContext().GetBuiltinTypeForEncodingAndBitSize(eEncodingUint, target_sp->GetArchitecture().GetAddressByteSize() * 8)); 1458c9e5383SSean Callanan } 1468c9e5383SSean Callanan } 1471ac04c30SGreg Clayton } 1487e9b1fd0SGreg Clayton return m_intptr_type; 1497e9b1fd0SGreg Clayton } 1507e9b1fd0SGreg Clayton 1517e9b1fd0SGreg Clayton 1528c9e5383SSean Callanan 1538c9e5383SSean Callanan ThreadPlanAssemblyTracer::~ThreadPlanAssemblyTracer() 1548c9e5383SSean Callanan { 1558c9e5383SSean Callanan } 1568c9e5383SSean Callanan 1575ccbd294SGreg Clayton void 1585ccbd294SGreg Clayton ThreadPlanAssemblyTracer::TracingStarted () 1598c9e5383SSean Callanan { 1605ccbd294SGreg Clayton RegisterContext *reg_ctx = m_thread.GetRegisterContext().get(); 1618c9e5383SSean Callanan 1628c9e5383SSean Callanan if (m_register_values.size() == 0) 163cd482e35SGreg Clayton m_register_values.resize (reg_ctx->GetRegisterCount()); 1648c9e5383SSean Callanan } 1658c9e5383SSean Callanan 1665ccbd294SGreg Clayton void 1675ccbd294SGreg Clayton ThreadPlanAssemblyTracer::TracingEnded () 1688c9e5383SSean Callanan { 169cd482e35SGreg Clayton m_register_values.clear(); 1708c9e5383SSean Callanan } 1718c9e5383SSean Callanan 1725ccbd294SGreg Clayton void 1735ccbd294SGreg Clayton ThreadPlanAssemblyTracer::Log () 1748c9e5383SSean Callanan { 1758c9e5383SSean Callanan Stream *stream = GetLogStream (); 1768c9e5383SSean Callanan 1778c9e5383SSean Callanan if (!stream) 1788c9e5383SSean Callanan return; 1798c9e5383SSean Callanan 1805ccbd294SGreg Clayton RegisterContext *reg_ctx = m_thread.GetRegisterContext().get(); 1818c9e5383SSean Callanan 1828c9e5383SSean Callanan lldb::addr_t pc = reg_ctx->GetPC(); 1831ac04c30SGreg Clayton ProcessSP process_sp (m_thread.GetProcess()); 1848c9e5383SSean Callanan Address pc_addr; 1858c9e5383SSean Callanan bool addr_valid = false; 1867e9b1fd0SGreg Clayton uint8_t buffer[16] = {0}; // Must be big enough for any single instruction 1871ac04c30SGreg Clayton addr_valid = process_sp->GetTarget().GetSectionLoadList().ResolveLoadAddress (pc, pc_addr); 1888c9e5383SSean Callanan 189cd482e35SGreg Clayton pc_addr.Dump(stream, &m_thread, Address::DumpStyleResolvedDescription, Address::DumpStyleModuleWithFileAddress); 1906cffdd2fSJim Ingham stream->PutCString (" "); 1918c9e5383SSean Callanan 1927e9b1fd0SGreg Clayton Disassembler *disassembler = GetDisassembler(); 1937e9b1fd0SGreg Clayton if (disassembler) 1948c9e5383SSean Callanan { 1958c9e5383SSean Callanan Error err; 1961ac04c30SGreg Clayton process_sp->ReadMemory(pc, buffer, sizeof(buffer), err); 1978c9e5383SSean Callanan 1988c9e5383SSean Callanan if (err.Success()) 1998c9e5383SSean Callanan { 2007e9b1fd0SGreg Clayton DataExtractor extractor(buffer, sizeof(buffer), 2011ac04c30SGreg Clayton process_sp->GetByteOrder(), 2021ac04c30SGreg Clayton process_sp->GetAddressByteSize()); 2038c9e5383SSean Callanan 2043faf47c4SGreg Clayton bool data_from_file = false; 2058c9e5383SSean Callanan if (addr_valid) 2063faf47c4SGreg Clayton disassembler->DecodeInstructions (pc_addr, extractor, 0, 1, false, data_from_file); 2078c9e5383SSean Callanan else 2083faf47c4SGreg Clayton disassembler->DecodeInstructions (Address (pc), extractor, 0, 1, false, data_from_file); 2098c9e5383SSean Callanan 2107e9b1fd0SGreg Clayton InstructionList &instruction_list = disassembler->GetInstructionList(); 211357132ebSGreg Clayton const uint32_t max_opcode_byte_size = instruction_list.GetMaxOpcocdeByteSize(); 2128c9e5383SSean Callanan 2138c9e5383SSean Callanan if (instruction_list.GetSize()) 2148c9e5383SSean Callanan { 215cd482e35SGreg Clayton const bool show_bytes = true; 216cd482e35SGreg Clayton const bool show_address = true; 2178c9e5383SSean Callanan Instruction *instruction = instruction_list.GetInstructionAtIndex(0).get(); 218554f68d3SGreg Clayton const FormatEntity::Entry *disassemble_format = m_thread.GetProcess()->GetTarget().GetDebugger().GetDisassemblyFormat(); 219cd482e35SGreg Clayton instruction->Dump (stream, 220357132ebSGreg Clayton max_opcode_byte_size, 221cd482e35SGreg Clayton show_address, 222cd482e35SGreg Clayton show_bytes, 223aff1b357SJason Molenda NULL, 224aff1b357SJason Molenda NULL, 225aff1b357SJason Molenda NULL, 226c980fa92SJason Molenda disassemble_format, 227c980fa92SJason Molenda 0); 2288c9e5383SSean Callanan } 2298c9e5383SSean Callanan } 2308c9e5383SSean Callanan } 2318c9e5383SSean Callanan 2321ac04c30SGreg Clayton const ABI *abi = process_sp->GetABI().get(); 2337e9b1fd0SGreg Clayton TypeFromUser intptr_type = GetIntPointerType(); 2347e9b1fd0SGreg Clayton 2357e9b1fd0SGreg Clayton if (abi && intptr_type.IsValid()) 2368c9e5383SSean Callanan { 2378c9e5383SSean Callanan ValueList value_list; 2388c9e5383SSean Callanan const int num_args = 1; 2398c9e5383SSean Callanan 2408c9e5383SSean Callanan for (int arg_index = 0; arg_index < num_args; ++arg_index) 2418c9e5383SSean Callanan { 2428c9e5383SSean Callanan Value value; 2438c9e5383SSean Callanan value.SetValueType (Value::eValueTypeScalar); 24457ee3067SGreg Clayton // value.SetContext (Value::eContextTypeClangType, intptr_type.GetOpaqueQualType()); 24557ee3067SGreg Clayton value.SetClangType (intptr_type); 2468c9e5383SSean Callanan value_list.PushValue (value); 2478c9e5383SSean Callanan } 2488c9e5383SSean Callanan 2497e9b1fd0SGreg Clayton if (abi->GetArgumentValues (m_thread, value_list)) 2508c9e5383SSean Callanan { 2518c9e5383SSean Callanan for (int arg_index = 0; arg_index < num_args; ++arg_index) 2528c9e5383SSean Callanan { 253cd482e35SGreg Clayton stream->Printf("\n\targ[%d]=%llx", arg_index, value_list.GetValueAtIndex(arg_index)->GetScalar().ULongLong()); 2548c9e5383SSean Callanan 2558c9e5383SSean Callanan if (arg_index + 1 < num_args) 256cd482e35SGreg Clayton stream->PutCString (", "); 2578c9e5383SSean Callanan } 2588c9e5383SSean Callanan } 2598c9e5383SSean Callanan } 2608c9e5383SSean Callanan 2618c9e5383SSean Callanan 262cd482e35SGreg Clayton RegisterValue reg_value; 263cd482e35SGreg Clayton for (uint32_t reg_num = 0, num_registers = reg_ctx->GetRegisterCount(); 264cd482e35SGreg Clayton reg_num < num_registers; 265cd482e35SGreg Clayton ++reg_num) 2668c9e5383SSean Callanan { 267cd482e35SGreg Clayton const RegisterInfo *reg_info = reg_ctx->GetRegisterInfoAtIndex(reg_num); 268cd482e35SGreg Clayton if (reg_ctx->ReadRegister (reg_info, reg_value)) 2698c9e5383SSean Callanan { 270cd482e35SGreg Clayton assert (reg_num < m_register_values.size()); 271cd482e35SGreg Clayton if (m_register_values[reg_num].GetType() == RegisterValue::eTypeInvalid || 272cd482e35SGreg Clayton reg_value != m_register_values[reg_num]) 273cd482e35SGreg Clayton { 274cd482e35SGreg Clayton if (reg_value.GetType() != RegisterValue::eTypeInvalid) 275cd482e35SGreg Clayton { 276cd482e35SGreg Clayton stream->PutCString ("\n\t"); 277cd482e35SGreg Clayton reg_value.Dump(stream, reg_info, true, false, eFormatDefault); 2788c9e5383SSean Callanan } 2798c9e5383SSean Callanan } 280cd482e35SGreg Clayton m_register_values[reg_num] = reg_value; 281cd482e35SGreg Clayton } 282cd482e35SGreg Clayton } 283cd482e35SGreg Clayton stream->EOL(); 28415356e7fSCaroline Tice stream->Flush(); 2858c9e5383SSean Callanan } 286